Add some css animation to make the project prettier #7
@@ -1,147 +1,147 @@
|
|||||||
import { useState, useEffect } from 'react';
|
import { useState, useEffect } from 'react';
|
||||||
import { Mail, Lock, LogIn } from 'lucide-react';
|
import { Mail, Lock, LogIn } from 'lucide-react';
|
||||||
import { useAuth } from '../contexts/AuthContext';
|
import { useAuth } from '../contexts/AuthContext';
|
||||||
import { Card } from '../types';
|
import { Card } from '../types';
|
||||||
import { getRandomCards } from '../services/api';
|
import { getRandomCards } from '../services/api';
|
||||||
|
|
||||||
export default function LoginForm() {
|
export default function LoginForm() {
|
||||||
const [email, setEmail] = useState('');
|
const [email, setEmail] = useState('');
|
||||||
const [password, setPassword] = useState('');
|
const [password, setPassword] = useState('');
|
||||||
const [isSignUp, setIsSignUp] = useState(false);
|
const [isSignUp, setIsSignUp] = useState(false);
|
||||||
const [error, setError] = useState<string | null>(null);
|
const [error, setError] = useState<string | null>(null);
|
||||||
const { signIn, signUp } = useAuth();
|
const { signIn, signUp } = useAuth();
|
||||||
const [cards, setCards] = useState<Card[]>([]);
|
const [cards, setCards] = useState<Card[]>([]);
|
||||||
const [loading, setLoading] = useState(true);
|
const [loading, setLoading] = useState(true);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const loadCards = async () => {
|
const loadCards = async () => {
|
||||||
try {
|
try {
|
||||||
const randomCards = await getRandomCards(6);
|
const randomCards = await getRandomCards(6);
|
||||||
setCards(randomCards);
|
setCards(randomCards);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Failed to load cards:', error);
|
console.error('Failed to load cards:', error);
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
loadCards();
|
loadCards();
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const handleSubmit = async (e: React.FormEvent) => {
|
const handleSubmit = async (e: React.FormEvent) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
setError(null);
|
setError(null);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (isSignUp) {
|
if (isSignUp) {
|
||||||
await signUp(email, password);
|
await signUp(email, password);
|
||||||
} else {
|
} else {
|
||||||
await signIn(email, password);
|
await signIn(email, password);
|
||||||
}
|
}
|
||||||
window.location.href = '/'; // Redirect to home after successful login
|
window.location.href = '/'; // Redirect to home after successful login
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
setError(error instanceof Error ? error.message : 'An error occurred');
|
setError(error instanceof Error ? error.message : 'An error occurred');
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
if (loading) {
|
if (loading) {
|
||||||
return <div className="animate-pulse h-96 bg-gray-700/50 rounded-lg"></div>;
|
return <div className="animate-pulse h-96 bg-gray-700/50 rounded-lg"></div>;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="relative min-h-screen flex items-center justify-center p-6 overflow-hidden">
|
<div className="relative min-h-screen flex items-center justify-center p-6 overflow-hidden">
|
||||||
{/* Animated Background */}
|
{/* Animated Background */}
|
||||||
<div className="absolute inset-0 overflow-hidden">
|
<div className="absolute inset-0 overflow-hidden">
|
||||||
<div
|
<div
|
||||||
className="flex animate-slide"
|
className="flex animate-slide"
|
||||||
style={{
|
style={{
|
||||||
width: `${cards.length * 100}%`,
|
width: `${cards.length * 100}%`,
|
||||||
animation: 'slide 60s linear infinite'
|
animation: 'slide 60s linear infinite'
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{[...cards, ...cards].map((card, index) => (
|
{[...cards, ...cards].map((card, index) => (
|
||||||
<div
|
<div
|
||||||
key={`${card.id}-${index}`}
|
key={`${card.id}-${index}`}
|
||||||
className="relative w-full h-screen"
|
className="relative w-full h-screen"
|
||||||
style={{
|
style={{
|
||||||
width: `${100 / (cards.length * 2)}%`
|
width: `${100 / (cards.length * 2)}%`
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className="absolute inset-0 bg-cover bg-center transform transition-transform duration-1000"
|
className="absolute inset-0 bg-cover bg-center transform transition-transform duration-1000"
|
||||||
style={{
|
style={{
|
||||||
backgroundImage: `url(${card.image_uris?.normal})`,
|
backgroundImage: `url(${card.image_uris?.normal})`,
|
||||||
filter: 'blur(8px) brightness(0.4)',
|
filter: 'blur(8px) brightness(0.4)',
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Login Form */}
|
{/* Login Form */}
|
||||||
<div className="relative z-10 bg-gray-900/80 p-8 rounded-lg shadow-xl backdrop-blur-sm w-full max-w-md">
|
<div className="relative z-10 bg-gray-900/80 p-8 rounded-lg shadow-xl backdrop-blur-sm w-full max-w-md glass-effect animate-scale-in">
|
||||||
<h2 className="text-3xl font-bold text-orange-500 mb-6 text-center">
|
<h2 className="text-3xl font-bold text-orange-500 mb-6 text-center animate-bounce-in">
|
||||||
Deckerr
|
Deckerr
|
||||||
</h2>
|
</h2>
|
||||||
|
|
||||||
{error && (
|
{error && (
|
||||||
<div className="mb-4 p-3 bg-red-500/10 border border-red-500 rounded text-red-500">
|
<div className="mb-4 p-3 bg-red-500/10 border border-red-500 rounded text-red-500 animate-fade-in">
|
||||||
{error}
|
{error}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<form onSubmit={handleSubmit} className="space-y-6">
|
<form onSubmit={handleSubmit} className="space-y-6">
|
||||||
<div>
|
<div>
|
||||||
<label className="block text-sm font-medium text-gray-300 mb-2">
|
<label className="block text-sm font-medium text-gray-300 mb-2">
|
||||||
Email
|
Email
|
||||||
</label>
|
</label>
|
||||||
<div className="relative">
|
<div className="relative">
|
||||||
<Mail className="absolute left-3 top-1/2 transform -translate-y-1/2 text-gray-400" size={20} />
|
<Mail className="absolute left-3 top-1/2 transform -translate-y-1/2 text-gray-400" size={20} />
|
||||||
<input
|
<input
|
||||||
type="email"
|
type="email"
|
||||||
value={email}
|
value={email}
|
||||||
onChange={(e) => setEmail(e.target.value)}
|
onChange={(e) => setEmail(e.target.value)}
|
||||||
className="w-full pl-10 pr-4 py-2 bg-gray-800/50 border border-gray-600 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-transparent text-white"
|
className="w-full pl-10 pr-4 py-2 bg-gray-800/50 border border-gray-600 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-transparent text-white transition-smooth"
|
||||||
placeholder="Enter your email"
|
placeholder="Enter your email"
|
||||||
required
|
required
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<label className="block text-sm font-medium text-gray-300 mb-2">
|
<label className="block text-sm font-medium text-gray-300 mb-2">
|
||||||
Password
|
Password
|
||||||
</label>
|
</label>
|
||||||
<div className="relative">
|
<div className="relative">
|
||||||
<Lock className="absolute left-3 top-1/2 transform -translate-y-1/2 text-gray-400" size={20} />
|
<Lock className="absolute left-3 top-1/2 transform -translate-y-1/2 text-gray-400" size={20} />
|
||||||
<input
|
<input
|
||||||
type="password"
|
type="password"
|
||||||
value={password}
|
value={password}
|
||||||
onChange={(e) => setPassword(e.target.value)}
|
onChange={(e) => setPassword(e.target.value)}
|
||||||
className="w-full pl-10 pr-4 py-2 bg-gray-800/50 border border-gray-600 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-transparent text-white"
|
className="w-full pl-10 pr-4 py-2 bg-gray-800/50 border border-gray-600 rounded-lg focus:ring-2 focus:ring-blue-500 focus:border-transparent text-white transition-smooth"
|
||||||
placeholder="Enter your password"
|
placeholder="Enter your password"
|
||||||
required
|
required
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<button
|
<button
|
||||||
type="submit"
|
type="submit"
|
||||||
className="w-full flex items-center justify-center gap-2 bg-blue-600 hover:bg-blue-700 text-white font-semibold py-2 px-4 rounded-lg transition duration-200"
|
className="w-full flex items-center justify-center gap-2 bg-blue-600 hover:bg-blue-700 text-white font-semibold py-2 px-4 rounded-lg btn-ripple glow-on-hover transition-smooth"
|
||||||
>
|
>
|
||||||
<LogIn size={20} />
|
<LogIn size={20} />
|
||||||
{isSignUp ? 'Sign Up' : 'Sign In'}
|
{isSignUp ? 'Sign Up' : 'Sign In'}
|
||||||
</button>
|
</button>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
<div className="mt-4 text-center">
|
<div className="mt-4 text-center">
|
||||||
<button
|
<button
|
||||||
onClick={() => setIsSignUp(!isSignUp)}
|
onClick={() => setIsSignUp(!isSignUp)}
|
||||||
className="text-blue-400 hover:text-blue-300"
|
className="text-blue-400 hover:text-blue-300 transition-smooth"
|
||||||
>
|
>
|
||||||
{isSignUp ? 'Already have an account? Sign In' : 'Need an account? Sign Up'}
|
{isSignUp ? 'Already have an account? Sign In' : 'Need an account? Sign Up'}
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user