Proceso login con google
This commit is contained in:
@@ -1,87 +1,99 @@
|
|||||||
import { useState } from 'react';
|
import React, { useState, useEffect, useRef } from 'react';
|
||||||
import { useNavigate, Link } from 'react-router-dom';
|
import { Link, useNavigate, useSearchParams } from 'react-router-dom';
|
||||||
import { useAuth } from '../context/AuthContext';
|
import { useAuth } from '../context/AuthContext';
|
||||||
import { authService } from '../services/authService';
|
import { authService } from '../services/authService';
|
||||||
|
|
||||||
export default function Login() {
|
export default function Login() {
|
||||||
const [email, setEmail] = useState('');
|
const [email, setEmail] = useState('');
|
||||||
const [password, setPassword] = useState('');
|
const [password, setPassword] = useState('');
|
||||||
const [error, setError] = useState('');
|
const [error, setError] = useState('');
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const { login } = useAuth();
|
const { login } = useAuth();
|
||||||
|
const [searchParams] = useSearchParams();
|
||||||
|
const googleLoginProcessed = useRef(false);
|
||||||
|
|
||||||
const handleSubmit = async (e) => {
|
useEffect(() => {
|
||||||
e.preventDefault();
|
const token = searchParams.get('token');
|
||||||
setError('');
|
|
||||||
|
|
||||||
|
if (token && !googleLoginProcessed.current) {
|
||||||
|
googleLoginProcessed.current = true;
|
||||||
|
|
||||||
|
const url = new URL(window.location);
|
||||||
|
url.searchParams.delete('token');
|
||||||
|
window.history.replaceState({}, '', url);
|
||||||
|
|
||||||
|
const fetchUserAndLogin = async () => {
|
||||||
try {
|
try {
|
||||||
const data = await authService.login(email, password);
|
localStorage.setItem('token', token);
|
||||||
|
|
||||||
const userData = {
|
const realUserData = await authService.getCurrentUser();
|
||||||
id: data.user_id,
|
|
||||||
username: data.username,
|
|
||||||
email: email
|
|
||||||
};
|
|
||||||
|
|
||||||
login(userData, data.token);
|
login({ user: realUserData, token });
|
||||||
navigate('/');
|
|
||||||
|
navigate('/', { replace: true });
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
setError(err.response?.data?.detail || 'Error al iniciar sesión. Revisa tus credenciales.');
|
console.error("Error al sincronizar perfil de Google:", err);
|
||||||
|
setError("Error de sincronización con Google. Inténtalo de nuevo.");
|
||||||
|
localStorage.removeItem('token');
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
fetchUserAndLogin();
|
||||||
|
}
|
||||||
|
}, [searchParams, login, navigate]);
|
||||||
|
|
||||||
|
const handleSubmit = async (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
setError('');
|
||||||
|
try {
|
||||||
|
const userData = await authService.login(email, password);
|
||||||
|
login(userData);
|
||||||
|
navigate('/');
|
||||||
|
} catch (err) {
|
||||||
|
setError('Credenciales incorrectas.');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleGoogleLogin = () => {
|
||||||
|
window.location.href = "http://localhost:8000/api/auth/google/login";
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex justify-center mt-4 sm:mt-8">
|
<div className="flex-1 flex items-center justify-center py-4">
|
||||||
<div className="max-w-md w-full bg-white rounded-2xl shadow-sm border border-slate-200 p-8">
|
<div className="max-w-md w-full bg-white p-10 rounded-3xl shadow-sm border border-slate-200">
|
||||||
|
|
||||||
<div className="text-center mb-8">
|
<div className="text-center mb-8">
|
||||||
<h2 className="text-3xl font-bold text-slate-800">Bienvenido</h2>
|
<h2 className="text-3xl font-black text-slate-800 tracking-tight">Deck of Cards</h2>
|
||||||
<p className="text-slate-500 mt-2">Inicia sesión para guardar tus espectros difusos</p>
|
<p className="text-slate-500 mt-2">Accede a tu panel de control</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{error && (
|
{error && (
|
||||||
<div className="bg-red-50 text-red-600 p-3 rounded-lg mb-6 text-sm font-medium border border-red-200">
|
<div className="bg-red-50 text-red-600 p-4 rounded-2xl text-sm font-bold mb-6 border border-red-100 text-center">{error}</div>
|
||||||
{error}
|
|
||||||
</div>
|
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<form onSubmit={handleSubmit} className="space-y-5">
|
<form onSubmit={handleSubmit} className="space-y-4">
|
||||||
<div>
|
<div className="space-y-1">
|
||||||
<label className="block text-sm font-semibold text-slate-700 mb-1">Email</label>
|
<label className="text-sm font-bold text-slate-700 ml-1">Email</label>
|
||||||
<input
|
<input type="email" required value={email} onChange={(e) => setEmail(e.target.value)} className="w-full px-5 py-3 rounded-2xl border border-slate-200 focus:ring-2 focus:ring-blue-500 outline-none transition-all bg-slate-50 focus:bg-white" placeholder="correo@ejemplo.com" />
|
||||||
type="email"
|
|
||||||
required
|
|
||||||
autoComplete="email"
|
|
||||||
className="w-full px-4 py-2 border border-slate-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500 outline-none transition-all"
|
|
||||||
value={email}
|
|
||||||
onChange={(e) => setEmail(e.target.value)}
|
|
||||||
placeholder="tu@email.com"
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
|
<div className="space-y-1">
|
||||||
<div>
|
<label className="text-sm font-bold text-slate-700 ml-1">Contraseña</label>
|
||||||
<label className="block text-sm font-semibold text-slate-700 mb-1">Contraseña</label>
|
<input type="password" required value={password} onChange={(e) => setPassword(e.target.value)} className="w-full px-5 py-3 rounded-2xl border border-slate-200 focus:ring-2 focus:ring-blue-500 outline-none transition-all bg-slate-50 focus:bg-white" placeholder="••••••••" />
|
||||||
<input
|
|
||||||
type="password"
|
|
||||||
required
|
|
||||||
autoComplete="current-password"
|
|
||||||
className="w-full px-4 py-2 border border-slate-300 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-blue-500 outline-none transition-all"
|
|
||||||
value={password}
|
|
||||||
onChange={(e) => setPassword(e.target.value)}
|
|
||||||
placeholder="••••••••"
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
|
<button type="submit" className="w-full py-4 bg-blue-600 hover:bg-blue-700 text-white font-bold rounded-2xl transition-all shadow-sm active:scale-95 mt-2">Entrar</button>
|
||||||
<button
|
|
||||||
type="submit"
|
|
||||||
className="w-full py-3 bg-blue-600 text-white font-bold rounded-xl hover:bg-blue-700 transition-colors shadow-md mt-4"
|
|
||||||
>
|
|
||||||
Entrar
|
|
||||||
</button>
|
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
<p className="text-center mt-6 text-sm text-slate-600">
|
<div className="relative my-8">
|
||||||
¿No tienes cuenta? <Link to="/register" className="text-blue-600 font-bold hover:underline">Regístrate aquí</Link>
|
<div className="absolute inset-0 flex items-center"><div className="w-full border-t border-slate-100"></div></div>
|
||||||
</p>
|
<div className="relative flex justify-center text-xs uppercase tracking-widest"><span className="px-3 bg-white text-slate-400 font-bold">O</span></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button type="button" onClick={handleGoogleLogin} className="w-full flex items-center justify-center gap-3 px-4 py-4 border-2 border-slate-100 rounded-2xl bg-white text-slate-700 font-bold hover:bg-slate-50 hover:border-slate-200 transition-all shadow-sm active:scale-95">
|
||||||
|
<svg className="w-5 h-5" viewBox="0 0 24 24"><path d="M22.56 12.25c0-.78-.07-1.53-.2-2.25H12v4.26h5.92c-.26 1.37-1.04 2.53-2.21 3.31v2.77h3.57c2.08-1.92 3.28-4.74 3.28-8.09z" fill="#4285F4" /><path d="M12 23c2.97 0 5.46-.98 7.28-2.66l-3.57-2.77c-.98.66-2.23 1.06-3.71 1.06-2.86 0-5.29-1.93-6.16-4.53H2.18v2.84C3.99 20.53 7.7 23 12 23z" fill="#34A853" /><path d="M5.84 14.09c-.22-.66-.35-1.36-.35-2.09s.13-1.43.35-2.09V7.07H2.18C1.43 8.55 1 10.22 1 12s.43 3.45 1.18 4.93l2.85-2.22.81-.62z" fill="#FBBC05" /><path d="M12 5.38c1.62 0 3.06.56 4.21 1.64l3.15-3.15C17.45 2.09 14.97 1 12 1 7.7 1 3.99 3.47 2.18 7.07l3.66 2.84c.87-2.6 3.3-4.53 6.16-4.53z" fill="#EA4335" /></svg>
|
||||||
|
Continuar con Google
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<p className="mt-8 text-center text-sm text-slate-500 font-medium">¿Nuevo por aquí? <Link to="/register" className="text-blue-600 hover:underline font-extrabold">Crea una cuenta</Link></p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -9,5 +9,10 @@ export const authService = {
|
|||||||
register: async (username, email, password) => {
|
register: async (username, email, password) => {
|
||||||
const response = await api.post('/auth/register', { username, email, password });
|
const response = await api.post('/auth/register', { username, email, password });
|
||||||
return response.data;
|
return response.data;
|
||||||
|
},
|
||||||
|
|
||||||
|
getCurrentUser: async () => {
|
||||||
|
const response = await api.get('/auth/me');
|
||||||
|
return response.data;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
Reference in New Issue
Block a user