1 | 'use client';
|
---|
2 |
|
---|
3 | import { useEffect, useReducer, useCallback, useMemo } from 'react';
|
---|
4 | import {
|
---|
5 | signOut,
|
---|
6 | signInWithPopup,
|
---|
7 | onAuthStateChanged,
|
---|
8 | GoogleAuthProvider,
|
---|
9 | GithubAuthProvider,
|
---|
10 | TwitterAuthProvider,
|
---|
11 | sendEmailVerification,
|
---|
12 | sendPasswordResetEmail,
|
---|
13 | signInWithEmailAndPassword,
|
---|
14 | createUserWithEmailAndPassword,
|
---|
15 | } from 'firebase/auth';
|
---|
16 | import { collection, doc, getDoc, setDoc } from 'firebase/firestore';
|
---|
17 | // firebase lib
|
---|
18 | import { auth, db } from 'src/lib/firebase';
|
---|
19 | import { collections } from 'src/lib/firestore';
|
---|
20 | //
|
---|
21 | import { AuthContext } from './auth-context';
|
---|
22 | import { ActionMapType, AuthStateType, AuthUserType } from '../../types';
|
---|
23 | import { 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 |
|
---|
33 | enum Types {
|
---|
34 | INITIAL = 'INITIAL',
|
---|
35 | }
|
---|
36 |
|
---|
37 | type Payload = {
|
---|
38 | [Types.INITIAL]: {
|
---|
39 | user: AuthUserType;
|
---|
40 | };
|
---|
41 | };
|
---|
42 |
|
---|
43 | type Action = ActionMapType<Payload>[keyof ActionMapType<Payload>];
|
---|
44 |
|
---|
45 | const initialState: AuthStateType = {
|
---|
46 | user: null,
|
---|
47 | loading: true,
|
---|
48 | };
|
---|
49 |
|
---|
50 | const 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 |
|
---|
62 | type Props = {
|
---|
63 | children: React.ReactNode;
|
---|
64 | };
|
---|
65 |
|
---|
66 | export 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 | }
|
---|