@riligar/auth-react 1.9.4 → 1.11.0
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.
- package/README.md +100 -85
- package/dist/index.esm.js +906 -15
- package/dist/index.esm.js.map +1 -1
- package/dist/index.js +933 -19
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
package/dist/index.esm.js
CHANGED
|
@@ -2,10 +2,11 @@ import { create } from 'zustand';
|
|
|
2
2
|
import { useMemo, useEffect, createContext, useState, useRef } from 'react';
|
|
3
3
|
import { useShallow } from 'zustand/react/shallow';
|
|
4
4
|
import { jsx, jsxs, Fragment } from 'react/jsx-runtime';
|
|
5
|
-
import { Navigate, Outlet } from 'react-router-dom';
|
|
6
|
-
import { Paper, Stack, Image, Title, Text, TextInput, PasswordInput, Anchor, Button, Divider, Group, Center, Loader } from '@mantine/core';
|
|
5
|
+
import { Navigate, Outlet, useNavigate } from 'react-router-dom';
|
|
6
|
+
import { Paper, Stack, Image, Title, Text, TextInput, PasswordInput, Anchor, Button, Divider, Group, Center, Loader, Modal, Box, Avatar, Collapse, Card, Tooltip, ThemeIcon } from '@mantine/core';
|
|
7
7
|
import { useForm } from '@mantine/form';
|
|
8
|
-
import { IconMail, IconLock, IconArrowRight, IconBrandGoogle, IconBrandGithub, IconUser, IconSend, IconCheck, IconX, IconRefresh } from '@tabler/icons-react';
|
|
8
|
+
import { IconMail, IconLock, IconArrowRight, IconBrandGoogle, IconBrandGithub, IconUser, IconSend, IconCheck, IconX, IconRefresh, IconPhoto, IconTrash, IconPencil, IconShield, IconKey, IconUserCircle } from '@tabler/icons-react';
|
|
9
|
+
import { notifications } from '@mantine/notifications';
|
|
9
10
|
|
|
10
11
|
// Config - pode ser sobrescrita pelo AuthProvider
|
|
11
12
|
// Tenta detectar ambiente Vite, mas falha graciosamente se não existir
|
|
@@ -281,6 +282,34 @@ const getApplicationInfo = async () => {
|
|
|
281
282
|
}
|
|
282
283
|
};
|
|
283
284
|
|
|
285
|
+
/*--- Profile Management ---------------------------*/
|
|
286
|
+
const updateProfile = async data => {
|
|
287
|
+
return await api('/auth/update-user', {
|
|
288
|
+
method: 'POST',
|
|
289
|
+
body: JSON.stringify(data)
|
|
290
|
+
});
|
|
291
|
+
};
|
|
292
|
+
const changePassword = async (currentPassword, newPassword, revokeOtherSessions = false) => {
|
|
293
|
+
return await api('/auth/change-password', {
|
|
294
|
+
method: 'POST',
|
|
295
|
+
body: JSON.stringify({
|
|
296
|
+
currentPassword,
|
|
297
|
+
newPassword,
|
|
298
|
+
revokeOtherSessions
|
|
299
|
+
})
|
|
300
|
+
});
|
|
301
|
+
};
|
|
302
|
+
const changeEmail = async (newEmail, callbackURL) => {
|
|
303
|
+
const callback = callbackURL || (typeof window !== 'undefined' ? `${window.location.origin}/auth/verify-email` : '/auth/verify-email');
|
|
304
|
+
return await api('/auth/change-email', {
|
|
305
|
+
method: 'POST',
|
|
306
|
+
body: JSON.stringify({
|
|
307
|
+
newEmail,
|
|
308
|
+
callbackURL: callback
|
|
309
|
+
})
|
|
310
|
+
});
|
|
311
|
+
};
|
|
312
|
+
|
|
284
313
|
/* Social login redirect (ex.: Google) -----------*/
|
|
285
314
|
function socialRedirect(provider, redirectTo = typeof window !== 'undefined' ? window.location.href : '/') {
|
|
286
315
|
if (typeof window === 'undefined') return;
|
|
@@ -301,7 +330,10 @@ const useAuthStore = create((set, get) => ({
|
|
|
301
330
|
verifyMagicLink: false,
|
|
302
331
|
resetPassword: false,
|
|
303
332
|
verifyEmail: false,
|
|
304
|
-
resendVerification: false
|
|
333
|
+
resendVerification: false,
|
|
334
|
+
updateProfile: false,
|
|
335
|
+
changePassword: false,
|
|
336
|
+
changeEmail: false
|
|
305
337
|
},
|
|
306
338
|
// Application info (logo, nome, etc)
|
|
307
339
|
applicationInfo: null,
|
|
@@ -668,7 +700,75 @@ const useAuthStore = create((set, get) => ({
|
|
|
668
700
|
/* Atualizar usuário manualmente */
|
|
669
701
|
setUser: user => set({
|
|
670
702
|
user
|
|
671
|
-
})
|
|
703
|
+
}),
|
|
704
|
+
/* Profile Management */
|
|
705
|
+
updateProfile: async data => {
|
|
706
|
+
const {
|
|
707
|
+
setLoading
|
|
708
|
+
} = get();
|
|
709
|
+
setLoading('updateProfile', true);
|
|
710
|
+
set({
|
|
711
|
+
error: null
|
|
712
|
+
});
|
|
713
|
+
try {
|
|
714
|
+
const result = await updateProfile(data);
|
|
715
|
+
// Atualiza o user no store com os novos dados
|
|
716
|
+
set(state => ({
|
|
717
|
+
user: state.user ? {
|
|
718
|
+
...state.user,
|
|
719
|
+
...data
|
|
720
|
+
} : null
|
|
721
|
+
}));
|
|
722
|
+
setLoading('updateProfile', false);
|
|
723
|
+
return result;
|
|
724
|
+
} catch (err) {
|
|
725
|
+
set({
|
|
726
|
+
error: err
|
|
727
|
+
});
|
|
728
|
+
setLoading('updateProfile', false);
|
|
729
|
+
throw err;
|
|
730
|
+
}
|
|
731
|
+
},
|
|
732
|
+
changePassword: async (currentPassword, newPassword, revokeOtherSessions = false) => {
|
|
733
|
+
const {
|
|
734
|
+
setLoading
|
|
735
|
+
} = get();
|
|
736
|
+
setLoading('changePassword', true);
|
|
737
|
+
set({
|
|
738
|
+
error: null
|
|
739
|
+
});
|
|
740
|
+
try {
|
|
741
|
+
const result = await changePassword(currentPassword, newPassword, revokeOtherSessions);
|
|
742
|
+
setLoading('changePassword', false);
|
|
743
|
+
return result;
|
|
744
|
+
} catch (err) {
|
|
745
|
+
set({
|
|
746
|
+
error: err
|
|
747
|
+
});
|
|
748
|
+
setLoading('changePassword', false);
|
|
749
|
+
throw err;
|
|
750
|
+
}
|
|
751
|
+
},
|
|
752
|
+
changeEmail: async (newEmail, callbackURL) => {
|
|
753
|
+
const {
|
|
754
|
+
setLoading
|
|
755
|
+
} = get();
|
|
756
|
+
setLoading('changeEmail', true);
|
|
757
|
+
set({
|
|
758
|
+
error: null
|
|
759
|
+
});
|
|
760
|
+
try {
|
|
761
|
+
const result = await changeEmail(newEmail, callbackURL);
|
|
762
|
+
setLoading('changeEmail', false);
|
|
763
|
+
return result;
|
|
764
|
+
} catch (err) {
|
|
765
|
+
set({
|
|
766
|
+
error: err
|
|
767
|
+
});
|
|
768
|
+
setLoading('changeEmail', false);
|
|
769
|
+
throw err;
|
|
770
|
+
}
|
|
771
|
+
}
|
|
672
772
|
}));
|
|
673
773
|
|
|
674
774
|
const AuthContext = /*#__PURE__*/createContext(); // só para ter o Provider em JSX
|
|
@@ -792,6 +892,21 @@ const useSession = () => useAuthStore(useShallow(s => ({
|
|
|
792
892
|
// Loading States Hook
|
|
793
893
|
const useAuthLoading = () => useAuthStore(s => s.loadingStates);
|
|
794
894
|
|
|
895
|
+
// Profile Management Hook (novo nome estilo Clerk)
|
|
896
|
+
const useUser = () => useAuthStore(useShallow(s => ({
|
|
897
|
+
user: s.user,
|
|
898
|
+
updateProfile: s.updateProfile,
|
|
899
|
+
changePassword: s.changePassword,
|
|
900
|
+
changeEmail: s.changeEmail,
|
|
901
|
+
loadingUpdateProfile: s.loadingStates.updateProfile,
|
|
902
|
+
loadingChangePassword: s.loadingStates.changePassword,
|
|
903
|
+
loadingChangeEmail: s.loadingStates.changeEmail,
|
|
904
|
+
error: s.error
|
|
905
|
+
})));
|
|
906
|
+
|
|
907
|
+
// Alias deprecado para backwards compatibility
|
|
908
|
+
const useProfile = useUser;
|
|
909
|
+
|
|
795
910
|
// Application Logo Hook
|
|
796
911
|
const useApplicationLogo = () => {
|
|
797
912
|
const applicationInfo = useAuthStore(s => s.applicationInfo);
|
|
@@ -799,11 +914,11 @@ const useApplicationLogo = () => {
|
|
|
799
914
|
return applicationInfo?.image || null;
|
|
800
915
|
};
|
|
801
916
|
|
|
802
|
-
function
|
|
917
|
+
function Protect({
|
|
803
918
|
fallback = /*#__PURE__*/jsx("p", {
|
|
804
919
|
children: "\u231B Carregando..."
|
|
805
920
|
}),
|
|
806
|
-
redirectTo =
|
|
921
|
+
redirectTo = '/login'
|
|
807
922
|
}) {
|
|
808
923
|
const {
|
|
809
924
|
user,
|
|
@@ -857,7 +972,7 @@ function AuthCard({
|
|
|
857
972
|
|
|
858
973
|
var img = "";
|
|
859
974
|
|
|
860
|
-
function
|
|
975
|
+
function SignIn({
|
|
861
976
|
// Configuração
|
|
862
977
|
logo,
|
|
863
978
|
// Removido default, será calculado abaixo
|
|
@@ -1025,7 +1140,7 @@ function SignInForm({
|
|
|
1025
1140
|
});
|
|
1026
1141
|
}
|
|
1027
1142
|
|
|
1028
|
-
function
|
|
1143
|
+
function SignUp({
|
|
1029
1144
|
// Configuração
|
|
1030
1145
|
logo,
|
|
1031
1146
|
title = 'Criar Conta',
|
|
@@ -1159,7 +1274,7 @@ function SignUpForm({
|
|
|
1159
1274
|
});
|
|
1160
1275
|
}
|
|
1161
1276
|
|
|
1162
|
-
function
|
|
1277
|
+
function MagicLink({
|
|
1163
1278
|
// Configuração
|
|
1164
1279
|
logo,
|
|
1165
1280
|
title = 'Login sem Senha',
|
|
@@ -1237,7 +1352,7 @@ function MagicLinkForm({
|
|
|
1237
1352
|
});
|
|
1238
1353
|
}
|
|
1239
1354
|
|
|
1240
|
-
function
|
|
1355
|
+
function MagicLinkCallback({
|
|
1241
1356
|
// Configuração
|
|
1242
1357
|
logo,
|
|
1243
1358
|
// Token pode ser passado diretamente ou extraído da URL
|
|
@@ -1347,7 +1462,7 @@ function MagicLinkVerify({
|
|
|
1347
1462
|
});
|
|
1348
1463
|
}
|
|
1349
1464
|
|
|
1350
|
-
function
|
|
1465
|
+
function ForgotPassword({
|
|
1351
1466
|
// Configuração
|
|
1352
1467
|
logo,
|
|
1353
1468
|
title = 'Recuperar Senha',
|
|
@@ -1425,7 +1540,7 @@ function ForgotPasswordForm({
|
|
|
1425
1540
|
});
|
|
1426
1541
|
}
|
|
1427
1542
|
|
|
1428
|
-
function
|
|
1543
|
+
function ResetPassword({
|
|
1429
1544
|
// Configuração
|
|
1430
1545
|
logo,
|
|
1431
1546
|
title = 'Nova Senha',
|
|
@@ -1564,7 +1679,7 @@ function ResetPasswordForm({
|
|
|
1564
1679
|
});
|
|
1565
1680
|
}
|
|
1566
1681
|
|
|
1567
|
-
function
|
|
1682
|
+
function VerifyEmail({
|
|
1568
1683
|
// Configuração
|
|
1569
1684
|
logo,
|
|
1570
1685
|
// Token pode ser passado diretamente ou extraído da URL
|
|
@@ -1728,5 +1843,781 @@ function VerifyEmailCard({
|
|
|
1728
1843
|
});
|
|
1729
1844
|
}
|
|
1730
1845
|
|
|
1731
|
-
|
|
1846
|
+
function UserProfile({
|
|
1847
|
+
// Controle do modal
|
|
1848
|
+
opened,
|
|
1849
|
+
onClose,
|
|
1850
|
+
// Callbacks
|
|
1851
|
+
onProfileUpdate,
|
|
1852
|
+
onPasswordChange,
|
|
1853
|
+
onEmailChange,
|
|
1854
|
+
onError,
|
|
1855
|
+
// Features toggle
|
|
1856
|
+
showAvatar = true,
|
|
1857
|
+
showName = true,
|
|
1858
|
+
showEmail = true,
|
|
1859
|
+
showPassword = true,
|
|
1860
|
+
// Customização
|
|
1861
|
+
labels = {},
|
|
1862
|
+
title = 'Account',
|
|
1863
|
+
subtitle = 'Manage your account info.',
|
|
1864
|
+
// Avatar config
|
|
1865
|
+
maxAvatarSize = 500 * 1024,
|
|
1866
|
+
// 500KB
|
|
1867
|
+
|
|
1868
|
+
...modalProps
|
|
1869
|
+
}) {
|
|
1870
|
+
// Local state - which section is expanded
|
|
1871
|
+
const [editingSection, setEditingSection] = useState(null); // 'password' | 'email' | 'name' | 'avatar' | null
|
|
1872
|
+
|
|
1873
|
+
// Hook para profile
|
|
1874
|
+
const {
|
|
1875
|
+
user,
|
|
1876
|
+
updateProfile,
|
|
1877
|
+
changePassword,
|
|
1878
|
+
changeEmail,
|
|
1879
|
+
loadingUpdateProfile,
|
|
1880
|
+
loadingChangePassword,
|
|
1881
|
+
loadingChangeEmail
|
|
1882
|
+
} = useProfile();
|
|
1883
|
+
|
|
1884
|
+
// Password form
|
|
1885
|
+
const passwordForm = useForm({
|
|
1886
|
+
initialValues: {
|
|
1887
|
+
currentPassword: '',
|
|
1888
|
+
newPassword: '',
|
|
1889
|
+
confirmPassword: ''
|
|
1890
|
+
},
|
|
1891
|
+
validate: {
|
|
1892
|
+
currentPassword: v => !v ? labels.currentPasswordRequired || 'Senha atual obrigatória' : null,
|
|
1893
|
+
newPassword: v => {
|
|
1894
|
+
if (!v) return labels.newPasswordRequired || 'Nova senha obrigatória';
|
|
1895
|
+
if (v.length < 8) return labels.passwordMinLength || 'Mínimo 8 caracteres';
|
|
1896
|
+
return null;
|
|
1897
|
+
},
|
|
1898
|
+
confirmPassword: (v, values) => v !== values.newPassword ? labels.passwordMismatch || 'Senhas não coincidem' : null
|
|
1899
|
+
}
|
|
1900
|
+
});
|
|
1901
|
+
|
|
1902
|
+
// Email form
|
|
1903
|
+
const emailForm = useForm({
|
|
1904
|
+
initialValues: {
|
|
1905
|
+
newEmail: ''
|
|
1906
|
+
},
|
|
1907
|
+
validate: {
|
|
1908
|
+
newEmail: v => {
|
|
1909
|
+
if (!v) return labels.emailRequired || 'Email obrigatório';
|
|
1910
|
+
if (!/^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(v)) return labels.emailInvalid || 'Email inválido';
|
|
1911
|
+
return null;
|
|
1912
|
+
}
|
|
1913
|
+
}
|
|
1914
|
+
});
|
|
1915
|
+
|
|
1916
|
+
// Name form
|
|
1917
|
+
const nameForm = useForm({
|
|
1918
|
+
initialValues: {
|
|
1919
|
+
name: ''
|
|
1920
|
+
},
|
|
1921
|
+
validate: {
|
|
1922
|
+
name: v => !v ? labels.nameRequired || 'Nome obrigatório' : null
|
|
1923
|
+
}
|
|
1924
|
+
});
|
|
1925
|
+
|
|
1926
|
+
// Avatar state (base64)
|
|
1927
|
+
const [avatarPreview, setAvatarPreview] = useState(null);
|
|
1928
|
+
const [avatarFile, setAvatarFile] = useState(null);
|
|
1929
|
+
|
|
1930
|
+
// Handle file selection and convert to base64
|
|
1931
|
+
const handleAvatarFileChange = file => {
|
|
1932
|
+
if (!file) {
|
|
1933
|
+
setAvatarPreview(null);
|
|
1934
|
+
setAvatarFile(null);
|
|
1935
|
+
return;
|
|
1936
|
+
}
|
|
1937
|
+
|
|
1938
|
+
// Validate file type
|
|
1939
|
+
if (!file.type.startsWith('image/')) {
|
|
1940
|
+
notifications.show({
|
|
1941
|
+
title: labels.errorTitle || 'Erro',
|
|
1942
|
+
message: labels.avatarInvalidType || 'Por favor, selecione uma imagem válida',
|
|
1943
|
+
color: 'red'
|
|
1944
|
+
});
|
|
1945
|
+
return;
|
|
1946
|
+
}
|
|
1947
|
+
|
|
1948
|
+
// Validate file size
|
|
1949
|
+
if (file.size > maxAvatarSize) {
|
|
1950
|
+
notifications.show({
|
|
1951
|
+
title: labels.errorTitle || 'Erro',
|
|
1952
|
+
message: labels.avatarTooLarge || `Imagem muito grande. Máximo ${Math.round(maxAvatarSize / 1024)}KB.`,
|
|
1953
|
+
color: 'red'
|
|
1954
|
+
});
|
|
1955
|
+
return;
|
|
1956
|
+
}
|
|
1957
|
+
setAvatarFile(file);
|
|
1958
|
+
|
|
1959
|
+
// Convert to base64
|
|
1960
|
+
const reader = new FileReader();
|
|
1961
|
+
reader.onloadend = () => {
|
|
1962
|
+
setAvatarPreview(reader.result);
|
|
1963
|
+
};
|
|
1964
|
+
reader.readAsDataURL(file);
|
|
1965
|
+
};
|
|
1966
|
+
|
|
1967
|
+
// Populate forms when user data is available or section opens
|
|
1968
|
+
useEffect(() => {
|
|
1969
|
+
if (editingSection === 'name' && user?.name) {
|
|
1970
|
+
nameForm.setValues({
|
|
1971
|
+
name: user.name
|
|
1972
|
+
});
|
|
1973
|
+
}
|
|
1974
|
+
if (editingSection === 'avatar' && user?.image) {
|
|
1975
|
+
setAvatarPreview(user.image);
|
|
1976
|
+
}
|
|
1977
|
+
}, [editingSection, user]);
|
|
1978
|
+
const handleToggleSection = section => {
|
|
1979
|
+
if (editingSection === section) {
|
|
1980
|
+
setEditingSection(null);
|
|
1981
|
+
passwordForm.reset();
|
|
1982
|
+
emailForm.reset();
|
|
1983
|
+
nameForm.reset();
|
|
1984
|
+
setAvatarPreview(null);
|
|
1985
|
+
setAvatarFile(null);
|
|
1986
|
+
} else {
|
|
1987
|
+
setEditingSection(section);
|
|
1988
|
+
}
|
|
1989
|
+
};
|
|
1990
|
+
const handleChangePassword = async values => {
|
|
1991
|
+
try {
|
|
1992
|
+
await changePassword(values.currentPassword, values.newPassword);
|
|
1993
|
+
notifications.show({
|
|
1994
|
+
title: labels.successTitle || 'Sucesso',
|
|
1995
|
+
message: labels.passwordChanged || 'Senha alterada com sucesso',
|
|
1996
|
+
color: 'green'
|
|
1997
|
+
});
|
|
1998
|
+
passwordForm.reset();
|
|
1999
|
+
setEditingSection(null);
|
|
2000
|
+
onPasswordChange?.();
|
|
2001
|
+
} catch (error) {
|
|
2002
|
+
notifications.show({
|
|
2003
|
+
title: labels.errorTitle || 'Erro',
|
|
2004
|
+
message: error.message || labels.passwordChangeFailed || 'Falha ao alterar senha',
|
|
2005
|
+
color: 'red'
|
|
2006
|
+
});
|
|
2007
|
+
onError?.(error);
|
|
2008
|
+
}
|
|
2009
|
+
};
|
|
2010
|
+
const handleChangeEmail = async values => {
|
|
2011
|
+
try {
|
|
2012
|
+
await changeEmail(values.newEmail);
|
|
2013
|
+
notifications.show({
|
|
2014
|
+
title: labels.successTitle || 'Sucesso',
|
|
2015
|
+
message: labels.emailVerificationSent || 'Verifique seu novo email para confirmar a alteração',
|
|
2016
|
+
color: 'green'
|
|
2017
|
+
});
|
|
2018
|
+
emailForm.reset();
|
|
2019
|
+
setEditingSection(null);
|
|
2020
|
+
onEmailChange?.(values.newEmail);
|
|
2021
|
+
} catch (error) {
|
|
2022
|
+
notifications.show({
|
|
2023
|
+
title: labels.errorTitle || 'Erro',
|
|
2024
|
+
message: error.message || labels.emailChangeFailed || 'Falha ao alterar email',
|
|
2025
|
+
color: 'red'
|
|
2026
|
+
});
|
|
2027
|
+
onError?.(error);
|
|
2028
|
+
}
|
|
2029
|
+
};
|
|
2030
|
+
const handleChangeName = async values => {
|
|
2031
|
+
try {
|
|
2032
|
+
await updateProfile({
|
|
2033
|
+
name: values.name
|
|
2034
|
+
});
|
|
2035
|
+
notifications.show({
|
|
2036
|
+
title: labels.successTitle || 'Sucesso',
|
|
2037
|
+
message: labels.nameUpdated || 'Nome atualizado com sucesso',
|
|
2038
|
+
color: 'green'
|
|
2039
|
+
});
|
|
2040
|
+
nameForm.reset();
|
|
2041
|
+
setEditingSection(null);
|
|
2042
|
+
onProfileUpdate?.({
|
|
2043
|
+
name: values.name
|
|
2044
|
+
});
|
|
2045
|
+
} catch (error) {
|
|
2046
|
+
notifications.show({
|
|
2047
|
+
title: labels.errorTitle || 'Erro',
|
|
2048
|
+
message: error.message || labels.nameUpdateFailed || 'Falha ao atualizar nome',
|
|
2049
|
+
color: 'red'
|
|
2050
|
+
});
|
|
2051
|
+
onError?.(error);
|
|
2052
|
+
}
|
|
2053
|
+
};
|
|
2054
|
+
const handleChangeAvatar = async () => {
|
|
2055
|
+
if (!avatarPreview) {
|
|
2056
|
+
notifications.show({
|
|
2057
|
+
title: labels.errorTitle || 'Erro',
|
|
2058
|
+
message: labels.avatarRequired || 'Selecione uma imagem',
|
|
2059
|
+
color: 'red'
|
|
2060
|
+
});
|
|
2061
|
+
return;
|
|
2062
|
+
}
|
|
2063
|
+
try {
|
|
2064
|
+
await updateProfile({
|
|
2065
|
+
image: avatarPreview
|
|
2066
|
+
});
|
|
2067
|
+
notifications.show({
|
|
2068
|
+
title: labels.successTitle || 'Sucesso',
|
|
2069
|
+
message: labels.avatarUpdated || 'Foto de perfil atualizada com sucesso',
|
|
2070
|
+
color: 'green'
|
|
2071
|
+
});
|
|
2072
|
+
setAvatarPreview(null);
|
|
2073
|
+
setAvatarFile(null);
|
|
2074
|
+
setEditingSection(null);
|
|
2075
|
+
onProfileUpdate?.({
|
|
2076
|
+
image: avatarPreview
|
|
2077
|
+
});
|
|
2078
|
+
} catch (error) {
|
|
2079
|
+
notifications.show({
|
|
2080
|
+
title: labels.errorTitle || 'Erro',
|
|
2081
|
+
message: error.message || labels.avatarUpdateFailed || 'Falha ao atualizar foto',
|
|
2082
|
+
color: 'red'
|
|
2083
|
+
});
|
|
2084
|
+
onError?.(error);
|
|
2085
|
+
}
|
|
2086
|
+
};
|
|
2087
|
+
const handleRemoveAvatar = async () => {
|
|
2088
|
+
try {
|
|
2089
|
+
await updateProfile({
|
|
2090
|
+
image: ''
|
|
2091
|
+
});
|
|
2092
|
+
notifications.show({
|
|
2093
|
+
title: labels.successTitle || 'Sucesso',
|
|
2094
|
+
message: labels.avatarRemoved || 'Foto de perfil removida',
|
|
2095
|
+
color: 'green'
|
|
2096
|
+
});
|
|
2097
|
+
setAvatarPreview(null);
|
|
2098
|
+
setAvatarFile(null);
|
|
2099
|
+
setEditingSection(null);
|
|
2100
|
+
onProfileUpdate?.({
|
|
2101
|
+
image: ''
|
|
2102
|
+
});
|
|
2103
|
+
} catch (error) {
|
|
2104
|
+
notifications.show({
|
|
2105
|
+
title: labels.errorTitle || 'Erro',
|
|
2106
|
+
message: error.message || labels.avatarRemoveFailed || 'Falha ao remover foto',
|
|
2107
|
+
color: 'red'
|
|
2108
|
+
});
|
|
2109
|
+
onError?.(error);
|
|
2110
|
+
}
|
|
2111
|
+
};
|
|
2112
|
+
|
|
2113
|
+
// Reusable Section Header
|
|
2114
|
+
const SectionHeader = ({
|
|
2115
|
+
icon: Icon,
|
|
2116
|
+
sectionTitle,
|
|
2117
|
+
description
|
|
2118
|
+
}) => /*#__PURE__*/jsxs(Group, {
|
|
2119
|
+
gap: "sm",
|
|
2120
|
+
mb: "lg",
|
|
2121
|
+
children: [/*#__PURE__*/jsx(ThemeIcon, {
|
|
2122
|
+
size: 36,
|
|
2123
|
+
variant: "light",
|
|
2124
|
+
children: /*#__PURE__*/jsx(Icon, {
|
|
2125
|
+
size: 18
|
|
2126
|
+
})
|
|
2127
|
+
}), /*#__PURE__*/jsxs(Box, {
|
|
2128
|
+
children: [/*#__PURE__*/jsx(Text, {
|
|
2129
|
+
fw: 600,
|
|
2130
|
+
size: "md",
|
|
2131
|
+
children: sectionTitle
|
|
2132
|
+
}), description && /*#__PURE__*/jsx(Text, {
|
|
2133
|
+
size: "xs",
|
|
2134
|
+
c: "dimmed",
|
|
2135
|
+
children: description
|
|
2136
|
+
})]
|
|
2137
|
+
})]
|
|
2138
|
+
});
|
|
2139
|
+
|
|
2140
|
+
// Reusable Row component
|
|
2141
|
+
const SettingRow = ({
|
|
2142
|
+
label,
|
|
2143
|
+
children,
|
|
2144
|
+
action,
|
|
2145
|
+
actionLabel,
|
|
2146
|
+
onClick,
|
|
2147
|
+
expanded
|
|
2148
|
+
}) => /*#__PURE__*/jsx(Card, {
|
|
2149
|
+
p: "xs",
|
|
2150
|
+
children: /*#__PURE__*/jsxs(Group, {
|
|
2151
|
+
justify: "space-between",
|
|
2152
|
+
wrap: "nowrap",
|
|
2153
|
+
children: [/*#__PURE__*/jsxs(Group, {
|
|
2154
|
+
gap: "xl",
|
|
2155
|
+
wrap: "nowrap",
|
|
2156
|
+
flex: 1,
|
|
2157
|
+
children: [/*#__PURE__*/jsx(Text, {
|
|
2158
|
+
size: "sm",
|
|
2159
|
+
c: "dimmed",
|
|
2160
|
+
w: 100,
|
|
2161
|
+
style: {
|
|
2162
|
+
flexShrink: 0
|
|
2163
|
+
},
|
|
2164
|
+
children: label
|
|
2165
|
+
}), /*#__PURE__*/jsx(Box, {
|
|
2166
|
+
flex: 1,
|
|
2167
|
+
children: children
|
|
2168
|
+
})]
|
|
2169
|
+
}), action && /*#__PURE__*/jsx(Tooltip, {
|
|
2170
|
+
label: actionLabel || action,
|
|
2171
|
+
position: "left",
|
|
2172
|
+
children: /*#__PURE__*/jsx(Anchor, {
|
|
2173
|
+
size: "sm",
|
|
2174
|
+
fw: 500,
|
|
2175
|
+
onClick: onClick,
|
|
2176
|
+
c: expanded ? 'red' : 'blue',
|
|
2177
|
+
children: expanded ? labels.cancel || 'Cancel' : action
|
|
2178
|
+
})
|
|
2179
|
+
})]
|
|
2180
|
+
})
|
|
2181
|
+
});
|
|
2182
|
+
return /*#__PURE__*/jsxs(Modal, {
|
|
2183
|
+
opened: opened,
|
|
2184
|
+
onClose: onClose,
|
|
2185
|
+
size: 550,
|
|
2186
|
+
withCloseButton: true,
|
|
2187
|
+
radius: "lg",
|
|
2188
|
+
overlayProps: {
|
|
2189
|
+
backgroundOpacity: 0.55,
|
|
2190
|
+
blur: 3
|
|
2191
|
+
},
|
|
2192
|
+
title: /*#__PURE__*/jsxs(Group, {
|
|
2193
|
+
gap: "md",
|
|
2194
|
+
children: [/*#__PURE__*/jsx(ThemeIcon, {
|
|
2195
|
+
size: 44,
|
|
2196
|
+
variant: "light",
|
|
2197
|
+
children: /*#__PURE__*/jsx(IconUserCircle, {
|
|
2198
|
+
size: 24
|
|
2199
|
+
})
|
|
2200
|
+
}), /*#__PURE__*/jsxs(Box, {
|
|
2201
|
+
children: [/*#__PURE__*/jsx(Title, {
|
|
2202
|
+
order: 4,
|
|
2203
|
+
children: title
|
|
2204
|
+
}), /*#__PURE__*/jsx(Text, {
|
|
2205
|
+
size: "sm",
|
|
2206
|
+
c: "dimmed",
|
|
2207
|
+
children: subtitle
|
|
2208
|
+
})]
|
|
2209
|
+
})]
|
|
2210
|
+
}),
|
|
2211
|
+
...modalProps,
|
|
2212
|
+
children: [/*#__PURE__*/jsx(Divider, {
|
|
2213
|
+
mb: "md"
|
|
2214
|
+
}), (showAvatar || showName || showEmail) && /*#__PURE__*/jsxs(Box, {
|
|
2215
|
+
mb: "xl",
|
|
2216
|
+
children: [/*#__PURE__*/jsx(SectionHeader, {
|
|
2217
|
+
icon: IconUser,
|
|
2218
|
+
sectionTitle: labels.profileSection || 'Profile',
|
|
2219
|
+
description: labels.profileDescription || 'Your personal information'
|
|
2220
|
+
}), /*#__PURE__*/jsxs(Stack, {
|
|
2221
|
+
gap: "sm",
|
|
2222
|
+
children: [showAvatar && /*#__PURE__*/jsxs(Fragment, {
|
|
2223
|
+
children: [/*#__PURE__*/jsx(SettingRow, {
|
|
2224
|
+
label: labels.avatar || 'Avatar',
|
|
2225
|
+
action: labels.update || 'Update',
|
|
2226
|
+
actionLabel: labels.updateAvatar || 'Update your profile picture',
|
|
2227
|
+
onClick: () => handleToggleSection('avatar'),
|
|
2228
|
+
expanded: editingSection === 'avatar',
|
|
2229
|
+
children: /*#__PURE__*/jsx(Group, {
|
|
2230
|
+
gap: "md",
|
|
2231
|
+
children: /*#__PURE__*/jsx(Avatar, {
|
|
2232
|
+
src: user?.image,
|
|
2233
|
+
name: user?.name || user?.email,
|
|
2234
|
+
size: 48,
|
|
2235
|
+
radius: "xl",
|
|
2236
|
+
color: "initials"
|
|
2237
|
+
})
|
|
2238
|
+
})
|
|
2239
|
+
}), /*#__PURE__*/jsx(Collapse, {
|
|
2240
|
+
in: editingSection === 'avatar',
|
|
2241
|
+
children: /*#__PURE__*/jsx(Card, {
|
|
2242
|
+
p: "md",
|
|
2243
|
+
withBorder: true,
|
|
2244
|
+
children: /*#__PURE__*/jsxs(Stack, {
|
|
2245
|
+
gap: "md",
|
|
2246
|
+
align: "center",
|
|
2247
|
+
children: [/*#__PURE__*/jsx("input", {
|
|
2248
|
+
type: "file",
|
|
2249
|
+
id: "avatar-upload",
|
|
2250
|
+
accept: "image/*",
|
|
2251
|
+
style: {
|
|
2252
|
+
display: 'none'
|
|
2253
|
+
},
|
|
2254
|
+
onChange: e => handleAvatarFileChange(e.target.files?.[0])
|
|
2255
|
+
}), /*#__PURE__*/jsx(Tooltip, {
|
|
2256
|
+
label: labels.clickToChange || 'Clique para alterar',
|
|
2257
|
+
position: "bottom",
|
|
2258
|
+
children: /*#__PURE__*/jsxs(Box, {
|
|
2259
|
+
onClick: () => document.getElementById('avatar-upload')?.click(),
|
|
2260
|
+
style: {
|
|
2261
|
+
position: 'relative',
|
|
2262
|
+
cursor: 'pointer',
|
|
2263
|
+
borderRadius: '50%'
|
|
2264
|
+
},
|
|
2265
|
+
children: [/*#__PURE__*/jsx(Avatar, {
|
|
2266
|
+
src: avatarPreview || user?.image,
|
|
2267
|
+
name: user?.name || user?.email,
|
|
2268
|
+
size: 100,
|
|
2269
|
+
radius: 100,
|
|
2270
|
+
color: "initials"
|
|
2271
|
+
}), /*#__PURE__*/jsx(ThemeIcon, {
|
|
2272
|
+
size: 32,
|
|
2273
|
+
radius: "xl",
|
|
2274
|
+
color: "blue",
|
|
2275
|
+
style: {
|
|
2276
|
+
position: 'absolute',
|
|
2277
|
+
bottom: 0,
|
|
2278
|
+
right: 0,
|
|
2279
|
+
border: '2px solid var(--mantine-color-body)'
|
|
2280
|
+
},
|
|
2281
|
+
children: /*#__PURE__*/jsx(IconPhoto, {
|
|
2282
|
+
size: 16
|
|
2283
|
+
})
|
|
2284
|
+
})]
|
|
2285
|
+
})
|
|
2286
|
+
}), /*#__PURE__*/jsx(Text, {
|
|
2287
|
+
size: "xs",
|
|
2288
|
+
c: "dimmed",
|
|
2289
|
+
ta: "center",
|
|
2290
|
+
children: labels.avatarHint || `Máximo ${Math.round(maxAvatarSize / 1024)}KB • JPG, PNG, GIF, WebP`
|
|
2291
|
+
}), /*#__PURE__*/jsxs(Group, {
|
|
2292
|
+
justify: "center",
|
|
2293
|
+
gap: "sm",
|
|
2294
|
+
children: [user?.image && /*#__PURE__*/jsx(Tooltip, {
|
|
2295
|
+
label: labels.removePhoto || 'Remover foto',
|
|
2296
|
+
children: /*#__PURE__*/jsx(Button, {
|
|
2297
|
+
variant: "light",
|
|
2298
|
+
color: "red",
|
|
2299
|
+
size: "xs",
|
|
2300
|
+
onClick: handleRemoveAvatar,
|
|
2301
|
+
loading: loadingUpdateProfile,
|
|
2302
|
+
leftSection: /*#__PURE__*/jsx(IconTrash, {
|
|
2303
|
+
size: 14
|
|
2304
|
+
}),
|
|
2305
|
+
children: labels.remove || 'Remover'
|
|
2306
|
+
})
|
|
2307
|
+
}), /*#__PURE__*/jsx(Button, {
|
|
2308
|
+
variant: "default",
|
|
2309
|
+
size: "xs",
|
|
2310
|
+
onClick: () => handleToggleSection('avatar'),
|
|
2311
|
+
children: labels.cancel || 'Cancelar'
|
|
2312
|
+
}), /*#__PURE__*/jsx(Button, {
|
|
2313
|
+
size: "xs",
|
|
2314
|
+
loading: loadingUpdateProfile,
|
|
2315
|
+
leftSection: /*#__PURE__*/jsx(IconCheck, {
|
|
2316
|
+
size: 14
|
|
2317
|
+
}),
|
|
2318
|
+
onClick: handleChangeAvatar,
|
|
2319
|
+
disabled: !avatarPreview || avatarPreview === user?.image,
|
|
2320
|
+
children: labels.save || 'Salvar'
|
|
2321
|
+
})]
|
|
2322
|
+
})]
|
|
2323
|
+
})
|
|
2324
|
+
})
|
|
2325
|
+
})]
|
|
2326
|
+
}), showName && /*#__PURE__*/jsxs(Fragment, {
|
|
2327
|
+
children: [/*#__PURE__*/jsx(SettingRow, {
|
|
2328
|
+
label: labels.name || 'Nome',
|
|
2329
|
+
action: labels.change || 'Change',
|
|
2330
|
+
actionLabel: labels.changeName || 'Change your display name',
|
|
2331
|
+
onClick: () => handleToggleSection('name'),
|
|
2332
|
+
expanded: editingSection === 'name',
|
|
2333
|
+
children: /*#__PURE__*/jsxs(Group, {
|
|
2334
|
+
gap: "xs",
|
|
2335
|
+
children: [/*#__PURE__*/jsx(IconPencil, {
|
|
2336
|
+
size: 16,
|
|
2337
|
+
color: "var(--mantine-color-dimmed)"
|
|
2338
|
+
}), /*#__PURE__*/jsx(Text, {
|
|
2339
|
+
size: "sm",
|
|
2340
|
+
children: user?.name || labels.notDefined || 'Não definido'
|
|
2341
|
+
})]
|
|
2342
|
+
})
|
|
2343
|
+
}), /*#__PURE__*/jsx(Collapse, {
|
|
2344
|
+
in: editingSection === 'name',
|
|
2345
|
+
children: /*#__PURE__*/jsx(Card, {
|
|
2346
|
+
p: "md",
|
|
2347
|
+
withBorder: true,
|
|
2348
|
+
children: /*#__PURE__*/jsx("form", {
|
|
2349
|
+
onSubmit: nameForm.onSubmit(handleChangeName),
|
|
2350
|
+
children: /*#__PURE__*/jsxs(Stack, {
|
|
2351
|
+
gap: "sm",
|
|
2352
|
+
children: [/*#__PURE__*/jsx(TextInput, {
|
|
2353
|
+
label: labels.name || 'Nome',
|
|
2354
|
+
placeholder: labels.namePlaceholder || 'Digite seu nome',
|
|
2355
|
+
leftSection: /*#__PURE__*/jsx(IconUser, {
|
|
2356
|
+
size: 16
|
|
2357
|
+
}),
|
|
2358
|
+
...nameForm.getInputProps('name')
|
|
2359
|
+
}), /*#__PURE__*/jsxs(Group, {
|
|
2360
|
+
justify: "flex-end",
|
|
2361
|
+
gap: "sm",
|
|
2362
|
+
children: [/*#__PURE__*/jsx(Button, {
|
|
2363
|
+
variant: "default",
|
|
2364
|
+
size: "xs",
|
|
2365
|
+
onClick: () => handleToggleSection('name'),
|
|
2366
|
+
leftSection: /*#__PURE__*/jsx(IconX, {
|
|
2367
|
+
size: 14
|
|
2368
|
+
}),
|
|
2369
|
+
children: labels.cancel || 'Cancelar'
|
|
2370
|
+
}), /*#__PURE__*/jsx(Button, {
|
|
2371
|
+
type: "submit",
|
|
2372
|
+
size: "xs",
|
|
2373
|
+
loading: loadingUpdateProfile,
|
|
2374
|
+
leftSection: /*#__PURE__*/jsx(IconCheck, {
|
|
2375
|
+
size: 14
|
|
2376
|
+
}),
|
|
2377
|
+
children: labels.save || 'Salvar'
|
|
2378
|
+
})]
|
|
2379
|
+
})]
|
|
2380
|
+
})
|
|
2381
|
+
})
|
|
2382
|
+
})
|
|
2383
|
+
})]
|
|
2384
|
+
}), showEmail && /*#__PURE__*/jsxs(Fragment, {
|
|
2385
|
+
children: [/*#__PURE__*/jsx(SettingRow, {
|
|
2386
|
+
label: labels.email || 'Email',
|
|
2387
|
+
action: labels.change || 'Change',
|
|
2388
|
+
actionLabel: labels.changeEmail || 'Change your email',
|
|
2389
|
+
onClick: () => handleToggleSection('email'),
|
|
2390
|
+
expanded: editingSection === 'email',
|
|
2391
|
+
children: /*#__PURE__*/jsxs(Group, {
|
|
2392
|
+
gap: "xs",
|
|
2393
|
+
children: [/*#__PURE__*/jsx(IconMail, {
|
|
2394
|
+
size: 16,
|
|
2395
|
+
color: "var(--mantine-color-dimmed)"
|
|
2396
|
+
}), /*#__PURE__*/jsx(Text, {
|
|
2397
|
+
size: "sm",
|
|
2398
|
+
children: user?.email || 'email@exemplo.com'
|
|
2399
|
+
})]
|
|
2400
|
+
})
|
|
2401
|
+
}), /*#__PURE__*/jsx(Collapse, {
|
|
2402
|
+
in: editingSection === 'email',
|
|
2403
|
+
children: /*#__PURE__*/jsx(Card, {
|
|
2404
|
+
p: "md",
|
|
2405
|
+
withBorder: true,
|
|
2406
|
+
children: /*#__PURE__*/jsx("form", {
|
|
2407
|
+
onSubmit: emailForm.onSubmit(handleChangeEmail),
|
|
2408
|
+
children: /*#__PURE__*/jsxs(Stack, {
|
|
2409
|
+
gap: "sm",
|
|
2410
|
+
children: [/*#__PURE__*/jsx(TextInput, {
|
|
2411
|
+
label: labels.newEmail || 'Novo Email',
|
|
2412
|
+
placeholder: labels.newEmailPlaceholder || 'Digite o novo email',
|
|
2413
|
+
leftSection: /*#__PURE__*/jsx(IconMail, {
|
|
2414
|
+
size: 16
|
|
2415
|
+
}),
|
|
2416
|
+
...emailForm.getInputProps('newEmail')
|
|
2417
|
+
}), /*#__PURE__*/jsxs(Group, {
|
|
2418
|
+
justify: "flex-end",
|
|
2419
|
+
gap: "sm",
|
|
2420
|
+
children: [/*#__PURE__*/jsx(Button, {
|
|
2421
|
+
variant: "default",
|
|
2422
|
+
size: "xs",
|
|
2423
|
+
onClick: () => handleToggleSection('email'),
|
|
2424
|
+
leftSection: /*#__PURE__*/jsx(IconX, {
|
|
2425
|
+
size: 14
|
|
2426
|
+
}),
|
|
2427
|
+
children: labels.cancel || 'Cancelar'
|
|
2428
|
+
}), /*#__PURE__*/jsx(Button, {
|
|
2429
|
+
type: "submit",
|
|
2430
|
+
size: "xs",
|
|
2431
|
+
loading: loadingChangeEmail,
|
|
2432
|
+
leftSection: /*#__PURE__*/jsx(IconCheck, {
|
|
2433
|
+
size: 14
|
|
2434
|
+
}),
|
|
2435
|
+
children: labels.save || 'Salvar'
|
|
2436
|
+
})]
|
|
2437
|
+
})]
|
|
2438
|
+
})
|
|
2439
|
+
})
|
|
2440
|
+
})
|
|
2441
|
+
})]
|
|
2442
|
+
})]
|
|
2443
|
+
})]
|
|
2444
|
+
}), showPassword && /*#__PURE__*/jsxs(Box, {
|
|
2445
|
+
mb: "md",
|
|
2446
|
+
children: [/*#__PURE__*/jsx(SectionHeader, {
|
|
2447
|
+
icon: IconShield,
|
|
2448
|
+
sectionTitle: labels.securitySection || 'Security',
|
|
2449
|
+
description: labels.securityDescription || 'Protect your account'
|
|
2450
|
+
}), /*#__PURE__*/jsxs(Stack, {
|
|
2451
|
+
gap: "sm",
|
|
2452
|
+
children: [/*#__PURE__*/jsx(SettingRow, {
|
|
2453
|
+
label: labels.password || 'Password',
|
|
2454
|
+
action: labels.change || 'Change',
|
|
2455
|
+
actionLabel: labels.changePassword || 'Change your password',
|
|
2456
|
+
onClick: () => handleToggleSection('password'),
|
|
2457
|
+
expanded: editingSection === 'password',
|
|
2458
|
+
children: /*#__PURE__*/jsxs(Group, {
|
|
2459
|
+
gap: "xs",
|
|
2460
|
+
children: [/*#__PURE__*/jsx(IconKey, {
|
|
2461
|
+
size: 16,
|
|
2462
|
+
color: "var(--mantine-color-dimmed)"
|
|
2463
|
+
}), /*#__PURE__*/jsx(Text, {
|
|
2464
|
+
size: "sm",
|
|
2465
|
+
c: "dimmed",
|
|
2466
|
+
children: "\u2022\u2022\u2022\u2022\u2022\u2022\u2022\u2022\u2022\u2022\u2022\u2022"
|
|
2467
|
+
})]
|
|
2468
|
+
})
|
|
2469
|
+
}), /*#__PURE__*/jsx(Collapse, {
|
|
2470
|
+
in: editingSection === 'password',
|
|
2471
|
+
children: /*#__PURE__*/jsx(Card, {
|
|
2472
|
+
p: "md",
|
|
2473
|
+
withBorder: true,
|
|
2474
|
+
children: /*#__PURE__*/jsx("form", {
|
|
2475
|
+
onSubmit: passwordForm.onSubmit(handleChangePassword),
|
|
2476
|
+
children: /*#__PURE__*/jsxs(Stack, {
|
|
2477
|
+
gap: "sm",
|
|
2478
|
+
children: [/*#__PURE__*/jsx(PasswordInput, {
|
|
2479
|
+
label: labels.currentPassword || 'Senha Atual',
|
|
2480
|
+
placeholder: labels.currentPasswordPlaceholder || 'Digite sua senha atual',
|
|
2481
|
+
...passwordForm.getInputProps('currentPassword')
|
|
2482
|
+
}), /*#__PURE__*/jsx(PasswordInput, {
|
|
2483
|
+
label: labels.newPassword || 'Nova Senha',
|
|
2484
|
+
placeholder: labels.newPasswordPlaceholder || 'Digite a nova senha',
|
|
2485
|
+
description: labels.passwordDescription || 'Mínimo 8 caracteres',
|
|
2486
|
+
...passwordForm.getInputProps('newPassword')
|
|
2487
|
+
}), /*#__PURE__*/jsx(PasswordInput, {
|
|
2488
|
+
label: labels.confirmPassword || 'Confirmar Nova Senha',
|
|
2489
|
+
placeholder: labels.confirmPasswordPlaceholder || 'Confirme a nova senha',
|
|
2490
|
+
...passwordForm.getInputProps('confirmPassword')
|
|
2491
|
+
}), /*#__PURE__*/jsxs(Group, {
|
|
2492
|
+
justify: "flex-end",
|
|
2493
|
+
gap: "xs",
|
|
2494
|
+
children: [/*#__PURE__*/jsx(Button, {
|
|
2495
|
+
variant: "default",
|
|
2496
|
+
size: "xs",
|
|
2497
|
+
onClick: () => handleToggleSection('password'),
|
|
2498
|
+
leftSection: /*#__PURE__*/jsx(IconX, {
|
|
2499
|
+
size: 14
|
|
2500
|
+
}),
|
|
2501
|
+
children: labels.cancel || 'Cancelar'
|
|
2502
|
+
}), /*#__PURE__*/jsx(Button, {
|
|
2503
|
+
type: "submit",
|
|
2504
|
+
size: "xs",
|
|
2505
|
+
loading: loadingChangePassword,
|
|
2506
|
+
leftSection: /*#__PURE__*/jsx(IconCheck, {
|
|
2507
|
+
size: 14
|
|
2508
|
+
}),
|
|
2509
|
+
children: labels.save || 'Salvar'
|
|
2510
|
+
})]
|
|
2511
|
+
})]
|
|
2512
|
+
})
|
|
2513
|
+
})
|
|
2514
|
+
})
|
|
2515
|
+
})]
|
|
2516
|
+
})]
|
|
2517
|
+
})]
|
|
2518
|
+
});
|
|
2519
|
+
}
|
|
2520
|
+
|
|
2521
|
+
/**
|
|
2522
|
+
* Renderiza children apenas quando o usuário está autenticado
|
|
2523
|
+
* Equivalente ao <SignedIn> do Clerk
|
|
2524
|
+
*/
|
|
2525
|
+
function SignedIn({
|
|
2526
|
+
children
|
|
2527
|
+
}) {
|
|
2528
|
+
const {
|
|
2529
|
+
user,
|
|
2530
|
+
loading
|
|
2531
|
+
} = useAuth();
|
|
2532
|
+
if (loading || !user) return null;
|
|
2533
|
+
return children;
|
|
2534
|
+
}
|
|
2535
|
+
|
|
2536
|
+
/**
|
|
2537
|
+
* Renderiza children apenas quando o usuário NÃO está autenticado
|
|
2538
|
+
* Equivalente ao <SignedOut> do Clerk
|
|
2539
|
+
*/
|
|
2540
|
+
function SignedOut({
|
|
2541
|
+
children
|
|
2542
|
+
}) {
|
|
2543
|
+
const {
|
|
2544
|
+
user,
|
|
2545
|
+
loading
|
|
2546
|
+
} = useAuth();
|
|
2547
|
+
if (loading || user) return null;
|
|
2548
|
+
return children;
|
|
2549
|
+
}
|
|
2550
|
+
|
|
2551
|
+
/**
|
|
2552
|
+
* Renderiza children enquanto a autenticação está carregando
|
|
2553
|
+
* Equivalente ao <ClerkLoading> do Clerk
|
|
2554
|
+
*/
|
|
2555
|
+
function AuthLoading({
|
|
2556
|
+
children
|
|
2557
|
+
}) {
|
|
2558
|
+
const {
|
|
2559
|
+
loading
|
|
2560
|
+
} = useAuth();
|
|
2561
|
+
if (!loading) return null;
|
|
2562
|
+
return children;
|
|
2563
|
+
}
|
|
2564
|
+
|
|
2565
|
+
/**
|
|
2566
|
+
* Renderiza children quando a autenticação terminou de carregar
|
|
2567
|
+
* Equivalente ao <ClerkLoaded> do Clerk
|
|
2568
|
+
*/
|
|
2569
|
+
function AuthLoaded({
|
|
2570
|
+
children
|
|
2571
|
+
}) {
|
|
2572
|
+
const {
|
|
2573
|
+
loading
|
|
2574
|
+
} = useAuth();
|
|
2575
|
+
if (loading) return null;
|
|
2576
|
+
return children;
|
|
2577
|
+
}
|
|
2578
|
+
|
|
2579
|
+
function SignInButton({
|
|
2580
|
+
children,
|
|
2581
|
+
redirectTo = '/login',
|
|
2582
|
+
...props
|
|
2583
|
+
}) {
|
|
2584
|
+
const navigate = useNavigate();
|
|
2585
|
+
return /*#__PURE__*/jsx("button", {
|
|
2586
|
+
onClick: () => navigate(redirectTo),
|
|
2587
|
+
...props,
|
|
2588
|
+
children: children || 'Sign In'
|
|
2589
|
+
});
|
|
2590
|
+
}
|
|
2591
|
+
|
|
2592
|
+
function SignUpButton({
|
|
2593
|
+
children,
|
|
2594
|
+
redirectTo = '/register',
|
|
2595
|
+
...props
|
|
2596
|
+
}) {
|
|
2597
|
+
const navigate = useNavigate();
|
|
2598
|
+
return /*#__PURE__*/jsx("button", {
|
|
2599
|
+
onClick: () => navigate(redirectTo),
|
|
2600
|
+
...props,
|
|
2601
|
+
children: children || 'Sign Up'
|
|
2602
|
+
});
|
|
2603
|
+
}
|
|
2604
|
+
|
|
2605
|
+
function SignOutButton({
|
|
2606
|
+
children,
|
|
2607
|
+
onSignOut,
|
|
2608
|
+
...props
|
|
2609
|
+
}) {
|
|
2610
|
+
const signOut = useSignOut();
|
|
2611
|
+
const handleClick = async () => {
|
|
2612
|
+
await signOut();
|
|
2613
|
+
onSignOut?.();
|
|
2614
|
+
};
|
|
2615
|
+
return /*#__PURE__*/jsx("button", {
|
|
2616
|
+
onClick: handleClick,
|
|
2617
|
+
...props,
|
|
2618
|
+
children: children || 'Sign Out'
|
|
2619
|
+
});
|
|
2620
|
+
}
|
|
2621
|
+
|
|
2622
|
+
export { UserProfile as AccountModal, AuthCard, AuthLoaded, AuthLoading, AuthProvider, ForgotPassword, ForgotPassword as ForgotPasswordForm, MagicLink, MagicLinkCallback, MagicLink as MagicLinkForm, MagicLinkCallback as MagicLinkVerify, Protect, Protect as ProtectedRoute, ResetPassword, ResetPassword as ResetPasswordForm, SignIn, SignInButton, SignIn as SignInForm, SignOutButton, SignUp, SignUpButton, SignUp as SignUpForm, SignedIn, SignedOut, UserProfile, VerifyEmail, VerifyEmail as VerifyEmailCard, changeEmail, changePassword, configure, decodeJWT, forgotPassword, getApplicationInfo, getCurrentUser, getSession, isAuthenticated, refreshToken, resendVerification, resetPassword, sendMagicLink, signIn, signOut, signUp, socialRedirect, updateProfile, useApplicationLogo, useAuth, useAuthLoading, useAuthStore, useCheckToken, useEmailVerification, useMagicLink, usePasswordReset, useUser as useProfile, useSession, useSignIn, useSignOut, useSignUp, useUser, verifyEmail, verifyMagicLink };
|
|
1732
2623
|
//# sourceMappingURL=index.esm.js.map
|