@digitaldefiance/express-suite-react-components 2.9.37 → 2.9.38

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 (257) hide show
  1. package/LICENSE +21 -0
  2. package/package.json +4 -5
  3. package/src/auth/Private.tsx +17 -0
  4. package/src/auth/PrivateRoute.tsx +28 -0
  5. package/src/auth/UnAuth.tsx +16 -0
  6. package/src/auth/UnAuthRoute.tsx +30 -0
  7. package/src/auth/{index.d.ts → index.ts} +1 -2
  8. package/src/components/ApiAccess.tsx +174 -0
  9. package/src/components/BackupCodeLoginForm.tsx +488 -0
  10. package/src/components/BackupCodesForm.tsx +286 -0
  11. package/src/components/ChangePasswordForm.tsx +272 -0
  12. package/src/components/ConfirmationDialog.tsx +48 -0
  13. package/src/components/CurrencyCodeSelector.tsx +60 -0
  14. package/src/components/CurrencyInput.tsx +80 -0
  15. package/src/components/DashboardPage.tsx +24 -0
  16. package/src/components/DropdownMenu.tsx +92 -0
  17. package/src/components/ExpirationSecondsSelector.tsx +60 -0
  18. package/src/components/Flag.tsx +52 -0
  19. package/src/components/ForgotPasswordForm.tsx +173 -0
  20. package/src/components/LoginForm.tsx +455 -0
  21. package/src/components/LogoutPage.tsx +21 -0
  22. package/src/components/RegisterForm.tsx +602 -0
  23. package/src/components/ResetPasswordForm.tsx +246 -0
  24. package/src/components/SideMenu.tsx +46 -0
  25. package/src/components/SideMenuListItem.tsx +74 -0
  26. package/src/components/TopMenu.tsx +145 -0
  27. package/src/components/TranslatedTitle.tsx +29 -0
  28. package/src/components/UserLanguageSelector.tsx +45 -0
  29. package/src/components/UserMenu.tsx +15 -0
  30. package/src/components/UserSettingsForm.tsx +505 -0
  31. package/src/components/VerifyEmailPage.tsx +184 -0
  32. package/src/components/{index.d.ts → index.ts} +1 -1
  33. package/src/contexts/AuthProvider.spec.tsx +1195 -0
  34. package/src/contexts/AuthProvider.tsx +924 -0
  35. package/src/contexts/I18nProvider.tsx +114 -0
  36. package/src/contexts/MenuContext.tsx +398 -0
  37. package/src/contexts/SuiteConfigProvider.tsx +93 -0
  38. package/src/contexts/ThemeProvider.tsx +67 -0
  39. package/src/contexts/{index.d.ts → index.ts} +0 -1
  40. package/src/hooks/{index.d.ts → index.ts} +0 -1
  41. package/src/hooks/useBackupCodes.ts +105 -0
  42. package/src/hooks/useEmailVerification.ts +49 -0
  43. package/src/hooks/useExpiringValue.ts +78 -0
  44. package/src/hooks/useLocalStorage.ts +18 -0
  45. package/src/hooks/useUserSettings.ts +269 -0
  46. package/src/{index.d.ts → index.ts} +1 -1
  47. package/src/interfaces/IAppConfig.ts +5 -0
  48. package/src/interfaces/IMenuConfig.ts +11 -0
  49. package/src/interfaces/IMenuOption.ts +55 -0
  50. package/src/interfaces/index.ts +3 -0
  51. package/src/services/__mocks__/authService.ts +14 -0
  52. package/src/services/api.ts +13 -0
  53. package/src/services/authService.ts +500 -0
  54. package/src/services/authenticatedApi.ts +17 -0
  55. package/src/services/index.ts +3 -0
  56. package/src/types/MenuType.ts +15 -0
  57. package/src/types/expirationSeconds.ts +18 -0
  58. package/src/types/index.ts +1 -0
  59. package/src/types/translation.ts +20 -0
  60. package/src/wrappers/BackupCodeLoginWrapper.tsx +34 -0
  61. package/src/wrappers/BackupCodesWrapper.tsx +28 -0
  62. package/src/wrappers/ChangePasswordFormWrapper.tsx +34 -0
  63. package/src/wrappers/LoginFormWrapper.tsx +59 -0
  64. package/src/wrappers/LogoutPageWrapper.tsx +30 -0
  65. package/src/wrappers/RegisterFormWrapper.tsx +61 -0
  66. package/src/wrappers/UserSettingsFormWrapper.tsx +39 -0
  67. package/src/wrappers/VerifyEmailPageWrapper.tsx +27 -0
  68. package/src/wrappers/{index.d.ts → index.tsx} +8 -1
  69. package/src/auth/Private.d.ts +0 -6
  70. package/src/auth/Private.d.ts.map +0 -1
  71. package/src/auth/Private.js +0 -14
  72. package/src/auth/PrivateRoute.d.ts +0 -8
  73. package/src/auth/PrivateRoute.d.ts.map +0 -1
  74. package/src/auth/PrivateRoute.js +0 -23
  75. package/src/auth/UnAuth.d.ts +0 -6
  76. package/src/auth/UnAuth.d.ts.map +0 -1
  77. package/src/auth/UnAuth.js +0 -14
  78. package/src/auth/UnAuthRoute.d.ts +0 -8
  79. package/src/auth/UnAuthRoute.d.ts.map +0 -1
  80. package/src/auth/UnAuthRoute.js +0 -22
  81. package/src/auth/index.d.ts.map +0 -1
  82. package/src/auth/index.js +0 -10
  83. package/src/components/ApiAccess.d.ts +0 -16
  84. package/src/components/ApiAccess.d.ts.map +0 -1
  85. package/src/components/ApiAccess.js +0 -77
  86. package/src/components/BackupCodeLoginForm.d.ts +0 -43
  87. package/src/components/BackupCodeLoginForm.d.ts.map +0 -1
  88. package/src/components/BackupCodeLoginForm.js +0 -139
  89. package/src/components/BackupCodesForm.d.ts +0 -26
  90. package/src/components/BackupCodesForm.d.ts.map +0 -1
  91. package/src/components/BackupCodesForm.js +0 -120
  92. package/src/components/ChangePasswordForm.d.ts +0 -26
  93. package/src/components/ChangePasswordForm.d.ts.map +0 -1
  94. package/src/components/ChangePasswordForm.js +0 -78
  95. package/src/components/ConfirmationDialog.d.ts +0 -13
  96. package/src/components/ConfirmationDialog.d.ts.map +0 -1
  97. package/src/components/ConfirmationDialog.js +0 -10
  98. package/src/components/CurrencyCodeSelector.d.ts +0 -9
  99. package/src/components/CurrencyCodeSelector.d.ts.map +0 -1
  100. package/src/components/CurrencyCodeSelector.js +0 -31
  101. package/src/components/CurrencyInput.d.ts +0 -13
  102. package/src/components/CurrencyInput.d.ts.map +0 -1
  103. package/src/components/CurrencyInput.js +0 -22
  104. package/src/components/DashboardPage.d.ts +0 -8
  105. package/src/components/DashboardPage.d.ts.map +0 -1
  106. package/src/components/DashboardPage.js +0 -10
  107. package/src/components/DropdownMenu.d.ts +0 -9
  108. package/src/components/DropdownMenu.d.ts.map +0 -1
  109. package/src/components/DropdownMenu.js +0 -56
  110. package/src/components/ExpirationSecondsSelector.d.ts +0 -13
  111. package/src/components/ExpirationSecondsSelector.d.ts.map +0 -1
  112. package/src/components/ExpirationSecondsSelector.js +0 -32
  113. package/src/components/Flag.d.ts +0 -20
  114. package/src/components/Flag.d.ts.map +0 -1
  115. package/src/components/Flag.js +0 -43
  116. package/src/components/ForgotPasswordForm.d.ts +0 -18
  117. package/src/components/ForgotPasswordForm.d.ts.map +0 -1
  118. package/src/components/ForgotPasswordForm.js +0 -61
  119. package/src/components/LoginForm.d.ts +0 -44
  120. package/src/components/LoginForm.d.ts.map +0 -1
  121. package/src/components/LoginForm.js +0 -122
  122. package/src/components/LogoutPage.d.ts +0 -8
  123. package/src/components/LogoutPage.d.ts.map +0 -1
  124. package/src/components/LogoutPage.js +0 -16
  125. package/src/components/RegisterForm.d.ts +0 -56
  126. package/src/components/RegisterForm.d.ts.map +0 -1
  127. package/src/components/RegisterForm.js +0 -140
  128. package/src/components/ResetPasswordForm.d.ts +0 -23
  129. package/src/components/ResetPasswordForm.d.ts.map +0 -1
  130. package/src/components/ResetPasswordForm.js +0 -78
  131. package/src/components/SideMenu.d.ts +0 -8
  132. package/src/components/SideMenu.d.ts.map +0 -1
  133. package/src/components/SideMenu.js +0 -25
  134. package/src/components/SideMenuListItem.d.ts +0 -13
  135. package/src/components/SideMenuListItem.d.ts.map +0 -1
  136. package/src/components/SideMenuListItem.js +0 -44
  137. package/src/components/TopMenu.d.ts +0 -24
  138. package/src/components/TopMenu.d.ts.map +0 -1
  139. package/src/components/TopMenu.js +0 -35
  140. package/src/components/TranslatedTitle.d.ts +0 -7
  141. package/src/components/TranslatedTitle.d.ts.map +0 -1
  142. package/src/components/TranslatedTitle.js +0 -15
  143. package/src/components/UserLanguageSelector.d.ts +0 -4
  144. package/src/components/UserLanguageSelector.d.ts.map +0 -1
  145. package/src/components/UserLanguageSelector.js +0 -31
  146. package/src/components/UserMenu.d.ts +0 -4
  147. package/src/components/UserMenu.d.ts.map +0 -1
  148. package/src/components/UserMenu.js +0 -12
  149. package/src/components/UserSettingsForm.d.ts +0 -57
  150. package/src/components/UserSettingsForm.d.ts.map +0 -1
  151. package/src/components/UserSettingsForm.js +0 -126
  152. package/src/components/VerifyEmailPage.d.ts +0 -23
  153. package/src/components/VerifyEmailPage.d.ts.map +0 -1
  154. package/src/components/VerifyEmailPage.js +0 -70
  155. package/src/components/index.d.ts.map +0 -1
  156. package/src/components/index.js +0 -28
  157. package/src/contexts/AuthProvider.d.ts +0 -152
  158. package/src/contexts/AuthProvider.d.ts.map +0 -1
  159. package/src/contexts/AuthProvider.js +0 -502
  160. package/src/contexts/I18nProvider.d.ts +0 -16
  161. package/src/contexts/I18nProvider.d.ts.map +0 -1
  162. package/src/contexts/I18nProvider.js +0 -46
  163. package/src/contexts/MenuContext.d.ts +0 -20
  164. package/src/contexts/MenuContext.d.ts.map +0 -1
  165. package/src/contexts/MenuContext.js +0 -273
  166. package/src/contexts/SuiteConfigProvider.d.ts +0 -44
  167. package/src/contexts/SuiteConfigProvider.d.ts.map +0 -1
  168. package/src/contexts/SuiteConfigProvider.js +0 -43
  169. package/src/contexts/ThemeProvider.d.ts +0 -15
  170. package/src/contexts/ThemeProvider.d.ts.map +0 -1
  171. package/src/contexts/ThemeProvider.js +0 -36
  172. package/src/contexts/index.d.ts.map +0 -1
  173. package/src/contexts/index.js +0 -8
  174. package/src/hooks/index.d.ts.map +0 -1
  175. package/src/hooks/index.js +0 -8
  176. package/src/hooks/useBackupCodes.d.ts +0 -15
  177. package/src/hooks/useBackupCodes.d.ts.map +0 -1
  178. package/src/hooks/useBackupCodes.js +0 -74
  179. package/src/hooks/useEmailVerification.d.ts +0 -10
  180. package/src/hooks/useEmailVerification.d.ts.map +0 -1
  181. package/src/hooks/useEmailVerification.js +0 -40
  182. package/src/hooks/useExpiringValue.d.ts +0 -14
  183. package/src/hooks/useExpiringValue.d.ts.map +0 -1
  184. package/src/hooks/useExpiringValue.js +0 -53
  185. package/src/hooks/useLocalStorage.d.ts +0 -2
  186. package/src/hooks/useLocalStorage.d.ts.map +0 -1
  187. package/src/hooks/useLocalStorage.js +0 -15
  188. package/src/hooks/useUserSettings.d.ts +0 -48
  189. package/src/hooks/useUserSettings.d.ts.map +0 -1
  190. package/src/hooks/useUserSettings.js +0 -169
  191. package/src/index.d.ts.map +0 -1
  192. package/src/index.js +0 -12
  193. package/src/interfaces/IAppConfig.d.ts +0 -6
  194. package/src/interfaces/IAppConfig.d.ts.map +0 -1
  195. package/src/interfaces/IAppConfig.js +0 -2
  196. package/src/interfaces/IMenuConfig.d.ts +0 -11
  197. package/src/interfaces/IMenuConfig.d.ts.map +0 -1
  198. package/src/interfaces/IMenuConfig.js +0 -2
  199. package/src/interfaces/IMenuOption.d.ts +0 -58
  200. package/src/interfaces/IMenuOption.d.ts.map +0 -1
  201. package/src/interfaces/IMenuOption.js +0 -2
  202. package/src/interfaces/index.d.ts +0 -4
  203. package/src/interfaces/index.d.ts.map +0 -1
  204. package/src/interfaces/index.js +0 -6
  205. package/src/services/__mocks__/authService.d.ts +0 -21
  206. package/src/services/__mocks__/authService.d.ts.map +0 -1
  207. package/src/services/__mocks__/authService.js +0 -15
  208. package/src/services/api.d.ts +0 -3
  209. package/src/services/api.d.ts.map +0 -1
  210. package/src/services/api.js +0 -14
  211. package/src/services/authService.d.ts +0 -72
  212. package/src/services/authService.d.ts.map +0 -1
  213. package/src/services/authService.js +0 -335
  214. package/src/services/authenticatedApi.d.ts +0 -3
  215. package/src/services/authenticatedApi.d.ts.map +0 -1
  216. package/src/services/authenticatedApi.js +0 -18
  217. package/src/services/index.d.ts +0 -4
  218. package/src/services/index.d.ts.map +0 -1
  219. package/src/services/index.js +0 -6
  220. package/src/types/MenuType.d.ts +0 -11
  221. package/src/types/MenuType.d.ts.map +0 -1
  222. package/src/types/MenuType.js +0 -12
  223. package/src/types/expirationSeconds.d.ts +0 -3
  224. package/src/types/expirationSeconds.d.ts.map +0 -1
  225. package/src/types/expirationSeconds.js +0 -17
  226. package/src/types/index.d.ts +0 -2
  227. package/src/types/index.d.ts.map +0 -1
  228. package/src/types/index.js +0 -4
  229. package/src/types/translation.d.ts +0 -10
  230. package/src/types/translation.d.ts.map +0 -1
  231. package/src/types/translation.js +0 -9
  232. package/src/wrappers/BackupCodeLoginWrapper.d.ts +0 -8
  233. package/src/wrappers/BackupCodeLoginWrapper.d.ts.map +0 -1
  234. package/src/wrappers/BackupCodeLoginWrapper.js +0 -20
  235. package/src/wrappers/BackupCodesWrapper.d.ts +0 -7
  236. package/src/wrappers/BackupCodesWrapper.d.ts.map +0 -1
  237. package/src/wrappers/BackupCodesWrapper.js +0 -17
  238. package/src/wrappers/ChangePasswordFormWrapper.d.ts +0 -8
  239. package/src/wrappers/ChangePasswordFormWrapper.d.ts.map +0 -1
  240. package/src/wrappers/ChangePasswordFormWrapper.js +0 -21
  241. package/src/wrappers/LoginFormWrapper.d.ts +0 -9
  242. package/src/wrappers/LoginFormWrapper.d.ts.map +0 -1
  243. package/src/wrappers/LoginFormWrapper.js +0 -43
  244. package/src/wrappers/LogoutPageWrapper.d.ts +0 -9
  245. package/src/wrappers/LogoutPageWrapper.d.ts.map +0 -1
  246. package/src/wrappers/LogoutPageWrapper.js +0 -21
  247. package/src/wrappers/RegisterFormWrapper.d.ts +0 -9
  248. package/src/wrappers/RegisterFormWrapper.d.ts.map +0 -1
  249. package/src/wrappers/RegisterFormWrapper.js +0 -31
  250. package/src/wrappers/UserSettingsFormWrapper.d.ts +0 -8
  251. package/src/wrappers/UserSettingsFormWrapper.d.ts.map +0 -1
  252. package/src/wrappers/UserSettingsFormWrapper.js +0 -24
  253. package/src/wrappers/VerifyEmailPageWrapper.d.ts +0 -8
  254. package/src/wrappers/VerifyEmailPageWrapper.d.ts.map +0 -1
  255. package/src/wrappers/VerifyEmailPageWrapper.js +0 -20
  256. package/src/wrappers/index.d.ts.map +0 -1
  257. package/src/wrappers/index.js +0 -20
@@ -0,0 +1,246 @@
1
+ import {
2
+ Constants,
3
+ SuiteCoreComponentId,
4
+ SuiteCoreStringKey,
5
+ } from '@digitaldefiance/suite-core-lib';
6
+ import {
7
+ Alert,
8
+ Box,
9
+ Button,
10
+ Container,
11
+ TextField,
12
+ Typography,
13
+ } from '@mui/material';
14
+ import { useFormik } from 'formik';
15
+ import { FC, useState } from 'react';
16
+ import * as Yup from 'yup';
17
+ import { useI18n } from '../contexts';
18
+
19
+ export interface ResetPasswordFormValues {
20
+ password: string;
21
+ confirmPassword: string;
22
+ }
23
+
24
+ export interface ResetPasswordFormProps {
25
+ token: string | null;
26
+ onSubmit: (token: string, password: string) => Promise<void>;
27
+ passwordValidation?: Yup.StringSchema;
28
+ confirmPasswordValidation?: Yup.StringSchema;
29
+ labels?: {
30
+ title?: string;
31
+ password?: string;
32
+ confirmPassword?: string;
33
+ resetButton?: string;
34
+ successMessage?: string;
35
+ invalidToken?: string;
36
+ };
37
+ }
38
+
39
+ export const ResetPasswordForm: FC<ResetPasswordFormProps> = ({
40
+ token,
41
+ onSubmit,
42
+ passwordValidation,
43
+ confirmPasswordValidation,
44
+ labels = {},
45
+ }) => {
46
+ const { tComponent } = useI18n();
47
+ const [success, setSuccess] = useState(false);
48
+ const [apiError, setApiError] = useState<string>('');
49
+
50
+ const validation = {
51
+ password:
52
+ passwordValidation ||
53
+ Yup.string()
54
+ .min(
55
+ Constants.PasswordMinLength,
56
+ tComponent<SuiteCoreStringKey>(
57
+ SuiteCoreComponentId,
58
+ SuiteCoreStringKey.Validation_PasswordMinLengthTemplate
59
+ )
60
+ )
61
+ .required(
62
+ tComponent<SuiteCoreStringKey>(
63
+ SuiteCoreComponentId,
64
+ SuiteCoreStringKey.Validation_Required
65
+ )
66
+ ),
67
+ confirmPassword:
68
+ confirmPasswordValidation ||
69
+ Yup.string()
70
+ .oneOf(
71
+ [Yup.ref('password')],
72
+ tComponent<SuiteCoreStringKey>(
73
+ SuiteCoreComponentId,
74
+ SuiteCoreStringKey.Validation_PasswordMatch
75
+ )
76
+ )
77
+ .required(
78
+ tComponent<SuiteCoreStringKey>(
79
+ SuiteCoreComponentId,
80
+ SuiteCoreStringKey.Validation_Required
81
+ )
82
+ ),
83
+ };
84
+
85
+ const translatedLabels = {
86
+ title:
87
+ labels.title ||
88
+ tComponent<SuiteCoreStringKey>(
89
+ SuiteCoreComponentId,
90
+ SuiteCoreStringKey.PasswordReset_Title
91
+ ),
92
+ password:
93
+ labels.password ||
94
+ tComponent<SuiteCoreStringKey>(
95
+ SuiteCoreComponentId,
96
+ SuiteCoreStringKey.Common_NewPassword
97
+ ),
98
+ confirmPassword:
99
+ labels.confirmPassword ||
100
+ tComponent<SuiteCoreStringKey>(
101
+ SuiteCoreComponentId,
102
+ SuiteCoreStringKey.Common_ConfirmNewPassword
103
+ ),
104
+ resetButton:
105
+ labels.resetButton ||
106
+ tComponent<SuiteCoreStringKey>(
107
+ SuiteCoreComponentId,
108
+ SuiteCoreStringKey.PasswordReset_Button
109
+ ),
110
+ successMessage:
111
+ labels.successMessage ||
112
+ tComponent<SuiteCoreStringKey>(
113
+ SuiteCoreComponentId,
114
+ SuiteCoreStringKey.PasswordReset_Success
115
+ ),
116
+ invalidToken:
117
+ labels.invalidToken ||
118
+ tComponent<SuiteCoreStringKey>(
119
+ SuiteCoreComponentId,
120
+ SuiteCoreStringKey.ForgotPassword_InvalidToken
121
+ ),
122
+ };
123
+
124
+ const formik = useFormik<ResetPasswordFormValues>({
125
+ initialValues: {
126
+ password: '',
127
+ confirmPassword: '',
128
+ },
129
+ validationSchema: Yup.object({
130
+ password: validation.password,
131
+ confirmPassword: validation.confirmPassword,
132
+ }),
133
+ onSubmit: async (values) => {
134
+ if (!token) {
135
+ setApiError(translatedLabels.invalidToken);
136
+ return;
137
+ }
138
+
139
+ try {
140
+ await onSubmit(token, values.password);
141
+ setSuccess(true);
142
+ setApiError('');
143
+ } catch (error: unknown) {
144
+ const err = error as { response?: { data?: { message?: string } } };
145
+ setApiError(
146
+ err.response?.data?.message ||
147
+ tComponent<SuiteCoreStringKey>(
148
+ SuiteCoreComponentId,
149
+ SuiteCoreStringKey.Error_PasswordChange
150
+ )
151
+ );
152
+ setSuccess(false);
153
+ }
154
+ },
155
+ });
156
+
157
+ if (!token) {
158
+ return (
159
+ <Container maxWidth="sm">
160
+ <Alert severity="error" sx={{ mt: 4 }}>
161
+ {translatedLabels.invalidToken}
162
+ </Alert>
163
+ </Container>
164
+ );
165
+ }
166
+
167
+ return (
168
+ <Container maxWidth="sm">
169
+ <Box
170
+ sx={{
171
+ mt: 8,
172
+ display: 'flex',
173
+ flexDirection: 'column',
174
+ alignItems: 'center',
175
+ }}
176
+ >
177
+ <Typography variant="h4" component="h1" gutterBottom>
178
+ {translatedLabels.title}
179
+ </Typography>
180
+
181
+ <Box
182
+ component="form"
183
+ onSubmit={formik.handleSubmit}
184
+ sx={{ mt: 1, width: '100%' }}
185
+ >
186
+ <TextField
187
+ fullWidth
188
+ id="password"
189
+ name="password"
190
+ label={translatedLabels.password}
191
+ type="password"
192
+ value={formik.values.password}
193
+ onChange={formik.handleChange}
194
+ onBlur={formik.handleBlur}
195
+ error={Boolean(formik.touched.password && formik.errors.password)}
196
+ helperText={formik.touched.password && formik.errors.password}
197
+ margin="normal"
198
+ />
199
+
200
+ <TextField
201
+ fullWidth
202
+ id="confirmPassword"
203
+ name="confirmPassword"
204
+ label={translatedLabels.confirmPassword}
205
+ type="password"
206
+ value={formik.values.confirmPassword}
207
+ onChange={formik.handleChange}
208
+ onBlur={formik.handleBlur}
209
+ error={Boolean(
210
+ formik.touched.confirmPassword && formik.errors.confirmPassword
211
+ )}
212
+ helperText={
213
+ formik.touched.confirmPassword && formik.errors.confirmPassword
214
+ }
215
+ margin="normal"
216
+ />
217
+
218
+ {apiError && (
219
+ <Alert severity="error" sx={{ mt: 2, mb: 2 }}>
220
+ {apiError}
221
+ </Alert>
222
+ )}
223
+
224
+ {success && (
225
+ <Alert severity="success" sx={{ mt: 2, mb: 2 }}>
226
+ {translatedLabels.successMessage}
227
+ </Alert>
228
+ )}
229
+
230
+ <Button
231
+ type="submit"
232
+ fullWidth
233
+ variant="contained"
234
+ color="primary"
235
+ sx={{ mt: 3, mb: 2 }}
236
+ disabled={formik.isSubmitting}
237
+ >
238
+ {translatedLabels.resetButton}
239
+ </Button>
240
+ </Box>
241
+ </Box>
242
+ </Container>
243
+ );
244
+ };
245
+
246
+ export default ResetPasswordForm;
@@ -0,0 +1,46 @@
1
+ import { Drawer, List } from '@mui/material';
2
+ import { FC } from 'react';
3
+ import { useNavigate } from 'react-router-dom';
4
+ import { useMenu } from '../contexts/MenuContext';
5
+ import { IMenuOption } from '../interfaces/IMenuOption';
6
+ import { MenuTypes } from '../types/MenuType';
7
+ import { SideMenuListItem } from './SideMenuListItem';
8
+
9
+ interface SideMenuProps {
10
+ isOpen: boolean;
11
+ onClose: () => void;
12
+ }
13
+
14
+ export const SideMenu: FC<SideMenuProps> = ({ isOpen, onClose }) => {
15
+ const { getMenuOptions } = useMenu();
16
+ const navigate = useNavigate();
17
+
18
+ const menuOptions = getMenuOptions(MenuTypes.SideMenu, true);
19
+
20
+ const handleNavigate = (
21
+ link: string | Partial<{ pathname: string; state?: unknown }>
22
+ ) => {
23
+ if (typeof link === 'string') {
24
+ navigate(link);
25
+ } else if (link.pathname) {
26
+ navigate(link.pathname, { state: link.state });
27
+ }
28
+ };
29
+
30
+ return (
31
+ <Drawer anchor="left" open={isOpen} onClose={onClose}>
32
+ <List>
33
+ {menuOptions.map((item: IMenuOption) => (
34
+ <SideMenuListItem
35
+ key={item.id}
36
+ menuItem={item}
37
+ onClose={onClose}
38
+ onNavigate={handleNavigate}
39
+ />
40
+ ))}
41
+ </List>
42
+ </Drawer>
43
+ );
44
+ };
45
+
46
+ export default SideMenu;
@@ -0,0 +1,74 @@
1
+ import {
2
+ Divider,
3
+ ListItemButton,
4
+ ListItemIcon,
5
+ ListItemText,
6
+ } from '@mui/material';
7
+ import { FC, useCallback } from 'react';
8
+ import { IMenuOption } from '../interfaces';
9
+
10
+ export interface SideMenuListItemProps {
11
+ menuItem: IMenuOption;
12
+ onClose: () => void;
13
+ onNavigate?: (
14
+ link: string | Partial<{ pathname: string; state?: unknown }>
15
+ ) => void;
16
+ }
17
+
18
+ export const SideMenuListItem: FC<SideMenuListItemProps> = ({
19
+ menuItem,
20
+ onClose,
21
+ onNavigate,
22
+ }) => {
23
+ const handleMenuItemClick = useCallback(
24
+ (option: IMenuOption) => (event: React.MouseEvent<HTMLElement>) => {
25
+ event.stopPropagation();
26
+ if (option.action) {
27
+ option.action();
28
+ } else if (option.link !== undefined && onNavigate) {
29
+ if (typeof option.link === 'string') {
30
+ onNavigate(option.link);
31
+ } else if (
32
+ typeof option.link === 'object' &&
33
+ 'pathname' in option.link &&
34
+ option.link.pathname
35
+ ) {
36
+ onNavigate({
37
+ pathname: option.link.pathname,
38
+ state: 'state' in option.link ? option.link.state : undefined,
39
+ });
40
+ }
41
+ }
42
+ onClose();
43
+ },
44
+ [onNavigate, onClose]
45
+ );
46
+
47
+ if (menuItem.divider) {
48
+ return <Divider key={menuItem.label} />;
49
+ } else if (menuItem.link) {
50
+ return (
51
+ <ListItemButton key={menuItem.id} onClick={handleMenuItemClick(menuItem)}>
52
+ {menuItem.icon && <ListItemIcon>{menuItem.icon}</ListItemIcon>}
53
+ <ListItemText primary={menuItem.label} />
54
+ </ListItemButton>
55
+ );
56
+ } else if (menuItem.action) {
57
+ const action = menuItem.action;
58
+ return (
59
+ <ListItemButton
60
+ key={menuItem.id}
61
+ onClick={async () => {
62
+ await action();
63
+ onClose();
64
+ }}
65
+ >
66
+ {menuItem.icon && <ListItemIcon>{menuItem.icon}</ListItemIcon>}
67
+ <ListItemText primary={menuItem.label} />
68
+ </ListItemButton>
69
+ );
70
+ }
71
+ return null;
72
+ };
73
+
74
+ export default SideMenuListItem;
@@ -0,0 +1,145 @@
1
+ import {
2
+ SuiteCoreComponentId,
3
+ SuiteCoreStringKey,
4
+ } from '@digitaldefiance/suite-core-lib';
5
+ import MenuIcon from '@mui/icons-material/Menu';
6
+ import {
7
+ AppBar,
8
+ Box,
9
+ Button,
10
+ IconButton,
11
+ Toolbar,
12
+ Typography,
13
+ } from '@mui/material';
14
+ import React, { FC, ReactElement, useContext, useState } from 'react';
15
+ import { Link } from 'react-router-dom';
16
+ import { AuthContext } from '../contexts/AuthProvider';
17
+ import { useI18n } from '../contexts/I18nProvider';
18
+ import { useMenu } from '../contexts/MenuContext';
19
+ import { MenuType } from '../types/MenuType';
20
+ import { DropdownMenu } from './DropdownMenu';
21
+ import { SideMenu } from './SideMenu';
22
+ import { UserLanguageSelector } from './UserLanguageSelector';
23
+ import { UserMenu } from './UserMenu';
24
+
25
+ // Extend Window interface for APP_CONFIG
26
+ declare global {
27
+ interface Window {
28
+ APP_CONFIG?: {
29
+ hostname: string;
30
+ siteTitle: string;
31
+ server: string;
32
+ [key: string]: unknown;
33
+ };
34
+ }
35
+ }
36
+
37
+ export interface AdditionalDropdownMenu {
38
+ menuType: MenuType;
39
+ menuIcon: ReactElement;
40
+ priority?: number;
41
+ }
42
+
43
+ export interface TopMenuProps {
44
+ Logo: React.ReactNode;
45
+ additionalMenus?: Array<AdditionalDropdownMenu>;
46
+ }
47
+
48
+ export const TopMenu: FC<TopMenuProps> = ({ Logo }) => {
49
+ const { isAuthenticated } = useContext(AuthContext);
50
+ const { getTopMenus } = useMenu();
51
+ const [isSideMenuOpen, setIsSideMenuOpen] = useState(false);
52
+
53
+ const handleOpenSideMenu = () => setIsSideMenuOpen(true);
54
+ const handleCloseSideMenu = () => setIsSideMenuOpen(false);
55
+ const { tComponent } = useI18n();
56
+ const siteTitle = tComponent<SuiteCoreStringKey>(
57
+ SuiteCoreComponentId,
58
+ SuiteCoreStringKey.Common_SiteTemplate
59
+ );
60
+
61
+ return (
62
+ <AppBar position="fixed" sx={{ top: 10 }}>
63
+ <Toolbar>
64
+ <IconButton
65
+ size="large"
66
+ edge="start"
67
+ color="inherit"
68
+ aria-label="menu"
69
+ sx={{ mr: 2 }}
70
+ onClick={handleOpenSideMenu}
71
+ >
72
+ <MenuIcon />
73
+ </IconButton>
74
+ <Box
75
+ sx={{
76
+ height: 40,
77
+ width: 40,
78
+ marginRight: 2,
79
+ display: 'flex',
80
+ alignItems: 'center',
81
+ }}
82
+ >
83
+ {Logo}
84
+ </Box>
85
+ <Typography variant="h6" component="div" sx={{ flexGrow: 1 }}>
86
+ {siteTitle}
87
+ </Typography>
88
+ <Box sx={{ display: 'flex', alignItems: 'center' }}>
89
+ {isAuthenticated ? (
90
+ <>
91
+ <Button
92
+ color="inherit"
93
+ component={Link}
94
+ to="/dashboard"
95
+ >
96
+ {tComponent<SuiteCoreStringKey>(
97
+ SuiteCoreComponentId,
98
+ SuiteCoreStringKey.Common_Dashboard
99
+ )}
100
+ </Button>
101
+ {getTopMenus().map((menu, index) =>
102
+ menu.isUserMenu ? (
103
+ <UserMenu key={`user-menu`} />
104
+ ) : (
105
+ <DropdownMenu
106
+ key={`menu-${index}`}
107
+ menuType={menu.menuType}
108
+ menuIcon={menu.menuIcon as ReactElement}
109
+ />
110
+ )
111
+ )}
112
+ </>
113
+ ) : (
114
+ <>
115
+ <Button
116
+ color="inherit"
117
+ component={Link}
118
+ to="/login"
119
+ >
120
+ {tComponent<SuiteCoreStringKey>(
121
+ SuiteCoreComponentId,
122
+ SuiteCoreStringKey.Login_LoginButton
123
+ )}
124
+ </Button>
125
+ <Button
126
+ color="inherit"
127
+ component={Link}
128
+ to="/register"
129
+ >
130
+ {tComponent<SuiteCoreStringKey>(
131
+ SuiteCoreComponentId,
132
+ SuiteCoreStringKey.RegisterButton
133
+ )}
134
+ </Button>
135
+ </>
136
+ )}
137
+ <UserLanguageSelector />
138
+ </Box>
139
+ </Toolbar>
140
+ <SideMenu isOpen={isSideMenuOpen} onClose={handleCloseSideMenu} />
141
+ </AppBar>
142
+ );
143
+ };
144
+
145
+ export default TopMenu;
@@ -0,0 +1,29 @@
1
+ // src/app/components/TranslatedTitle.tsx
2
+
3
+ import { useEffect } from 'react';
4
+ import { useI18n } from '../contexts';
5
+
6
+ interface FCParams<TEnum extends string> {
7
+ componentId: string;
8
+ stringKey: TEnum;
9
+ }
10
+
11
+ export const TranslatedTitle = <TEnum extends string>({
12
+ componentId,
13
+ stringKey,
14
+ }: FCParams<TEnum>): null => {
15
+ const { tComponent, currentLanguage } = useI18n();
16
+
17
+ useEffect(() => {
18
+ document.title = tComponent<TEnum>(
19
+ componentId,
20
+ stringKey,
21
+ undefined,
22
+ currentLanguage
23
+ );
24
+ }, [tComponent, componentId, stringKey, currentLanguage]);
25
+
26
+ return null;
27
+ };
28
+
29
+ export default TranslatedTitle;
@@ -0,0 +1,45 @@
1
+ import { LanguageRegistry, LanguageDefinition } from '@digitaldefiance/i18n-lib';
2
+ import { Button, Menu, MenuItem } from '@mui/material';
3
+ import { FC, MouseEvent, useState, useMemo } from 'react';
4
+ import { Flag } from './Flag';
5
+ import { useUserSettings } from '../hooks';
6
+ import { createAuthenticatedApiClient } from '../services';
7
+ import { useAuth, useSuiteConfig } from '../contexts';
8
+
9
+ export const UserLanguageSelector: FC = () => {
10
+ const { baseUrl } = useSuiteConfig();
11
+ const authenticatedApi = useMemo(() => createAuthenticatedApiClient(baseUrl), [baseUrl]);
12
+ const { isAuthenticated } = useAuth();
13
+ const { currentLanguage, changeLanguage } = useUserSettings({ authenticatedApi, isAuthenticated });
14
+ const [anchorEl, setAnchorEl] = useState<null | HTMLElement>(null);
15
+
16
+ const handleClick = (event: MouseEvent<HTMLButtonElement>) => {
17
+ setAnchorEl(event.currentTarget);
18
+ };
19
+
20
+ const handleClose = () => {
21
+ setAnchorEl(null);
22
+ };
23
+
24
+ const handleLanguageChange = (newLanguage: string) => {
25
+ changeLanguage(newLanguage);
26
+ handleClose();
27
+ };
28
+
29
+ return (
30
+ <>
31
+ <Button onClick={handleClick}>
32
+ <Flag language={currentLanguage} />
33
+ </Button>
34
+ <Menu anchorEl={anchorEl} open={Boolean(anchorEl)} onClose={handleClose}>
35
+ {Object.values(LanguageRegistry.getAllLanguages()).map((lang: LanguageDefinition) => (
36
+ <MenuItem key={lang.code} onClick={() => handleLanguageChange(lang.code)}>
37
+ <Flag language={lang.code} sx={{ mr: 1 }} /> {lang.name}
38
+ </MenuItem>
39
+ ))}
40
+ </Menu>
41
+ </>
42
+ );
43
+ };
44
+
45
+ export default UserLanguageSelector;
@@ -0,0 +1,15 @@
1
+ import { FC } from 'react';
2
+ import { AccountCircle } from '@mui/icons-material';
3
+ import { MenuTypes } from '../types/MenuType';
4
+ import { DropdownMenu } from './DropdownMenu';
5
+
6
+ export const UserMenu: FC = () => {
7
+ return (
8
+ <DropdownMenu
9
+ menuType={MenuTypes.UserMenu}
10
+ menuIcon={<AccountCircle />}
11
+ />
12
+ );
13
+ };
14
+
15
+ export default UserMenu;