170 lines
6.0 KiB
TypeScript
Executable File
170 lines
6.0 KiB
TypeScript
Executable File
import { useState } from 'react'
|
|
import { supabase } from '../lib/supabase'
|
|
import { MicrosoftIcon } from './MicrosoftIcon'
|
|
|
|
interface LoginProps {
|
|
onLoginSuccess: () => void
|
|
}
|
|
|
|
export function Login({ onLoginSuccess }: LoginProps) {
|
|
const [email, setEmail] = useState('')
|
|
const [password, setPassword] = useState('')
|
|
const [loading, setLoading] = useState(false)
|
|
const [error, setError] = useState<string | null>(null)
|
|
const enableMicrosoftLogin = import.meta.env.VITE_ENABLE_MICROSOFT_LOGIN === 'true'
|
|
|
|
const handleLogin = async (e: React.FormEvent) => {
|
|
e.preventDefault()
|
|
setLoading(true)
|
|
setError(null)
|
|
|
|
try {
|
|
const { data, error } = await supabase.auth.signInWithPassword({
|
|
email,
|
|
password,
|
|
})
|
|
|
|
if (error) {
|
|
setError(error.message)
|
|
} else if (data.user) {
|
|
onLoginSuccess()
|
|
}
|
|
} catch (err) {
|
|
setError('An unexpected error occurred')
|
|
console.error('Login error:', err)
|
|
} finally {
|
|
setLoading(false)
|
|
}
|
|
}
|
|
|
|
const handleMicrosoftLogin = async () => {
|
|
setLoading(true)
|
|
setError(null)
|
|
|
|
try {
|
|
const { error } = await supabase.auth.signInWithOAuth({
|
|
provider: 'azure',
|
|
options: {
|
|
scopes: 'email openid profile',
|
|
redirectTo: `${window.location.origin}/`,
|
|
},
|
|
})
|
|
|
|
if (error) {
|
|
setError(error.message)
|
|
setLoading(false)
|
|
}
|
|
// If successful, user will be redirected to Microsoft login
|
|
// and then back to the app, so we don't stop loading here
|
|
} catch (err) {
|
|
setError('An unexpected error occurred during Microsoft login')
|
|
console.error('Microsoft login error:', err)
|
|
setLoading(false)
|
|
}
|
|
}
|
|
|
|
return (
|
|
<div className="min-h-screen flex items-center justify-center bg-gray-50 dark:bg-gray-900">
|
|
<div className="max-w-md w-full space-y-8">
|
|
<div>
|
|
<h2 className="mt-6 text-center text-3xl font-extrabold text-gray-900 dark:text-gray-100">
|
|
Sign in to your account
|
|
</h2>
|
|
<p className="mt-2 text-center text-sm text-gray-600 dark:text-gray-400">
|
|
RBAC Authentication System
|
|
</p>
|
|
</div>
|
|
<form className="mt-8 space-y-6" onSubmit={handleLogin}>
|
|
<div className="rounded-md shadow-sm -space-y-px">
|
|
<div>
|
|
<label htmlFor="email" className="sr-only">
|
|
Email address
|
|
</label>
|
|
<input
|
|
id="email"
|
|
name="email"
|
|
type="email"
|
|
autoComplete="email"
|
|
required
|
|
className="appearance-none rounded-none relative block w-full px-3 py-2
|
|
border border-gray-300 dark:border-gray-700
|
|
bg-white dark:bg-gray-800
|
|
placeholder-gray-500 dark:placeholder-gray-400
|
|
text-gray-900 dark:text-gray-100
|
|
rounded-t-md focus:outline-none focus:ring-indigo-500 focus:border-indigo-500 focus:z-10 sm:text-sm"
|
|
placeholder="Email address"
|
|
value={email}
|
|
onChange={(e) => setEmail(e.target.value)}
|
|
/>
|
|
</div>
|
|
<div>
|
|
<label htmlFor="password" className="sr-only">
|
|
Password
|
|
</label>
|
|
<input
|
|
id="password"
|
|
name="password"
|
|
type="password"
|
|
autoComplete="current-password"
|
|
required
|
|
className="appearance-none rounded-none relative block w-full px-3 py-2
|
|
border border-gray-300 dark:border-gray-700
|
|
bg-white dark:bg-gray-800
|
|
placeholder-gray-500 dark:placeholder-gray-400
|
|
text-gray-900 dark:text-gray-100
|
|
rounded-b-md focus:outline-none focus:ring-indigo-500 focus:border-indigo-500 focus:z-10 sm:text-sm"
|
|
placeholder="Password"
|
|
value={password}
|
|
onChange={(e) => setPassword(e.target.value)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
{error && (
|
|
<div className="rounded-md bg-red-50 p-4">
|
|
<div className="text-sm text-red-700">{error}</div>
|
|
</div>
|
|
)}
|
|
|
|
<div>
|
|
<button
|
|
type="submit"
|
|
disabled={loading}
|
|
className="group relative w-full flex justify-center py-2 px-4 border border-transparent text-sm font-medium rounded-md text-white bg-indigo-600 hover:bg-indigo-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 disabled:opacity-50 disabled:cursor-not-allowed"
|
|
>
|
|
{loading ? 'Signing in...' : 'Sign in'}
|
|
</button>
|
|
</div>
|
|
</form>
|
|
|
|
{enableMicrosoftLogin && (
|
|
<>
|
|
<div className="relative">
|
|
<div className="absolute inset-0 flex items-center">
|
|
<div className="w-full border-t border-gray-300 dark:border-gray-700" />
|
|
</div>
|
|
<div className="relative flex justify-center text-sm">
|
|
<span className="px-2 bg-gray-50 dark:bg-gray-900 text-gray-500 dark:text-gray-400">
|
|
Or continue with
|
|
</span>
|
|
</div>
|
|
</div>
|
|
|
|
<div>
|
|
<button
|
|
type="button"
|
|
onClick={handleMicrosoftLogin}
|
|
disabled={loading}
|
|
className="w-full flex items-center justify-center gap-3 py-2 px-4 border border-gray-300 dark:border-gray-700 rounded-md shadow-sm text-sm font-medium text-gray-700 dark:text-gray-300 bg-white dark:bg-gray-800 hover:bg-gray-50 dark:hover:bg-gray-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-indigo-500 disabled:opacity-50 disabled:cursor-not-allowed"
|
|
>
|
|
<MicrosoftIcon className="w-5 h-5" />
|
|
<span>Sign in with Microsoft</span>
|
|
</button>
|
|
</div>
|
|
</>
|
|
)}
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|