source: src/auth/context/firebase/auth-provider.tsx@ 87c9f1e

main
Last change on this file since 87c9f1e was 87c9f1e, checked in by Naum Shapkarovski <naumshapkarovski@…>, 5 weeks ago

update the seed script. update the prisma schema, use mapping

  • Property mode set to 100644
File size: 5.2 KB
Line 
1'use client';
2
3import { useEffect, useReducer, useCallback, useMemo } from 'react';
4import {
5 signOut,
6 signInWithPopup,
7 onAuthStateChanged,
8 GoogleAuthProvider,
9 GithubAuthProvider,
10 TwitterAuthProvider,
11 sendEmailVerification,
12 sendPasswordResetEmail,
13 signInWithEmailAndPassword,
14 createUserWithEmailAndPassword,
15} from 'firebase/auth';
16import { collection, doc, getDoc, setDoc } from 'firebase/firestore';
17// firebase lib
18import { auth, db } from 'src/lib/firebase';
19import { collections } from 'src/lib/firestore';
20//
21import { AuthContext } from './auth-context';
22import { ActionMapType, AuthStateType, AuthUserType } from '../../types';
23import { getUser } from 'src/api/user';
24
25// ----------------------------------------------------------------------
26
27// NOTE:
28// We only build demo at basic level.
29// Customer will need to do some extra handling yourself if you want to extend the logic and other features...
30
31// ----------------------------------------------------------------------
32
33enum Types {
34 INITIAL = 'INITIAL',
35}
36
37type Payload = {
38 [Types.INITIAL]: {
39 user: AuthUserType;
40 };
41};
42
43type Action = ActionMapType<Payload>[keyof ActionMapType<Payload>];
44
45const initialState: AuthStateType = {
46 user: null,
47 loading: true,
48};
49
50const reducer = (state: AuthStateType, action: Action) => {
51 if (action.type === Types.INITIAL) {
52 return {
53 loading: false,
54 user: action.payload.user,
55 };
56 }
57 return state;
58};
59
60// ----------------------------------------------------------------------
61
62type Props = {
63 children: React.ReactNode;
64};
65
66export function AuthProvider({ children }: Props) {
67 const [state, dispatch] = useReducer(reducer, initialState);
68
69 const initialize = useCallback(() => {
70 try {
71 onAuthStateChanged(auth, async (user) => {
72 if (user) {
73 if (user.emailVerified) {
74 try {
75 const profile = await getUser();
76 console.log('profile', profile);
77
78 dispatch({
79 type: Types.INITIAL,
80 payload: {
81 user: { ...profile, emailVerified: user.emailVerified },
82 },
83 });
84 } catch (error) {
85 console.error('Failed to fetch user profile:', error);
86 dispatch({
87 type: Types.INITIAL,
88 payload: {
89 user: null,
90 },
91 });
92 }
93 } else {
94 dispatch({
95 type: Types.INITIAL,
96 payload: {
97 user: null,
98 },
99 });
100 }
101 } else {
102 dispatch({
103 type: Types.INITIAL,
104 payload: {
105 user: null,
106 },
107 });
108 }
109 });
110 } catch (error) {
111 console.error(error);
112 dispatch({
113 type: Types.INITIAL,
114 payload: {
115 user: null,
116 },
117 });
118 }
119 }, []);
120
121 useEffect(() => {
122 initialize();
123 }, [initialize]);
124
125 // LOGIN
126 const login = useCallback(async (email: string, password: string) => {
127 await signInWithEmailAndPassword(auth, email, password);
128 }, []);
129
130 const loginWithGoogle = useCallback(async () => {
131 const provider = new GoogleAuthProvider();
132
133 await signInWithPopup(auth, provider);
134 }, []);
135
136 const loginWithGithub = useCallback(async () => {
137 const provider = new GithubAuthProvider();
138
139 await signInWithPopup(auth, provider);
140 }, []);
141
142 const loginWithTwitter = useCallback(async () => {
143 const provider = new TwitterAuthProvider();
144
145 await signInWithPopup(auth, provider);
146 }, []);
147
148 // REGISTER
149 const register = useCallback(
150 async (email: string, password: string, firstName: string, lastName: string) => {
151 // const newUser = await createUserWithEmailAndPassword(auth, email, password);
152 // await sendEmailVerification(newUser.user);
153 // const userProfile = doc(collection(db, collections.administrator), newUser.user?.uid);
154 // await setDoc(userProfile, {
155 // uid: newUser.user?.uid,
156 // email,
157 // displayName: `${firstName} ${lastName}`,
158 // });
159 },
160 []
161 );
162
163 // LOGOUT
164 const logout = useCallback(async () => {
165 await signOut(auth);
166 }, []);
167
168 // FORGOT PASSWORD
169 const forgotPassword = useCallback(async (email: string) => {
170 await sendPasswordResetEmail(auth, email);
171 }, []);
172
173 // ----------------------------------------------------------------------
174
175 const checkAuthenticated = state.user?.emailVerified ? 'authenticated' : 'unauthenticated';
176
177 const status = state.loading ? 'loading' : checkAuthenticated;
178
179 const memoizedValue = useMemo(
180 () => ({
181 user: state.user,
182 method: 'firebase',
183 loading: status === 'loading',
184 authenticated: status === 'authenticated',
185 unauthenticated: status === 'unauthenticated',
186 //
187 login,
188 logout,
189 register,
190 forgotPassword,
191 loginWithGoogle,
192 loginWithGithub,
193 loginWithTwitter,
194 }),
195 [
196 status,
197 state.user,
198 //
199 login,
200 logout,
201 register,
202 forgotPassword,
203 loginWithGithub,
204 loginWithGoogle,
205 loginWithTwitter,
206 ]
207 );
208
209 return <AuthContext.Provider value={memoizedValue}>{children}</AuthContext.Provider>;
210}
Note: See TracBrowser for help on using the repository browser.