@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,80 @@
1
+ import { TextField } from '@mui/material';
2
+ import { NumericFormat } from 'react-number-format';
3
+ import { getCurrencyFormat } from '@digitaldefiance/i18n-lib';
4
+
5
+ export interface CurrencyInputProps {
6
+ value: number;
7
+ onChange: (value: number) => void;
8
+ currencyCode?: string;
9
+ locale?: string;
10
+ label: string;
11
+ error?: boolean;
12
+ helperText?: string;
13
+ name: string;
14
+ }
15
+
16
+ export const CurrencyInput: React.FC<CurrencyInputProps> = ({
17
+ value,
18
+ onChange,
19
+ currencyCode = 'USD',
20
+ locale = 'en-US',
21
+ label,
22
+ error,
23
+ helperText,
24
+ name,
25
+ }) => {
26
+ const format = getCurrencyFormat(locale, currencyCode);
27
+
28
+ if (format.position === 'infix') {
29
+ const [whole, decimal] = value.toString().split('.');
30
+ const displayValue = `${whole}${format.symbol}${format.decimalSeparator}${
31
+ decimal || '00'
32
+ }`;
33
+
34
+ return (
35
+ <NumericFormat
36
+ customInput={TextField}
37
+ fullWidth
38
+ margin="normal"
39
+ label={label}
40
+ value={displayValue}
41
+ thousandSeparator={format.groupSeparator}
42
+ decimalSeparator={format.decimalSeparator}
43
+ decimalScale={2}
44
+ fixedDecimalScale
45
+ valueIsNumericString
46
+ onValueChange={(values) => {
47
+ onChange(values.floatValue || 0);
48
+ }}
49
+ error={error}
50
+ helperText={helperText}
51
+ name={name}
52
+ />
53
+ );
54
+ }
55
+
56
+ return (
57
+ <NumericFormat
58
+ customInput={TextField}
59
+ fullWidth
60
+ margin="normal"
61
+ label={label}
62
+ value={value}
63
+ thousandSeparator={format.groupSeparator}
64
+ decimalSeparator={format.decimalSeparator}
65
+ decimalScale={2}
66
+ fixedDecimalScale
67
+ prefix={format.position === 'prefix' ? format.symbol + ' ' : undefined}
68
+ suffix={format.position === 'postfix' ? ' ' + format.symbol : undefined}
69
+ valueIsNumericString
70
+ onValueChange={(values) => {
71
+ onChange(values.floatValue || 0);
72
+ }}
73
+ error={error}
74
+ helperText={helperText}
75
+ name={name}
76
+ />
77
+ );
78
+ };
79
+
80
+ export default CurrencyInput;
@@ -0,0 +1,24 @@
1
+ import { Box, Container, Typography } from '@mui/material';
2
+ import { FC, ReactNode } from 'react';
3
+
4
+ export interface DashboardPageProps {
5
+ title?: string;
6
+ children?: ReactNode;
7
+ }
8
+
9
+ export const DashboardPage: FC<DashboardPageProps> = ({ title = 'Dashboard', children }) => {
10
+ return (
11
+ <Container maxWidth="md">
12
+ <Box my={4}>
13
+ <Typography variant="h4" component="h1" gutterBottom align="center">
14
+ {title}
15
+ </Typography>
16
+ <Box display="flex" justifyContent="center" mt={3}>
17
+ {children}
18
+ </Box>
19
+ </Box>
20
+ </Container>
21
+ );
22
+ };
23
+
24
+ export default DashboardPage;
@@ -0,0 +1,92 @@
1
+ import { Box, Fade, IconButton, Menu, MenuItem } from '@mui/material';
2
+ import { FC, MouseEvent, ReactElement, useCallback, useState } from 'react';
3
+ import { useNavigate } from 'react-router-dom';
4
+ import { MenuType } from '../types/MenuType';
5
+ import { IMenuOption } from '../interfaces/IMenuOption';
6
+ import { useMenu } from '../contexts/MenuContext';
7
+
8
+ interface DropdownMenuProps {
9
+ menuType: MenuType;
10
+ menuIcon: ReactElement;
11
+ }
12
+
13
+ export const DropdownMenu: FC<DropdownMenuProps> = ({ menuType, menuIcon }) => {
14
+ const { getMenuOptions } = useMenu();
15
+ const [anchorEl, setAnchorEl] = useState<null | HTMLElement>(null);
16
+ const navigate = useNavigate();
17
+ const handleClose = useCallback(() => {
18
+ setAnchorEl(null);
19
+ }, []);
20
+ const handleMenuItemClick = useCallback(
21
+ (option: IMenuOption) => (event: React.MouseEvent<HTMLElement>) => {
22
+ event.stopPropagation();
23
+ if (option.action) {
24
+ option.action();
25
+ } else if (option.link !== undefined) {
26
+ if (
27
+ typeof option.link === 'object' &&
28
+ 'pathname' in option.link &&
29
+ 'state' in option.link
30
+ ) {
31
+ navigate(option.link.pathname, { state: option.link.state });
32
+ } else {
33
+ navigate(option.link);
34
+ }
35
+ }
36
+ handleClose(); // Call handleClose after handling the click
37
+ },
38
+ [navigate, handleClose], // Add handleClose to the dependency array
39
+ );
40
+
41
+ const handleClick = useCallback((event: MouseEvent<HTMLElement>) => {
42
+ setAnchorEl(event.currentTarget);
43
+ }, []);
44
+
45
+ const menuItems = getMenuOptions(menuType, false);
46
+
47
+ if (menuItems.length === 0) {
48
+ return null;
49
+ }
50
+
51
+ return (
52
+ <Box>
53
+ <IconButton color="inherit" onClick={handleClick}>
54
+ {menuIcon}
55
+ </IconButton>
56
+ <Menu
57
+ anchorEl={anchorEl}
58
+ open={Boolean(anchorEl)}
59
+ onClose={handleClose}
60
+ TransitionComponent={Fade}
61
+ sx={{
62
+ '& .MuiPopover-paper': {
63
+ opacity: 0.5,
64
+ overflow: 'visible',
65
+ },
66
+ }}
67
+ >
68
+ {menuItems.map((option) => (
69
+ <MenuItem
70
+ key={option.id}
71
+ component="li"
72
+ onClick={handleMenuItemClick(option)}
73
+ sx={{
74
+ display: 'flex',
75
+ alignItems: 'center',
76
+ '& > svg': {
77
+ marginRight: 2,
78
+ width: 24,
79
+ height: 24,
80
+ },
81
+ }}
82
+ >
83
+ {option.icon}
84
+ {option.label}
85
+ </MenuItem>
86
+ ))}
87
+ </Menu>
88
+ </Box>
89
+ );
90
+ };
91
+
92
+ export default DropdownMenu;
@@ -0,0 +1,60 @@
1
+ import { MenuItem, TextField } from '@mui/material';
2
+ import { FormikProps } from 'formik';
3
+ import { ChangeEvent, FC } from 'react';
4
+
5
+ export interface ExpirationSecondsSelectorProps<T = Record<string, unknown>> {
6
+ name: string;
7
+ label: string;
8
+ formik: FormikProps<T>;
9
+ optionValues: number[];
10
+ optionNames: string[];
11
+ onChange?: (value: number) => void;
12
+ }
13
+
14
+ export const ExpirationSecondsSelector: FC<
15
+ ExpirationSecondsSelectorProps<Record<string, unknown>>
16
+ > = ({ name, label, formik, optionValues, optionNames, onChange }) => {
17
+ return (
18
+ <TextField
19
+ select
20
+ fullWidth
21
+ label={label}
22
+ name={name}
23
+ value={formik.values[name] ?? ''}
24
+ onChange={(event: ChangeEvent<HTMLInputElement>) => {
25
+ const selectedValue = event.target.value;
26
+ formik.setFieldValue(name, selectedValue);
27
+ if (onChange) {
28
+ onChange(parseInt(selectedValue));
29
+ }
30
+ }}
31
+ error={formik.touched[name] && Boolean(formik.errors[name])}
32
+ helperText={formik.touched[name] && (formik.errors[name] as string)}
33
+ sx={{
34
+ mt: 1,
35
+ '& .MuiSelect-select': {
36
+ paddingRight: '32px',
37
+ },
38
+ '& .MuiOutlinedInput-root': {
39
+ '& fieldset': {
40
+ borderColor: 'rgba(0, 0, 0, 0.23)',
41
+ },
42
+ '&:hover fieldset': {
43
+ borderColor: 'rgba(0, 0, 0, 0.87)',
44
+ },
45
+ '&.Mui-focused fieldset': {
46
+ borderColor: 'primary.main',
47
+ },
48
+ },
49
+ }}
50
+ >
51
+ {optionNames.map((name: string, index: number) => (
52
+ <MenuItem key={name} value={optionValues[index]}>
53
+ {name}
54
+ </MenuItem>
55
+ ))}
56
+ </TextField>
57
+ );
58
+ };
59
+
60
+ export default ExpirationSecondsSelector;
@@ -0,0 +1,52 @@
1
+ import { getFlagCode } from '@digitaldefiance/suite-core-lib';
2
+ import { Box, SxProps, Theme } from '@mui/material';
3
+ import { FC } from 'react';
4
+
5
+ export interface FlagProps {
6
+ language: string;
7
+ sx?: SxProps<Theme>;
8
+ }
9
+
10
+ /**
11
+ * A simple component to display a flag icon for a given language.
12
+ *
13
+ * Props:
14
+ * language: The language to display a flag for, as a StringLanguages enum value.
15
+ * sx: Optional styles to apply to the component.
16
+ *
17
+ * Returns a Box component with an SVG flag icon from flagcdn.com as a ::before pseudo-element.
18
+ * The flag is sized to 1.5rem by default, but can be overridden by passing a custom sx prop.
19
+ * The component also includes an aria-label for accessibility, set to `Flag for <language>`.
20
+ */
21
+ export const Flag: FC<FlagProps> = ({ language, sx }) => {
22
+ const flagContent = getFlagCode(language);
23
+ if (!flagContent) {
24
+ return null;
25
+ }
26
+ return (
27
+ <Box
28
+ component="span"
29
+ aria-label={`Flag for ${language}`}
30
+ sx={{
31
+ fontSize: '1.5rem',
32
+ lineHeight: 1,
33
+ verticalAlign: 'middle',
34
+ '&::before': {
35
+ content: `" "`,
36
+ display: 'inline-block',
37
+ width: '1em',
38
+ height: '1em',
39
+ backgroundImage: `url(https://flagcdn.com/${flagContent.toLowerCase()}.svg)`,
40
+ backgroundSize: 'contain',
41
+ backgroundRepeat: 'no-repeat',
42
+ backgroundPosition: 'center',
43
+ border: '1px solid rgba(0, 0, 0, 0.1)',
44
+ borderRadius: '2px',
45
+ },
46
+ ...sx,
47
+ }}
48
+ />
49
+ );
50
+ };
51
+
52
+ export default Flag;
@@ -0,0 +1,173 @@
1
+ import {
2
+ SuiteCoreComponentId,
3
+ SuiteCoreStringKey,
4
+ } from '@digitaldefiance/suite-core-lib';
5
+ import {
6
+ Alert,
7
+ Box,
8
+ Button,
9
+ Container,
10
+ TextField,
11
+ Typography,
12
+ } from '@mui/material';
13
+ import { useFormik } from 'formik';
14
+ import { FC, useState } from 'react';
15
+ import * as Yup from 'yup';
16
+ import { useI18n } from '../contexts';
17
+
18
+ export interface ForgotPasswordFormValues {
19
+ email: string;
20
+ }
21
+
22
+ export interface ForgotPasswordFormProps {
23
+ onSubmit: (values: ForgotPasswordFormValues) => Promise<void>;
24
+ emailValidation?: Yup.StringSchema;
25
+ labels?: {
26
+ title?: string;
27
+ email?: string;
28
+ sendResetLink?: string;
29
+ successMessage?: string;
30
+ };
31
+ }
32
+
33
+ export const ForgotPasswordForm: FC<ForgotPasswordFormProps> = ({
34
+ onSubmit,
35
+ emailValidation,
36
+ labels = {},
37
+ }) => {
38
+ const { tComponent } = useI18n();
39
+ const [success, setSuccess] = useState(false);
40
+ const [apiError, setApiError] = useState<string>('');
41
+
42
+ const validation = {
43
+ email:
44
+ emailValidation ||
45
+ Yup.string()
46
+ .email(
47
+ tComponent<SuiteCoreStringKey>(
48
+ SuiteCoreComponentId,
49
+ SuiteCoreStringKey.Validation_InvalidEmail
50
+ )
51
+ )
52
+ .required(
53
+ tComponent<SuiteCoreStringKey>(
54
+ SuiteCoreComponentId,
55
+ SuiteCoreStringKey.Validation_Required
56
+ )
57
+ ),
58
+ };
59
+
60
+ const translatedLabels = {
61
+ title:
62
+ labels.title ||
63
+ tComponent<SuiteCoreStringKey>(
64
+ SuiteCoreComponentId,
65
+ SuiteCoreStringKey.ForgotPassword_Title
66
+ ),
67
+ email:
68
+ labels.email ||
69
+ tComponent<SuiteCoreStringKey>(
70
+ SuiteCoreComponentId,
71
+ SuiteCoreStringKey.Common_Email
72
+ ),
73
+ sendResetLink:
74
+ labels.sendResetLink ||
75
+ tComponent<SuiteCoreStringKey>(
76
+ SuiteCoreComponentId,
77
+ SuiteCoreStringKey.ForgotPassword_SendResetLink
78
+ ),
79
+ successMessage:
80
+ labels.successMessage ||
81
+ tComponent<SuiteCoreStringKey>(
82
+ SuiteCoreComponentId,
83
+ SuiteCoreStringKey.PasswordReset_Success
84
+ ),
85
+ };
86
+
87
+ const formik = useFormik<ForgotPasswordFormValues>({
88
+ initialValues: {
89
+ email: '',
90
+ },
91
+ validationSchema: Yup.object({
92
+ email: validation.email,
93
+ }),
94
+ onSubmit: async (values) => {
95
+ try {
96
+ await onSubmit(values);
97
+ setSuccess(true);
98
+ setApiError('');
99
+ } catch (error: unknown) {
100
+ const err = error as { response?: { data?: { message?: string } } };
101
+ setApiError(
102
+ err.response?.data?.message ||
103
+ tComponent<SuiteCoreStringKey>(
104
+ SuiteCoreComponentId,
105
+ SuiteCoreStringKey.ForgotPassword_Error
106
+ )
107
+ );
108
+ setSuccess(false);
109
+ }
110
+ },
111
+ });
112
+
113
+ return (
114
+ <Container maxWidth="sm">
115
+ <Box
116
+ sx={{
117
+ mt: 8,
118
+ display: 'flex',
119
+ flexDirection: 'column',
120
+ alignItems: 'center',
121
+ }}
122
+ >
123
+ <Typography variant="h4" component="h1" gutterBottom>
124
+ {translatedLabels.title}
125
+ </Typography>
126
+
127
+ <Box
128
+ component="form"
129
+ onSubmit={formik.handleSubmit}
130
+ sx={{ mt: 1, width: '100%' }}
131
+ >
132
+ <TextField
133
+ fullWidth
134
+ id="email"
135
+ name="email"
136
+ label={translatedLabels.email}
137
+ value={formik.values.email}
138
+ onChange={formik.handleChange}
139
+ onBlur={formik.handleBlur}
140
+ error={Boolean(formik.touched.email && formik.errors.email)}
141
+ helperText={formik.touched.email && formik.errors.email}
142
+ margin="normal"
143
+ />
144
+
145
+ {apiError && (
146
+ <Alert severity="error" sx={{ mt: 2, mb: 2 }}>
147
+ {apiError}
148
+ </Alert>
149
+ )}
150
+
151
+ {success && (
152
+ <Alert severity="success" sx={{ mt: 2, mb: 2 }}>
153
+ {translatedLabels.successMessage}
154
+ </Alert>
155
+ )}
156
+
157
+ <Button
158
+ type="submit"
159
+ fullWidth
160
+ variant="contained"
161
+ color="primary"
162
+ sx={{ mt: 3, mb: 2 }}
163
+ disabled={formik.isSubmitting}
164
+ >
165
+ {translatedLabels.sendResetLink}
166
+ </Button>
167
+ </Box>
168
+ </Box>
169
+ </Container>
170
+ );
171
+ };
172
+
173
+ export default ForgotPasswordForm;