feat: implement identity authentication, authorization policies, and MAUI platform support with Docker orchestration
This commit is contained in:
@@ -1,69 +1,83 @@
|
||||
@page "/account/login"
|
||||
@layout AuthLayout
|
||||
@attribute [AllowAnonymous]
|
||||
@using Microsoft.AspNetCore.Components.Forms
|
||||
@using NexusReader.UI.Shared.Services
|
||||
@using NexusReader.UI.Shared.Components.Atoms
|
||||
@inject IIdentityService IdentityService
|
||||
@inject NavigationManager NavigationManager
|
||||
|
||||
<div class="login-container">
|
||||
<div class="login-card">
|
||||
<div class="login-header">
|
||||
<h1>NEXUS<span>AI</span></h1>
|
||||
<p>Welcome back, Reader.</p>
|
||||
<div class="login-page-container">
|
||||
<div class="mesh-bg"></div>
|
||||
|
||||
<div class="auth-card">
|
||||
<div class="auth-header">
|
||||
<div class="logo-box">
|
||||
<NexusIcon Name="robot" Size="48" />
|
||||
</div>
|
||||
<h1 class="app-name">E-Czytnik <span>AI</span></h1>
|
||||
<p class="auth-subtitle">Zaloguj się do E-Czytnika AI</p>
|
||||
</div>
|
||||
|
||||
<EditForm Model="@_loginModel" OnValidSubmit="HandleLogin">
|
||||
<div class="social-auth">
|
||||
<button type="button" class="btn-google-auth" @onclick="HandleGoogleLogin">
|
||||
<img src="https://www.gstatic.com/images/branding/product/1x/gsa_512dp.png"
|
||||
alt="Google"
|
||||
style="width: 20px !important; height: 20px !important; flex-shrink: 0;" />
|
||||
<span>Zaloguj się przez Google</span>
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<div class="auth-divider">
|
||||
<span>lub</span>
|
||||
</div>
|
||||
|
||||
<EditForm Model="@_loginModel" OnValidSubmit="HandleLogin" class="auth-form">
|
||||
<DataAnnotationsValidator />
|
||||
|
||||
<div class="form-group">
|
||||
<label for="email">Email</label>
|
||||
<InputText id="email" @bind-Value="_loginModel.Email" class="form-control" placeholder="reader@nexus.ai" />
|
||||
<ValidationMessage For="@(() => _loginModel.Email)" />
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="password">Password</label>
|
||||
<InputText id="password" type="password" @bind-Value="_loginModel.Password" class="form-control" placeholder="••••••••" />
|
||||
<ValidationMessage For="@(() => _loginModel.Password)" />
|
||||
</div>
|
||||
|
||||
<div class="form-options">
|
||||
<div class="remember-me">
|
||||
<InputCheckbox id="remember" @bind-Value="_loginModel.RememberMe" />
|
||||
<label for="remember">Remember me</label>
|
||||
<div class="field-group">
|
||||
<div class="field-icon">
|
||||
<NexusIcon Name="mail" Size="18" />
|
||||
</div>
|
||||
<a href="/account/forgot-password" class="forgot-link">Forgot password?</a>
|
||||
<InputText id="email" @bind-Value="_loginModel.Email" placeholder="E-mail" class="field-input" />
|
||||
</div>
|
||||
<ValidationMessage For="@(() => _loginModel.Email)" class="auth-validation" />
|
||||
|
||||
<div class="field-group">
|
||||
<div class="field-icon">
|
||||
<NexusIcon Name="lock" Size="18" />
|
||||
</div>
|
||||
<InputText id="password" type="@(_showPassword ? "text" : "password")" @bind-Value="_loginModel.Password" placeholder="Hasło" class="field-input" />
|
||||
<button type="button" class="toggle-visibility" @onclick="TogglePassword">
|
||||
<NexusIcon Name="@(_showPassword ? "eye-off" : "eye")" Size="18" />
|
||||
</button>
|
||||
</div>
|
||||
<ValidationMessage For="@(() => _loginModel.Password)" class="auth-validation" />
|
||||
|
||||
@if (!string.IsNullOrEmpty(_errorMessage))
|
||||
{
|
||||
<div class="error-banner">
|
||||
@_errorMessage
|
||||
</div>
|
||||
<div class="auth-error">@_errorMessage</div>
|
||||
}
|
||||
|
||||
<button type="submit" class="btn-login" disabled="@_isSubmitting">
|
||||
<button type="submit" class="btn-submit-auth" disabled="@_isSubmitting">
|
||||
@if (_isSubmitting)
|
||||
{
|
||||
<span class="spinner"></span>
|
||||
<div class="auth-loader"></div>
|
||||
}
|
||||
else
|
||||
{
|
||||
<span>Login</span>
|
||||
<span>Zaloguj się</span>
|
||||
}
|
||||
</button>
|
||||
|
||||
<div class="separator">
|
||||
<span>OR</span>
|
||||
</div>
|
||||
|
||||
<button type="button" class="btn-google" @onclick="HandleGoogleLogin">
|
||||
<img src="https://www.gstatic.com/images/branding/product/1x/gsa_512dp.png" alt="Google" />
|
||||
Continue with Google
|
||||
</button>
|
||||
</EditForm>
|
||||
|
||||
<div class="login-footer">
|
||||
<p>Don't have an account? <a href="/account/register">Create one</a></p>
|
||||
<div class="auth-footer">
|
||||
<a href="/account/forgot-password" class="auth-link">Zapomniałem hasła?</a>
|
||||
<p class="auth-switch">Nie masz konta? <a href="/account/register">Zarejestruj się</a></p>
|
||||
</div>
|
||||
|
||||
<div class="auth-legal">
|
||||
Korzystając z usługi, akceptujesz <a href="/terms">Regulamin</a> i <a href="/privacy">Politykę Prywatności</a>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -72,6 +86,7 @@
|
||||
private LoginModel _loginModel = new();
|
||||
private string? _errorMessage;
|
||||
private bool _isSubmitting;
|
||||
private bool _showPassword;
|
||||
|
||||
private async Task HandleLogin()
|
||||
{
|
||||
@@ -81,30 +96,15 @@
|
||||
try
|
||||
{
|
||||
var success = await IdentityService.LoginAsync(_loginModel.Email, _loginModel.Password);
|
||||
if (success)
|
||||
{
|
||||
NavigationManager.NavigateTo("/");
|
||||
}
|
||||
else
|
||||
{
|
||||
_errorMessage = "Invalid email or password.";
|
||||
}
|
||||
}
|
||||
catch (Exception)
|
||||
{
|
||||
_errorMessage = "An error occurred during login. Please try again.";
|
||||
}
|
||||
finally
|
||||
{
|
||||
_isSubmitting = false;
|
||||
if (success) NavigationManager.NavigateTo("/");
|
||||
else _errorMessage = "Nieprawidłowy e-mail lub hasło.";
|
||||
}
|
||||
catch (Exception) { _errorMessage = "Wystąpił błąd logowania."; }
|
||||
finally { _isSubmitting = false; }
|
||||
}
|
||||
|
||||
private void HandleGoogleLogin()
|
||||
{
|
||||
// Redirect to external login endpoint
|
||||
NavigationManager.NavigateTo("identity/login/google", forceLoad: true);
|
||||
}
|
||||
private void HandleGoogleLogin() => NavigationManager.NavigateTo("identity/login/google", forceLoad: true);
|
||||
private void TogglePassword() => _showPassword = !_showPassword;
|
||||
|
||||
public class LoginModel
|
||||
{
|
||||
@@ -114,7 +114,5 @@
|
||||
|
||||
[System.ComponentModel.DataAnnotations.Required]
|
||||
public string Password { get; set; } = string.Empty;
|
||||
|
||||
public bool RememberMe { get; set; }
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user