'use client'; import { useEffect, useReducer, useCallback, useMemo } from 'react'; import { signOut, signInWithPopup, onAuthStateChanged, GoogleAuthProvider, GithubAuthProvider, TwitterAuthProvider, sendEmailVerification, sendPasswordResetEmail, signInWithEmailAndPassword, createUserWithEmailAndPassword, } from 'firebase/auth'; import { collection, doc, getDoc, setDoc } from 'firebase/firestore'; // firebase lib import { auth, db } from 'src/lib/firebase'; import { collections } from 'src/lib/firestore'; // import { AuthContext } from './auth-context'; import { ActionMapType, AuthStateType, AuthUserType } from '../../types'; import { getUser } from 'src/api/user'; // ---------------------------------------------------------------------- // NOTE: // We only build demo at basic level. // Customer will need to do some extra handling yourself if you want to extend the logic and other features... // ---------------------------------------------------------------------- enum Types { INITIAL = 'INITIAL', } type Payload = { [Types.INITIAL]: { user: AuthUserType; }; }; type Action = ActionMapType[keyof ActionMapType]; const initialState: AuthStateType = { user: null, loading: true, }; const reducer = (state: AuthStateType, action: Action) => { if (action.type === Types.INITIAL) { return { loading: false, user: action.payload.user, }; } return state; }; // ---------------------------------------------------------------------- type Props = { children: React.ReactNode; }; export function AuthProvider({ children }: Props) { const [state, dispatch] = useReducer(reducer, initialState); const initialize = useCallback(() => { try { onAuthStateChanged(auth, async (user) => { if (user) { if (user.emailVerified) { try { const profile = await getUser(); console.log('profile', profile); dispatch({ type: Types.INITIAL, payload: { user: { ...profile, emailVerified: user.emailVerified }, }, }); } catch (error) { console.error('Failed to fetch user profile:', error); dispatch({ type: Types.INITIAL, payload: { user: null, }, }); } } else { dispatch({ type: Types.INITIAL, payload: { user: null, }, }); } } else { dispatch({ type: Types.INITIAL, payload: { user: null, }, }); } }); } catch (error) { console.error(error); dispatch({ type: Types.INITIAL, payload: { user: null, }, }); } }, []); useEffect(() => { initialize(); }, [initialize]); // LOGIN const login = useCallback(async (email: string, password: string) => { await signInWithEmailAndPassword(auth, email, password); }, []); const loginWithGoogle = useCallback(async () => { const provider = new GoogleAuthProvider(); await signInWithPopup(auth, provider); }, []); const loginWithGithub = useCallback(async () => { const provider = new GithubAuthProvider(); await signInWithPopup(auth, provider); }, []); const loginWithTwitter = useCallback(async () => { const provider = new TwitterAuthProvider(); await signInWithPopup(auth, provider); }, []); // REGISTER const register = useCallback( async (email: string, password: string, firstName: string, lastName: string) => { // const newUser = await createUserWithEmailAndPassword(auth, email, password); // await sendEmailVerification(newUser.user); // const userProfile = doc(collection(db, collections.administrator), newUser.user?.uid); // await setDoc(userProfile, { // uid: newUser.user?.uid, // email, // displayName: `${firstName} ${lastName}`, // }); }, [] ); // LOGOUT const logout = useCallback(async () => { await signOut(auth); }, []); // FORGOT PASSWORD const forgotPassword = useCallback(async (email: string) => { await sendPasswordResetEmail(auth, email); }, []); // ---------------------------------------------------------------------- const checkAuthenticated = state.user?.emailVerified ? 'authenticated' : 'unauthenticated'; const status = state.loading ? 'loading' : checkAuthenticated; const memoizedValue = useMemo( () => ({ user: state.user, method: 'firebase', loading: status === 'loading', authenticated: status === 'authenticated', unauthenticated: status === 'unauthenticated', // login, logout, register, forgotPassword, loginWithGoogle, loginWithGithub, loginWithTwitter, }), [ status, state.user, // login, logout, register, forgotPassword, loginWithGithub, loginWithGoogle, loginWithTwitter, ] ); return {children}; }