Files
FrontOffice/src/FrontOffice.Main/Pages/Auth/Phone.razor.cs

181 lines
5.6 KiB
C#
Raw Normal View History

2025-09-28 03:49:17 +03:30
using System.ComponentModel.DataAnnotations;
2025-09-28 03:24:54 +03:30
using Blazored.LocalStorage;
using FrontOffice.BFF.User.Protobuf.Protos.User;
2025-09-28 03:49:17 +03:30
using FrontOffice.BFF.User.Protobuf.Validator;
using FrontOffice.Main.Utilities;
2025-09-28 03:24:54 +03:30
using Grpc.Core;
using Microsoft.AspNetCore.Components;
using Microsoft.AspNetCore.WebUtilities;
using MudBlazor;
namespace FrontOffice.Main.Pages.Auth;
public partial class Phone : IDisposable
{
private const string PhoneStorageKey = "auth:phone-number";
private const string RedirectStorageKey = "auth:redirect";
2025-09-28 03:49:17 +03:30
private const string TokenStorageKey = "auth:token";
2025-09-28 03:24:54 +03:30
private const string OtpPurpose = "Login";
2025-09-28 03:49:17 +03:30
private static readonly CreateNewOtpTokenRequestValidator RequestValidator = new();
2025-09-28 03:24:54 +03:30
private readonly PhoneInputModel _model = new();
private MudForm? _form;
private bool _isBusy;
private string? _errorMessage;
private string? _redirect;
private CancellationTokenSource? _sendCts;
[Inject] private ILocalStorageService LocalStorage { get; set; } = default!;
[Inject] private UserContract.UserContractClient UserClient { get; set; } = default!;
protected override async Task OnInitializedAsync()
{
var uri = Navigation.ToAbsoluteUri(Navigation.Uri);
var query = QueryHelpers.ParseQuery(uri.Query);
if (query.TryGetValue("redirect", out var redirectValues))
{
_redirect = redirectValues.LastOrDefault();
}
var storedPhone = await LocalStorage.GetItemAsync<string>(PhoneStorageKey);
if (!string.IsNullOrWhiteSpace(storedPhone))
{
_model.PhoneNumber = storedPhone;
}
if (string.IsNullOrWhiteSpace(_redirect))
{
var storedRedirect = await LocalStorage.GetItemAsync<string>(RedirectStorageKey);
if (!string.IsNullOrWhiteSpace(storedRedirect))
{
_redirect = storedRedirect;
}
}
}
private async Task SendOtpAsync()
{
_errorMessage = null;
if (_form is null)
{
return;
}
await _form.Validate();
if (!_form.IsValid)
{
return;
}
_isBusy = true;
_sendCts?.Cancel();
_sendCts?.Dispose();
_sendCts = new CancellationTokenSource();
try
{
var request = new CreateNewOtpTokenRequest
{
Mobile = _model.PhoneNumber,
Purpose = OtpPurpose
};
2025-09-28 03:49:17 +03:30
var validationResult = RequestValidator.Validate(request);
if (!validationResult.IsValid)
{
_errorMessage = string.Join(" ", validationResult.Errors.Select(e => e.ErrorMessage).Distinct());
return;
}
var metadata = await BuildAuthMetadataAsync();
CreateNewOtpTokenResponse response;
if (metadata is not null)
{
response = await UserClient.CreateNewOtpTokenAsync(request, metadata, cancellationToken: _sendCts.Token);
}
else
{
response = await UserClient.CreateNewOtpTokenAsync(request, cancellationToken: _sendCts.Token);
}
2025-09-28 03:24:54 +03:30
if (response?.Success != true)
{
_errorMessage = string.IsNullOrWhiteSpace(response?.Message)
? "ارسال رمز پویا با خطا مواجه شد. لطفاً دوباره تلاش کنید."
: response!.Message;
return;
}
await LocalStorage.SetItemAsync(PhoneStorageKey, _model.PhoneNumber);
if (!string.IsNullOrWhiteSpace(_redirect))
{
await LocalStorage.SetItemAsync(RedirectStorageKey, _redirect);
}
else
{
await LocalStorage.RemoveItemAsync(RedirectStorageKey);
}
2025-09-28 03:49:17 +03:30
var target = $"{RouteConstants.Auth.Verify}?phone={Uri.EscapeDataString(_model.PhoneNumber)}";
2025-09-28 03:24:54 +03:30
if (!string.IsNullOrEmpty(_redirect))
{
target += "&redirect=" + Uri.EscapeDataString(_redirect);
}
Navigation.NavigateTo(target, forceLoad: false);
}
catch (RpcException rpcEx)
{
_errorMessage = !string.IsNullOrWhiteSpace(rpcEx.Status.Detail)
? rpcEx.Status.Detail
: "ارسال رمز پویا با خطا مواجه شد. لطفاً دوباره تلاش کنید.";
}
catch (OperationCanceledException)
{
}
catch (Exception ex)
{
_errorMessage = ex.Message;
}
finally
{
_isBusy = false;
_sendCts?.Dispose();
_sendCts = null;
await InvokeAsync(StateHasChanged);
}
}
2025-09-28 03:49:17 +03:30
private async Task<Metadata?> BuildAuthMetadataAsync()
{
var token = await LocalStorage.GetItemAsync<string>(TokenStorageKey);
if (string.IsNullOrWhiteSpace(token))
{
return null;
}
return new Metadata
{
{ "Authorization", $"Bearer {token}" }
};
}
2025-09-28 03:24:54 +03:30
public void Dispose()
{
_sendCts?.Cancel();
_sendCts?.Dispose();
_sendCts = null;
}
private sealed class PhoneInputModel
{
2025-09-28 03:49:17 +03:30
[Required(ErrorMessage = "???? ???? ????? ?????? ?????? ???.")]
[RegularExpression(@"^09\\d{9}$", ErrorMessage = "????? ?????? ????? ????.")]
2025-09-28 03:24:54 +03:30
public string PhoneNumber { get; set; } = string.Empty;
2025-09-28 03:49:17 +03:30
[Range(typeof(bool), "true", "true", ErrorMessage = "????? ????? ? ?????? ????? ???.")]
2025-09-28 03:24:54 +03:30
public bool AcceptTerms { get; set; }
}
}