From cd2efe096a786f85b017fc2b8e6b9c716de0c388 Mon Sep 17 00:00:00 2001 From: AndriiSyrotenko Date: Sun, 17 Dec 2023 22:59:12 +0000 Subject: [PATCH] Add User Settings pop-up window --- ShoppingAssistantWebClient.Web/Models/Role.cs | 14 + ShoppingAssistantWebClient.Web/Models/User.cs | 17 ++ .../Pages/Cards.razor.cs | 2 - .../Pages/Settings.razor | 152 ++++++++++ .../Pages/Settings.razor.cs | 118 ++++++++ .../Pages/Settings.razor.css | 260 ++++++++++++++++++ .../Pages/_Host.cshtml | 2 + ShoppingAssistantWebClient.Web/Program.cs | 3 + .../Shared/MainLayout.razor | 4 + .../Shared/NavMenu.razor | 15 +- .../ShoppingAssistantWebClient.Web.csproj | 1 + ShoppingAssistantWebClient.Web/_Imports.razor | 3 + 12 files changed, 588 insertions(+), 3 deletions(-) create mode 100644 ShoppingAssistantWebClient.Web/Models/Role.cs create mode 100644 ShoppingAssistantWebClient.Web/Models/User.cs create mode 100644 ShoppingAssistantWebClient.Web/Pages/Settings.razor create mode 100644 ShoppingAssistantWebClient.Web/Pages/Settings.razor.cs create mode 100644 ShoppingAssistantWebClient.Web/Pages/Settings.razor.css diff --git a/ShoppingAssistantWebClient.Web/Models/Role.cs b/ShoppingAssistantWebClient.Web/Models/Role.cs new file mode 100644 index 0000000..26df7fc --- /dev/null +++ b/ShoppingAssistantWebClient.Web/Models/Role.cs @@ -0,0 +1,14 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Threading.Tasks; + +namespace ShoppingAssistantWebClient.Web.Models +{ + public class Role + { + public string Id { get; set; } + + public string Name { get; set; } + } +} \ No newline at end of file diff --git a/ShoppingAssistantWebClient.Web/Models/User.cs b/ShoppingAssistantWebClient.Web/Models/User.cs new file mode 100644 index 0000000..fcce482 --- /dev/null +++ b/ShoppingAssistantWebClient.Web/Models/User.cs @@ -0,0 +1,17 @@ +using ShoppingAssistantWebClient.Web.Models; + +namespace ShoppingAssistantWebClient.Web.Models +{ + public class User + { + public string Id { get; set; } + + public string GuestId { get; set; } + + public List? Roles { get; set; } = new List(); + + public string Email { get; set; } + + public string Phone { get; set; } + } +} \ No newline at end of file diff --git a/ShoppingAssistantWebClient.Web/Pages/Cards.razor.cs b/ShoppingAssistantWebClient.Web/Pages/Cards.razor.cs index 2a9fd60..1bddd24 100644 --- a/ShoppingAssistantWebClient.Web/Pages/Cards.razor.cs +++ b/ShoppingAssistantWebClient.Web/Pages/Cards.razor.cs @@ -2,8 +2,6 @@ using Microsoft.AspNetCore.Components; using ShoppingAssistantWebClient.Web.Models; using ShoppingAssistantWebClient.Web.Network; using GraphQL; -using Newtonsoft.Json; -using Microsoft.JSInterop; using ShoppingAssistantWebClient.Web.Services; namespace ShoppingAssistantWebClient.Web.Pages; diff --git a/ShoppingAssistantWebClient.Web/Pages/Settings.razor b/ShoppingAssistantWebClient.Web/Pages/Settings.razor new file mode 100644 index 0000000..3c79afe --- /dev/null +++ b/ShoppingAssistantWebClient.Web/Pages/Settings.razor @@ -0,0 +1,152 @@ +@using System.Text.RegularExpressions +@using Models.GlobalInstances +@using ShoppingAssistantWebClient.Web.Models + +@inject IHttpContextAccessor httpContextAccessor; + + + + + +@code { + [CascadingParameter] BlazoredModalInstance BlazoredModal { get; set; } + + private string phoneValidationMessage = ""; + private string emailValidationMessage = ""; + private bool isPhoneInvalid = false; + private bool isEmailInvalid = false; + private bool isApplyDisabled = true; + private string phone = ""; + private string email = ""; + private string password = ""; + + private void ValidatePhone(ChangeEventArgs e) + { + errorMessage = ""; + phone = e.Value.ToString(); + if (!string.IsNullOrWhiteSpace(phone) && !Regex.IsMatch(phone, @"^\+[0-9]{1,15}$")) + { + phoneValidationMessage = "Please enter a valid phone number"; + isPhoneInvalid = true; + } + else + { + phoneValidationMessage = ""; + isPhoneInvalid = false; + } + + UpdateApplyButtonState(); + } + + private void ValidateEmail(ChangeEventArgs e) + { + errorMessage = ""; + email = e.Value.ToString(); + if (!string.IsNullOrWhiteSpace(email) && !Regex.IsMatch(email, @"^[^@\s]+@[^@\s]+\.[^@\s]+$")) + { + emailValidationMessage = "Please enter a valid email address."; + isEmailInvalid = true; + } + else + { + emailValidationMessage = ""; + isEmailInvalid = false; + } + UpdateApplyButtonState(); + } + + private void UpdateApplyButtonState() + { + if(user.Roles.Any(role => role.Name == "User")) + isApplyDisabled = (string.IsNullOrWhiteSpace(phone) && string.IsNullOrWhiteSpace(email)) || isPhoneInvalid || isEmailInvalid; + else + isApplyDisabled = string.IsNullOrWhiteSpace(password) || (string.IsNullOrWhiteSpace(phone) && string.IsNullOrWhiteSpace(email)) || isPhoneInvalid || isEmailInvalid; + } + + private void OnPasswordInput(ChangeEventArgs e) + { + errorMessage = ""; + password = e.Value.ToString(); + UpdateApplyButtonState(); + } + + private async Task Close() => await BlazoredModal.CloseAsync(ModalResult.Ok(true)); + private async Task Cancel() => await BlazoredModal.CancelAsync(); + + private async Task Apply() { + + await UpdateUser(); + isApplyDisabled = true; + await GetUser(); + StateHasChanged(); + + if(user.Roles.Any(role => role.Name == "User")) { + await Task.Delay(3000); + await InvokeAsync(() => { + updateMessage = ""; + StateHasChanged(); + }); + } + } + + private string ShowErrorDivClass() + { + return string.IsNullOrEmpty(errorMessage) ? "hidden" : "alert alert-danger"; + } + + private string ShowUpdateDivClass() { + return string.IsNullOrEmpty(updateMessage) ? "hidden" : "alert alert-success"; + } +} diff --git a/ShoppingAssistantWebClient.Web/Pages/Settings.razor.cs b/ShoppingAssistantWebClient.Web/Pages/Settings.razor.cs new file mode 100644 index 0000000..25d9c8a --- /dev/null +++ b/ShoppingAssistantWebClient.Web/Pages/Settings.razor.cs @@ -0,0 +1,118 @@ +using Microsoft.AspNetCore.Components; +using ShoppingAssistantWebClient.Web.Network; +using ShoppingAssistantWebClient.Web.Models; +using ShoppingAssistantWebClient.Web.Models.GlobalInstances; +using GraphQL; +using Newtonsoft.Json; + + +namespace ShoppingAssistantWebClient.Web.Pages; + +public partial class Settings : ComponentBase +{ + [Inject] + private ApiClient _apiClient { get; set; } + + [Inject] + private IHttpContextAccessor _httpContextAccessor { get; set; } + + public User user = new User(); + + private string errorMessage = ""; + + private string updateMessage = ""; + + protected override async Task OnInitializedAsync() + { + await GetUser(); + } + + public async Task GetUser() { + try { + var request = new GraphQLRequest { + Query = @" + query User($id: String!) { + user(id: $id) { + roles { + id + name + } + phone + email + } + }", + Variables = new + { + id = GlobalUser.Id, + } + }; + + var response = await _apiClient.QueryAsync(request); + var responseData = response.Data; + //System.Console.WriteLine(responseData); + + var jsonCategoriesResponse = JsonConvert.SerializeObject(responseData.user); + this.user = JsonConvert.DeserializeObject(jsonCategoriesResponse); + user.GuestId = _httpContextAccessor.HttpContext.Request.Cookies["guestId"]; + + } + catch(Exception ex) + { + Console.WriteLine($"Error in GetUser: {ex}"); + } + } + + public async Task UpdateUser() + { + try { + if(user.Roles.Any(role => role.Name == "User")) + { + updateMessage = "Your data has been successfully updated"; + } + if(phone == "") { + phone = user.Phone; + } + if(email == "") { + email = user.Email; + } + var request = new GraphQLRequest + { + Query = @" + mutation UpdateUser($userDto: UserDtoInput!) { + updateUser(userDto: $userDto) { + tokens { accessToken, refreshToken }, + user { email, phone } + } + }", + Variables = new + { + userDto = new + { + id = GlobalUser.Id, + guestId = user.GuestId, + roles = user.Roles.Select(r => new { id = r.Id, name = r.Name }), + email = email, + phone = phone, + password = password + } + } + }; + + var response = await _apiClient.QueryAsync(request); + var responseData = response.Data; + System.Console.WriteLine(responseData); + errorMessage = ""; + phone = ""; + email = ""; + } + catch(Exception ex) + { + if (ex.Message.Contains("The HTTP request failed with status code InternalServerError")) { + errorMessage = "This user is already registered."; + } else { + errorMessage = "Something went wrong, please try again."; + } + Console.WriteLine($"Error in UpdateUser: {ex}"); + } + } +} diff --git a/ShoppingAssistantWebClient.Web/Pages/Settings.razor.css b/ShoppingAssistantWebClient.Web/Pages/Settings.razor.css new file mode 100644 index 0000000..cc2ba3a --- /dev/null +++ b/ShoppingAssistantWebClient.Web/Pages/Settings.razor.css @@ -0,0 +1,260 @@ +.modal-dialog { + display: flex; + align-items: center; + justify-content: center; + min-height: 100vh; + margin: auto; +} + +.close { + background: none; + border: none; + outline: none; +} + +.close span { + display: block; + color: #000; + font-size: 24px; +} + +.modal-content { + background-color: white; + color: #333; +} + +.form-group label { + color: #333; +} + +.form-control { + border: 1px solid #ccc; + border-radius: 4px; +} + +.form-group { + position: relative; +} + +.btn { + width: 120px; + height: 32px; + display: flex; + justify-content: center; + align-items: center; + line-height: 1; +} + +#phone, #email { + margin-bottom: 1.5rem; +} + +.validation-message { + position: absolute; + bottom: -20px; + left: 0; + color: red; + font-size: 0.8rem; + visibility: hidden; +} + +.validation-message.active { + visibility: visible; +} + +.btn-primary { + background-color: #007bff; + border-color: #007bff; +} + +.btn-secondary { + background-color: #6c757d; + border-color: #6c757d; +} + +.btn-primary:hover { + background-color: #0056b3; +} + +.btn-secondary:hover { + background-color: #545b62; +} + +.btn:active { + transform: scale(0.95); + box-shadow: 0 1px 3px rgba(0, 0, 0, 0.2); +} + +.btn:focus { + outline: none !important; + box-shadow: none !important; +} + +.modal-header, .modal-footer, .modal-body { + padding: 16px !important; +} + +.log-in-button-left { + margin-right: auto; + visibility: visible; +} + +.log-in-button-left-hidden { + visibility: hidden; +} + +.modal-footer { + display: flex; + justify-content: flex-end; +} + +.hidden { + display: none; +} + +@media (max-width: 320px) { + .modal-dialog { + max-width: 90%; + margin: auto; + } + + .modal-header, .modal-footer, .modal-body { + padding: 6px !important; + } + + .form-control { + font-size: 0.85rem; + padding: 0.375rem 0.75rem; + } + + .btn { + font-size: 0.85rem; + width: 80px; + height: 28px; + display: flex; + justify-content: center; + align-items: center; + line-height: 1; + } + + #phone, #email { + margin-bottom: 0.8rem; + } +} + +@media (min-width: 321px) and (max-width: 376px) { + .modal-dialog { + max-width: 300px; + margin: auto; + } + + .modal-header, .modal-footer, .modal-body { + padding: 8px !important; + } + + .form-control { + font-size: 0.85rem; + padding: 0.375rem 0.75rem; + } + + .btn { + font-size: 0.85rem; + width: 70px; + height: 28px; + display: flex; + justify-content: center; + align-items: center; + line-height: 1; + } + + #phone, #email { + margin-bottom: 1rem; + } + + #phone-validation, #email-validation{ + margin-top: -1rem; + } + +} + +@media (min-width: 376px) and (max-width: 426px) { + .modal-dialog { + max-width: 300px; + margin: auto; + } + + .modal-header, .modal-footer, .modal-body { + padding: 10px !important; + } + + .form-control { + font-size: 0.9rem; + padding: 0.375rem 0.75rem; + } + + .btn { + font-size: 0.9rem; + width: 80px; + height: 28px; + display: flex; + justify-content: center; + align-items: center; + line-height: 1; + } + + #phone, #email { + margin-bottom: 1rem; + } + + #phone-validation, #email-validation{ + margin-top: -1rem; + } +} + +@media (min-width: 426px) and (max-width: 768px) { + .modal-dialog { + width: 60%; + max-width: 400px; + min-width: 330px; + margin: auto; + } + + .modal-header, .modal-footer, .modal-body { + padding: 12px !important; + } + + .form-control { + font-size: 0.95rem; + padding: 0.375rem 0.75rem; + } + + .btn { + font-size: 0.95rem; + width: 100px; + height: 30px; + display: flex; + justify-content: center; + align-items: center; + line-height: 1; + } + + #phone, #email { + margin-bottom: 1.3rem; + } + + #phone-validation, #email-validation{ + margin-top: -1.3rem; + } +} + +@media (min-width: 426px) and (max-width: 582px) { + .btn { + font-size: 0.95rem; + width: 80px !important; + height: 28px !important; + display: flex; + justify-content: center; + align-items: center; + line-height: 1; + } +} diff --git a/ShoppingAssistantWebClient.Web/Pages/_Host.cshtml b/ShoppingAssistantWebClient.Web/Pages/_Host.cshtml index 5f51717..f9403ad 100644 --- a/ShoppingAssistantWebClient.Web/Pages/_Host.cshtml +++ b/ShoppingAssistantWebClient.Web/Pages/_Host.cshtml @@ -12,6 +12,8 @@ + + diff --git a/ShoppingAssistantWebClient.Web/Program.cs b/ShoppingAssistantWebClient.Web/Program.cs index 4809cc6..99b9198 100644 --- a/ShoppingAssistantWebClient.Web/Program.cs +++ b/ShoppingAssistantWebClient.Web/Program.cs @@ -3,6 +3,7 @@ using ShoppingAssistantWebClient.Web.Configurations; using ShoppingAssistantWebClient.Web.Data; using ShoppingAssistantWebClient.Web.Network; using ShoppingAssistantWebClient.Web.Services; +using Blazored.Modal; var builder = WebApplication.CreateBuilder(args); @@ -13,6 +14,8 @@ builder.Services.AddServerSideBlazor().AddCircuitOptions(options => { options.De builder.Services.AddSingleton(); builder.Services.AddApiClient(builder.Configuration); builder.Services.AddSingleton(); +builder.Services.AddBlazoredModal(); + var app = builder.Build(); diff --git a/ShoppingAssistantWebClient.Web/Shared/MainLayout.razor b/ShoppingAssistantWebClient.Web/Shared/MainLayout.razor index 2b9f5d4..e164adc 100644 --- a/ShoppingAssistantWebClient.Web/Shared/MainLayout.razor +++ b/ShoppingAssistantWebClient.Web/Shared/MainLayout.razor @@ -1,10 +1,14 @@ @inherits LayoutComponentBase @using ShoppingAssistantWebClient.Web.Pages +@using Blazored.Modal + CARTAID + +