@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,136 @@
1
+ "use client";
2
+
3
+ import { zodResolver } from "@hookform/resolvers/zod";
4
+ import { useTranslations } from "next-intl";
5
+ import Image from "next/image";
6
+ import { useEffect, useState } from "react";
7
+ import { SubmitHandler, useForm } from "react-hook-form";
8
+ import { toast } from "sonner";
9
+ import { z } from "zod";
10
+ import { errorToast, FormPassword } from "../../../../components";
11
+ import { Button, CardContent, CardDescription, CardHeader, CardTitle, Form } from "../../../../shadcnui";
12
+ import { useAuthContext } from "../../contexts";
13
+ import { AuthService } from "../../data";
14
+ import { AuthComponent } from "../../enums";
15
+
16
+ export function AcceptInvitation() {
17
+ const { setComponentType, params, setParams } = useAuthContext();
18
+ const [showConfirmation, setShowConfirmation] = useState<boolean>(false);
19
+ const [error, setError] = useState<string | undefined>(undefined);
20
+ const t = useTranslations();
21
+
22
+ useEffect(() => {
23
+ async function validateCode(code: string) {
24
+ try {
25
+ const payload: any = {
26
+ code: code,
27
+ };
28
+
29
+ await AuthService.validateCode(payload);
30
+ } catch (e) {
31
+ setError(e instanceof Error ? e.message : String(e));
32
+ errorToast({ title: t(`generic.errors.error`), error: e });
33
+ }
34
+ }
35
+
36
+ if (params && params.code) {
37
+ validateCode(params.code);
38
+ } else {
39
+ setError(t(`foundations.auth.errors.invalid_invitation_code`));
40
+ }
41
+ }, []);
42
+
43
+ const formSchema = z
44
+ .object({
45
+ password: z.string().min(1, {
46
+ message: t(`foundations.user.fields.password.error`),
47
+ }),
48
+ passwordRetype: z.string().min(1, {
49
+ message: t("foundations.auth.errors.password_retype_required"),
50
+ }),
51
+ })
52
+ .refine((data) => data.password === data.passwordRetype, {
53
+ message: t("foundations.auth.fields.retype_password.error_not_match"),
54
+ path: ["passwordRetype"],
55
+ });
56
+
57
+ const form = useForm<z.infer<typeof formSchema>>({
58
+ resolver: zodResolver(formSchema),
59
+ defaultValues: {
60
+ password: "",
61
+ passwordRetype: "",
62
+ },
63
+ });
64
+
65
+ const onSubmit: SubmitHandler<z.infer<typeof formSchema>> = async (values: z.infer<typeof formSchema>) => {
66
+ try {
67
+ if (!params?.code) return;
68
+
69
+ const payload = {
70
+ code: params?.code,
71
+ password: values.password,
72
+ };
73
+
74
+ await AuthService.acceptInvitation(payload);
75
+ setShowConfirmation(true);
76
+
77
+ toast.success(t("foundations.auth.account_activated"), {
78
+ description: t("foundations.auth.account_activated_description"),
79
+ });
80
+
81
+ setTimeout(() => {
82
+ setComponentType(AuthComponent.Login);
83
+ setParams(undefined);
84
+ }, 2000);
85
+ } catch (e) {
86
+ errorToast({ title: t(`generic.errors.error`), error });
87
+ }
88
+ };
89
+
90
+ return (
91
+ <>
92
+ <CardHeader>
93
+ <CardTitle className="text-primary flex flex-col items-center pb-10 text-4xl">
94
+ <Image src="/phlow-logo.webp" alt="Phlow" width={100} height={100} priority />
95
+ {t("foundations.auth.accept_invitation")}
96
+ </CardTitle>
97
+ <CardDescription className="text-center text-sm">
98
+ {error ? (
99
+ <>{t("foundations.auth.errors.activating_account")}</>
100
+ ) : (
101
+ <>{t("foundations.auth.select_password")}</>
102
+ )}
103
+ </CardDescription>
104
+ </CardHeader>
105
+ <CardContent>
106
+ {showConfirmation ? (
107
+ <CardDescription className="text-center text-xl">
108
+ {t("foundations.auth.activation_description")}
109
+ </CardDescription>
110
+ ) : error ? (
111
+ <CardDescription className="text-center text-xl">{error}</CardDescription>
112
+ ) : (
113
+ <Form {...form}>
114
+ <form onSubmit={form.handleSubmit(onSubmit)}>
115
+ <FormPassword
116
+ form={form}
117
+ id="password"
118
+ name={t(`foundations.user.fields.password.label`)}
119
+ placeholder={t(`foundations.user.fields.password.placeholder`)}
120
+ />
121
+ <FormPassword
122
+ form={form}
123
+ id="passwordRetype"
124
+ name={t("foundations.auth.fields.retype_password.label")}
125
+ placeholder={t(`foundations.auth.fields.retype_password.placeholder`)}
126
+ />
127
+ <Button className="mt-4 w-full" type={"submit"}>
128
+ {t("foundations.auth.accept_invitation")}
129
+ </Button>
130
+ </form>
131
+ </Form>
132
+ )}
133
+ </CardContent>
134
+ </>
135
+ );
136
+ }
@@ -0,0 +1,75 @@
1
+ "use client";
2
+
3
+ import { useTranslations } from "next-intl";
4
+ import Image from "next/image";
5
+ import { useEffect, useState } from "react";
6
+ import { toast } from "sonner";
7
+ import { errorToast } from "../../../../components";
8
+ import { CardContent, CardDescription, CardHeader, CardTitle } from "../../../../shadcnui";
9
+ import { useAuthContext } from "../../contexts";
10
+ import { AuthService } from "../../data";
11
+ import { AuthComponent } from "../../enums";
12
+
13
+ export function ActivateAccount() {
14
+ const { setComponentType, params, setParams } = useAuthContext();
15
+ const [showConfirmation, setShowConfirmation] = useState<boolean>(false);
16
+ const [error, setError] = useState<string | undefined>(undefined);
17
+ const t = useTranslations();
18
+
19
+ useEffect(() => {
20
+ async function ActivateAccount(code: string) {
21
+ try {
22
+ const payload = {
23
+ activationCode: code,
24
+ };
25
+
26
+ await AuthService.activate(payload);
27
+ setShowConfirmation(true);
28
+
29
+ setParams(undefined);
30
+
31
+ toast.success(t("foundations.auth.account_activated"), {
32
+ description: t("foundations.auth.account_activated_description"),
33
+ });
34
+
35
+ setTimeout(() => {
36
+ setComponentType(AuthComponent.Login);
37
+ }, 2000);
38
+ } catch (e) {
39
+ setError(e instanceof Error ? e.message : String(e));
40
+ errorToast({ title: t(`generic.errors.error`), error: e });
41
+ }
42
+ }
43
+
44
+ if (params && params.code) {
45
+ ActivateAccount(params.code);
46
+ } else {
47
+ setError(t(`foundations.auth.errors.invalid_invitation_code`));
48
+ }
49
+ }, []);
50
+
51
+ return (
52
+ <>
53
+ <CardHeader>
54
+ <CardTitle className="text-primary flex flex-col items-center pb-10 text-4xl">
55
+ <Image src="/phlow-logo.webp" alt="Phlow" width={100} height={100} priority />
56
+ {t("foundations.auth.accept_invitation")}
57
+ </CardTitle>
58
+ <CardDescription className="text-center text-sm">
59
+ {error ? <>{t("foundations.auth.errors.activating_account")}</> : <> </>}
60
+ </CardDescription>
61
+ </CardHeader>
62
+ <CardContent>
63
+ <CardDescription className="text-center text-xl">
64
+ {showConfirmation ? (
65
+ <>{t("foundations.auth.activation_description")}</>
66
+ ) : error ? (
67
+ <>{error}</>
68
+ ) : (
69
+ <>{t("foundations.auth.activation_wait")}</>
70
+ )}
71
+ </CardDescription>
72
+ </CardContent>
73
+ </>
74
+ );
75
+ }
@@ -0,0 +1,32 @@
1
+ "use client";
2
+
3
+ import { useEffect, useState } from "react";
4
+ import { JsonApiHydratedDataInterface, Modules, rehydrate } from "../../../../core";
5
+ import { useI18nRouter } from "../../../../i18n";
6
+ import { UserInterface } from "../../../user";
7
+ import { useCurrentUserContext } from "../../../user/contexts";
8
+ import { AuthInterface, AuthService } from "../../data";
9
+
10
+ export function Cookies({ dehydratedAuth, page }: { dehydratedAuth: JsonApiHydratedDataInterface; page?: string }) {
11
+ const { setUser } = useCurrentUserContext<UserInterface>();
12
+ const router = useI18nRouter();
13
+ const [hasSaved, setHasSaved] = useState(false);
14
+
15
+ useEffect(() => {
16
+ if (hasSaved) return;
17
+ async function saveTokenOnServer() {
18
+ await AuthService.saveToken({ dehydratedAuth });
19
+ const auth: AuthInterface = rehydrate(Modules.Auth, dehydratedAuth) as AuthInterface;
20
+ setUser(auth.user as UserInterface);
21
+ setHasSaved(true);
22
+
23
+ if (page) {
24
+ if (page.startsWith("/")) router.push(page ?? "/");
25
+ window.location.href = page;
26
+ }
27
+ }
28
+ saveTokenOnServer();
29
+ }, [dehydratedAuth, setUser, hasSaved, router]);
30
+
31
+ return null;
32
+ }
@@ -0,0 +1,108 @@
1
+ "use client";
2
+
3
+ import { zodResolver } from "@hookform/resolvers/zod";
4
+ import { useTranslations } from "next-intl";
5
+ import Image from "next/image";
6
+ import { useState } from "react";
7
+ import { SubmitHandler, useForm } from "react-hook-form";
8
+ import { z } from "zod";
9
+ import { errorToast, FormInput } from "../../../../components";
10
+ import {
11
+ Button,
12
+ CardContent,
13
+ CardDescription,
14
+ CardFooter,
15
+ CardHeader,
16
+ CardTitle,
17
+ Form,
18
+ Link,
19
+ } from "../../../../shadcnui";
20
+ import { useAuthContext } from "../../contexts";
21
+ import { AuthService } from "../../data";
22
+ import { AuthComponent } from "../../enums";
23
+
24
+ export function ForgotPassword() {
25
+ const t = useTranslations();
26
+ const { setComponentType } = useAuthContext();
27
+
28
+ const [showConfirmation, setShowConfirmation] = useState<boolean>(false);
29
+
30
+ const formSchema = z.object({
31
+ email: z.string().email({
32
+ message: t(`generic.errors.invalid_email`),
33
+ }),
34
+ });
35
+
36
+ const form = useForm<z.infer<typeof formSchema>>({
37
+ resolver: zodResolver(formSchema),
38
+ defaultValues: {
39
+ email: "",
40
+ },
41
+ });
42
+
43
+ const onSubmit: SubmitHandler<z.infer<typeof formSchema>> = async (values: z.infer<typeof formSchema>) => {
44
+ try {
45
+ const payload = {
46
+ email: values.email,
47
+ };
48
+
49
+ await AuthService.initialiseForgotPassword(payload);
50
+ setShowConfirmation(true);
51
+ } catch (e) {
52
+ errorToast({ error: e });
53
+ }
54
+ };
55
+
56
+ return (
57
+ <>
58
+ <CardHeader data-testid="page-forgot-password-container">
59
+ <CardTitle className="text-primary flex flex-col items-center pb-10 text-4xl">
60
+ <Image src="/phlow-logo.webp" alt="Phlows" width={100} height={100} priority />
61
+ {t("foundations.auth.forgot_password")}
62
+ </CardTitle>
63
+ <CardDescription className="text-sm">
64
+ {showConfirmation ? <> </> : <>{t(`foundations.auth.add_email_to_reset`)}</>}
65
+ </CardDescription>
66
+ </CardHeader>
67
+ {showConfirmation ? (
68
+ <CardContent>
69
+ <CardDescription className="text-center text-xl">{t(`foundations.auth.reset_confirmation`)}</CardDescription>
70
+ </CardContent>
71
+ ) : (
72
+ <Form {...form}>
73
+ <form onSubmit={form.handleSubmit(onSubmit)}>
74
+ <CardContent>
75
+ <FormInput
76
+ form={form}
77
+ id="email"
78
+ name={t(`generic.fields.email.label`)}
79
+ placeholder={t(`generic.fields.email.placeholder`)}
80
+ testId="form-forgot-password-input-email"
81
+ />
82
+ <Button className="mt-4 w-full" type={"submit"} data-testid="form-forgot-password-button-reset">
83
+ {t(`foundations.auth.buttons.reset_password`)}
84
+ </Button>
85
+ </CardContent>
86
+ <CardFooter className="flex w-full flex-row justify-between">
87
+ {/* <Link
88
+ href="#"
89
+ className="flex w-full justify-start"
90
+ onClick={() => setComponentType(AuthComponent.Register)}
91
+ >
92
+ {t("foundations.auth.buttons.register")}
93
+ </Link> */}
94
+ <Link
95
+ href="#"
96
+ className="flex w-full justify-end"
97
+ onClick={() => setComponentType(AuthComponent.Login)}
98
+ data-testid="form-forgot-password-link-login"
99
+ >
100
+ {t(`foundations.auth.buttons.login`)}
101
+ </Link>
102
+ </CardFooter>
103
+ </form>
104
+ </Form>
105
+ )}
106
+ </>
107
+ );
108
+ }
@@ -0,0 +1,118 @@
1
+ "use client";
2
+
3
+ import { zodResolver } from "@hookform/resolvers/zod";
4
+ import { useTranslations } from "next-intl";
5
+ import Image from "next/image";
6
+ import { SubmitHandler, useForm } from "react-hook-form";
7
+ import { z } from "zod";
8
+ import { errorToast, FormInput, FormPassword } from "../../../../components";
9
+ import { useI18nRouter, usePageUrlGenerator } from "../../../../hooks";
10
+ import {
11
+ Button,
12
+ CardContent,
13
+ CardDescription,
14
+ CardFooter,
15
+ CardHeader,
16
+ CardTitle,
17
+ Form,
18
+ Link,
19
+ } from "../../../../shadcnui";
20
+ import { UserInterface } from "../../../user";
21
+ import { useCurrentUserContext } from "../../../user/contexts";
22
+ import { useAuthContext } from "../../contexts";
23
+ import { AuthService } from "../../data";
24
+ import { AuthComponent } from "../../enums";
25
+
26
+ export function Login() {
27
+ const t = useTranslations();
28
+ const { setUser } = useCurrentUserContext<UserInterface>();
29
+ const { setComponentType } = useAuthContext();
30
+ const generateUrl = usePageUrlGenerator();
31
+ const router = useI18nRouter();
32
+
33
+ const formSchema = z.object({
34
+ email: z.string().email({
35
+ message: t(`generic.errors.invalid_email`),
36
+ }),
37
+ password: z.string().min(3, { message: t(`foundations.auth.errors.password_too_short`) }),
38
+ });
39
+
40
+ const form = useForm<z.infer<typeof formSchema>>({
41
+ resolver: zodResolver(formSchema),
42
+ defaultValues: {
43
+ email: "",
44
+ password: "",
45
+ },
46
+ });
47
+
48
+ const onSubmit: SubmitHandler<z.infer<typeof formSchema>> = async (values: z.infer<typeof formSchema>) => {
49
+ try {
50
+ const user: UserInterface = (await AuthService.login({
51
+ email: values.email,
52
+ password: values.password,
53
+ })) as UserInterface;
54
+
55
+ setUser(user);
56
+ router.replace(generateUrl({ page: `/` }));
57
+ } catch (e) {
58
+ errorToast({
59
+ title: t(`generic.errors.error`),
60
+ error: e,
61
+ });
62
+ }
63
+ };
64
+
65
+ return (
66
+ <>
67
+ <CardHeader data-testid="page-login-container">
68
+ <CardTitle className="text-primary flex flex-col items-center pb-10 text-4xl">
69
+ <Image src="/phlow-logo.webp" alt="Phlow" width={100} height={100} priority />
70
+ {t("foundations.auth.login")}
71
+ </CardTitle>
72
+
73
+ <CardDescription className="text-sm">{t(`foundations.auth.login_description`)}</CardDescription>
74
+ </CardHeader>
75
+ <Form {...form}>
76
+ <form onSubmit={form.handleSubmit(onSubmit)}>
77
+ <CardContent>
78
+ <FormInput
79
+ autoFocus
80
+ form={form}
81
+ id="email"
82
+ name={t(`generic.fields.email.label`)}
83
+ placeholder={t(`generic.fields.email.placeholder`)}
84
+ testId="form-login-input-email"
85
+ />
86
+ <FormPassword
87
+ form={form}
88
+ id="password"
89
+ name={t(`foundations.user.fields.password.label`)}
90
+ placeholder={t(`foundations.user.fields.password.placeholder`)}
91
+ testId="form-login-input-password"
92
+ />
93
+ <Button className="mt-4 w-full" type={"submit"} data-testid="form-login-button-submit">
94
+ {t(`foundations.auth.login`)}
95
+ </Button>
96
+ </CardContent>
97
+ <CardFooter className="flex w-full flex-row justify-between">
98
+ <Link
99
+ href="#"
100
+ className="flex w-full justify-start"
101
+ onClick={() => setComponentType(AuthComponent.Register)}
102
+ >
103
+ {t(`foundations.auth.register`)}
104
+ </Link>
105
+ <Link
106
+ href="#"
107
+ className="flex w-full justify-end"
108
+ onClick={() => setComponentType(AuthComponent.ForgotPassword)}
109
+ data-testid="form-login-link-forgot-password"
110
+ >
111
+ {t(`foundations.auth.forgot_password`)}
112
+ </Link>
113
+ </CardFooter>
114
+ </form>
115
+ </Form>
116
+ </>
117
+ );
118
+ }
@@ -0,0 +1,19 @@
1
+ "use client";
2
+
3
+ import { useEffect } from "react";
4
+ import { usePageUrlGenerator } from "../../../../hooks";
5
+ import { AuthService } from "../../data";
6
+
7
+ export function Logout() {
8
+ const generateUrl = usePageUrlGenerator();
9
+
10
+ useEffect(() => {
11
+ const logOut = async () => {
12
+ await AuthService.logout();
13
+ window.location.href = generateUrl({ page: `/` });
14
+ };
15
+ logOut();
16
+ }, []);
17
+
18
+ return <></>;
19
+ }
@@ -0,0 +1,39 @@
1
+ "use client";
2
+
3
+ import { deleteCookie, getCookie } from "cookies-next";
4
+ import { useEffect } from "react";
5
+ import { UserInterface, UserService } from "../../../user";
6
+ import { useCurrentUserContext } from "../../../user/contexts";
7
+ import { getTokenHandler } from "../../config";
8
+
9
+ export function RefreshUser() {
10
+ const { setUser } = useCurrentUserContext<UserInterface>();
11
+
12
+ const loadFullUser = async () => {
13
+ const fullUser = await UserService.findFullUser();
14
+
15
+ if (fullUser) {
16
+ setUser(fullUser);
17
+ const token = {
18
+ userId: fullUser.id,
19
+ companyId: fullUser.company?.id,
20
+ licenseExpirationDate: fullUser.company?.licenseExpirationDate,
21
+ roles: fullUser.roles.map((role) => role.id),
22
+ features: fullUser.company?.features?.map((feature) => feature.id) ?? [],
23
+ modules: fullUser.modules.map((module) => {
24
+ return { id: module.id, permissions: module.permissions };
25
+ }),
26
+ };
27
+
28
+ await getTokenHandler()?.updateToken(token);
29
+ deleteCookie("reloadData");
30
+ }
31
+ };
32
+
33
+ useEffect(() => {
34
+ const reloadData = getCookie("reloadData");
35
+ if (reloadData !== undefined) loadFullUser();
36
+ }, []);
37
+
38
+ return null;
39
+ }
@@ -0,0 +1,150 @@
1
+ "use client";
2
+
3
+ import { zodResolver } from "@hookform/resolvers/zod";
4
+ import { useTranslations } from "next-intl";
5
+ import Image from "next/image";
6
+ import { useState } from "react";
7
+ import { SubmitHandler, useForm } from "react-hook-form";
8
+ import { v4 } from "uuid";
9
+ import { z } from "zod";
10
+ import { errorToast, FormInput, FormPassword } from "../../../../components";
11
+ import {
12
+ Button,
13
+ CardContent,
14
+ CardDescription,
15
+ CardFooter,
16
+ CardHeader,
17
+ CardTitle,
18
+ Form,
19
+ Link,
20
+ } from "../../../../shadcnui";
21
+ import { useAuthContext } from "../../contexts";
22
+ import { AuthService } from "../../data";
23
+ import { AuthComponent } from "../../enums";
24
+
25
+ export default function Register() {
26
+ const t = useTranslations();
27
+ const { setComponentType } = useAuthContext();
28
+
29
+ const [showConfirmation, setShowConfirmation] = useState<boolean>(false);
30
+
31
+ const formSchema = z.object({
32
+ company: z.string().min(1, {
33
+ message: t(`generic.errors.missing_company_name`),
34
+ }),
35
+ name: z.string().min(1, {
36
+ message: t("generic.errors.missing_name"),
37
+ }),
38
+ email: z.string().email({
39
+ message: t(`generic.errors.invalid_email`),
40
+ }),
41
+ password: z
42
+ .string()
43
+ .min(8, t(`foundations.auth.errors.password_too_short`))
44
+ .regex(/^(?=.*[0-9])(?=.*[^a-zA-Z0-9]).*$/, {
45
+ message: t(`foundations.auth.errors.password_invalid_format`),
46
+ }),
47
+ });
48
+
49
+ const form = useForm<z.infer<typeof formSchema>>({
50
+ resolver: zodResolver(formSchema),
51
+ defaultValues: {
52
+ company: "",
53
+ name: "",
54
+ email: "",
55
+ password: "",
56
+ },
57
+ });
58
+
59
+ const onSubmit: SubmitHandler<z.infer<typeof formSchema>> = async (values: z.infer<typeof formSchema>) => {
60
+ try {
61
+ const payload = {
62
+ id: v4(),
63
+ companyName: values.company,
64
+ name: values.name,
65
+ email: values.email,
66
+ password: values.password,
67
+ };
68
+
69
+ await AuthService.register(payload);
70
+ setShowConfirmation(true);
71
+ } catch (e) {
72
+ errorToast({ error: e });
73
+ }
74
+ };
75
+
76
+ return (
77
+ <>
78
+ <CardHeader>
79
+ <CardTitle className="text-primary flex flex-col items-center pb-10 text-4xl">
80
+ <Image src="/phlow-logo.webp" alt="Phlow" width={100} height={100} priority />
81
+ {t(`foundations.auth.register`)}
82
+ </CardTitle>
83
+ <CardDescription className="text-sm">
84
+ {showConfirmation ? <> </> : <>{t(`foundations.auth.register_description`)}</>}
85
+ </CardDescription>
86
+ </CardHeader>
87
+ {showConfirmation ? (
88
+ <CardContent>
89
+ <CardDescription className="text-center text-xl">
90
+ {t("foundations.auth.register_confirmation")}
91
+ </CardDescription>
92
+ </CardContent>
93
+ ) : (
94
+ <Form {...form}>
95
+ <form onSubmit={form.handleSubmit(onSubmit)}>
96
+ <CardContent>
97
+ <FormInput
98
+ isRequired
99
+ form={form}
100
+ id="company"
101
+ name={t(`foundations.company.fields.name.label`)}
102
+ placeholder={t(`foundations.company.fields.name.placeholder`)}
103
+ />
104
+ <FormInput
105
+ isRequired
106
+ form={form}
107
+ id="name"
108
+ name={t(`foundations.user.fields.name.label`)}
109
+ placeholder={t(`foundations.user.fields.name.placeholder`)}
110
+ />
111
+ <FormInput
112
+ isRequired
113
+ form={form}
114
+ id="email"
115
+ name={t(`generic.fields.email.label`)}
116
+ placeholder={t(`generic.fields.email.placeholder`)}
117
+ />
118
+ <FormPassword
119
+ isRequired
120
+ form={form}
121
+ id="password"
122
+ name={t(`foundations.user.fields.password.label`)}
123
+ placeholder={t(`foundations.user.fields.password.placeholder`)}
124
+ />
125
+ <Button className="mt-4 w-full" type={"submit"}>
126
+ {t(`foundations.auth.buttons.register`)}
127
+ </Button>
128
+ </CardContent>
129
+ <CardFooter className="flex w-full flex-row justify-between">
130
+ <Link
131
+ href="#"
132
+ className="flex w-full justify-start"
133
+ onClick={() => setComponentType(AuthComponent.Login)}
134
+ >
135
+ {t(`foundations.auth.buttons.login`)}
136
+ </Link>
137
+ <Link
138
+ href="#"
139
+ className="flex w-full justify-end"
140
+ onClick={() => setComponentType(AuthComponent.ForgotPassword)}
141
+ >
142
+ {t(`foundations.auth.buttons.forgot_password`)}
143
+ </Link>
144
+ </CardFooter>
145
+ </form>
146
+ </Form>
147
+ )}
148
+ </>
149
+ );
150
+ }