diff --git a/src/BackOffice/App.razor b/src/BackOffice/App.razor index c93d097..598cb71 100644 --- a/src/BackOffice/App.razor +++ b/src/BackOffice/App.razor @@ -17,7 +17,7 @@ } }); } - + @@ -27,4 +27,10 @@ - \ No newline at end of file + + +@code{ + + + +} \ No newline at end of file diff --git a/src/BackOffice/BackOffice.csproj b/src/BackOffice/BackOffice.csproj index 6e27229..d2c7337 100644 --- a/src/BackOffice/BackOffice.csproj +++ b/src/BackOffice/BackOffice.csproj @@ -21,10 +21,15 @@ + + + + + diff --git a/src/BackOffice/Common/BaseComponents/BasePageComponent.razor b/src/BackOffice/Common/BaseComponents/BasePageComponent.razor index 715341f..6519ddd 100644 --- a/src/BackOffice/Common/BaseComponents/BasePageComponent.razor +++ b/src/BackOffice/Common/BaseComponents/BasePageComponent.razor @@ -6,9 +6,9 @@ فیلتر ها - @if (IsFilterd) + @if (IsFiltered) { - خذف فیلتر + حذف فیلتر } @@ -32,4 +32,3 @@ - diff --git a/src/BackOffice/Common/BaseComponents/BasePageComponent.razor.cs b/src/BackOffice/Common/BaseComponents/BasePageComponent.razor.cs index d7086be..7d930e0 100644 --- a/src/BackOffice/Common/BaseComponents/BasePageComponent.razor.cs +++ b/src/BackOffice/Common/BaseComponents/BasePageComponent.razor.cs @@ -3,15 +3,9 @@ using Microsoft.AspNetCore.Components; namespace BackOffice.Common.BaseComponents; public partial class BasePageComponent { - [Parameter] public RenderFragment Filters { get; set; } - [Parameter] public RenderFragment Content { get; set; } - [Parameter] public bool IsFilterd { get; set; } + [Parameter] public RenderFragment Filters { get; set; } = default!; + [Parameter] public RenderFragment Content { get; set; } = default!; + [Parameter] public bool IsFiltered { get; set; } [Parameter] public EventCallback OnSubmitClick { get; set; } [Parameter] public EventCallback OnClearFilterClick { get; set; } - private bool _open = true; - - private void ToggleDrawer() - { - _open = !_open; - } -} \ No newline at end of file +} diff --git a/src/BackOffice/Common/BaseComponents/DateRangePicker.razor.cs b/src/BackOffice/Common/BaseComponents/DateRangePicker.razor.cs index 754b321..39cd928 100644 --- a/src/BackOffice/Common/BaseComponents/DateRangePicker.razor.cs +++ b/src/BackOffice/Common/BaseComponents/DateRangePicker.razor.cs @@ -55,12 +55,17 @@ public partial class DateRangePicker } private async Task OnClickOK() { - _picker.CloseAsync(); - await OnChanged.InvokeAsync(_picker.DateRange); + if (_picker.DateRange is not null) + { + await OnChanged.InvokeAsync(_picker.DateRange); + } + + await _picker.CloseAsync(); } private async Task OnClickClear() { - _picker.CloseAsync(); - await OnChanged.InvokeAsync(_picker.DateRange); + _dateRange = new DateRange(); + await OnChanged.InvokeAsync(_dateRange); + await _picker.CloseAsync(); } -} \ No newline at end of file +} diff --git a/src/BackOffice/Common/Configure/ConfigureService.cs b/src/BackOffice/Common/Configure/ConfigureService.cs index de8bd70..7a2f140 100644 --- a/src/BackOffice/Common/Configure/ConfigureService.cs +++ b/src/BackOffice/Common/Configure/ConfigureService.cs @@ -2,23 +2,22 @@ using BackOffice.BFF.Otp.Protobuf.Protos.Otp; using BackOffice.BFF.Package.Protobuf.Protos.Package; using BackOffice.BFF.Role.Protobuf.Protos.Role; +using BackOffice.BFF.Products.Protobuf.Protos.Products; using BackOffice.BFF.User.Protobuf.Protos.User; using BackOffice.BFF.UserAddress.Protobuf.Protos.UserAddress; using BackOffice.BFF.UserOrder.Protobuf.Protos.UserOrder; using BackOffice.BFF.UserRole.Protobuf.Protos.UserRole; +using BackOffice.BFF.Category.Protobuf.Protos.Category; using BackOffice.Common.Utilities; using Blazored.LocalStorage; -using Google.Protobuf.Reflection; using Grpc.Core; using Grpc.Core.Interceptors; using Grpc.Net.Client; using Grpc.Net.Client.Web; using Microsoft.AspNetCore.Components.Authorization; -using Microsoft.AspNetCore.Components.WebAssembly.Authentication; using MudBlazor.Services; using System.Text.Json; using System.Text.Json.Serialization; -using static MudBlazor.Colors; namespace Microsoft.Extensions.DependencyInjection; @@ -48,25 +47,35 @@ public static class ConfigureServices - public static IServiceCollection AddGrpcServices(this IServiceCollection services, IConfiguration configuration) // + public static IServiceCollection AddGrpcServices(this IServiceCollection services, IConfiguration configuration) { - var baseUri = configuration["GwUrl"]; - Console.WriteLine(); - Console.WriteLine(baseUri); - var httpClient = new HttpClient(new GrpcWebHandler(GrpcWebMode.GrpcWeb, new HttpClientHandler())); - httpClient.Timeout = TimeSpan.FromMinutes(10); // TODO Check Timeout - var serviceProvider = services.BuildServiceProvider(); - var channel = CreateAuthenticatedChannel(baseUri, httpClient, serviceProvider); + services.AddSingleton(sp => + { + var config = sp.GetRequiredService(); + var baseUri = config["GwUrl"]; + if (string.IsNullOrWhiteSpace(baseUri)) + { + throw new InvalidOperationException("Configuration value 'GwUrl' is missing or empty."); + } + var httpClient = new HttpClient(new GrpcWebHandler(GrpcWebMode.GrpcWeb, new HttpClientHandler())) + { + Timeout = TimeSpan.FromMinutes(10) + }; - services.AddTransient(sp => new OtpContract.OtpContractClient(channel)); - services.AddTransient(sp => new PackageContract.PackageContractClient(channel)); - services.AddTransient(sp => new RoleContract.RoleContractClient(channel)); - services.AddTransient(sp => new UserContract.UserContractClient(channel)); - services.AddTransient(sp => new UserAddressContract.UserAddressContractClient(channel)); - services.AddTransient(sp => new UserOrderContract.UserOrderContractClient(channel)); - services.AddTransient(sp => new UserRoleContract.UserRoleContractClient(channel)); + return CreateAuthenticatedChannel(baseUri, httpClient, sp); + }); + + services.AddTransient(sp => new OtpContract.OtpContractClient(sp.GetRequiredService())); + services.AddTransient(sp => new PackageContract.PackageContractClient(sp.GetRequiredService())); + services.AddTransient(sp => new ProductsContract.ProductsContractClient(sp.GetRequiredService())); + services.AddTransient(sp => new RoleContract.RoleContractClient(sp.GetRequiredService())); + services.AddTransient(sp => new UserContract.UserContractClient(sp.GetRequiredService())); + services.AddTransient(sp => new UserAddressContract.UserAddressContractClient(sp.GetRequiredService())); + services.AddTransient(sp => new UserOrderContract.UserOrderContractClient(sp.GetRequiredService())); + services.AddTransient(sp => new UserRoleContract.UserRoleContractClient(sp.GetRequiredService())); + services.AddTransient(sp => new CategoryContract.CategoryContractClient(sp.GetRequiredService())); return services; } @@ -98,7 +107,6 @@ public static class ConfigureServices MaxReceiveMessageSize = 1000 * 1024 * 1024, // 1 GB MaxSendMessageSize = 1000 * 1024 * 1024 // 1 GB }); - var invoker = channel.Intercept(new ErrorHandlerInterceptor()); - return invoker; + return channel.Intercept(new ErrorHandlerInterceptor()); } -} \ No newline at end of file +} diff --git a/src/BackOffice/Common/Utilities/ErrorHandlerInterceptor.cs b/src/BackOffice/Common/Utilities/ErrorHandlerInterceptor.cs index a8aeb80..24dad33 100644 --- a/src/BackOffice/Common/Utilities/ErrorHandlerInterceptor.cs +++ b/src/BackOffice/Common/Utilities/ErrorHandlerInterceptor.cs @@ -1,14 +1,15 @@ -using Grpc.Core.Interceptors; -using Grpc.Core; +using Grpc.Core; +using Grpc.Core.Interceptors; +using MudBlazor; namespace BackOffice.Common.Utilities; public class ErrorHandlerInterceptor : Interceptor { public override AsyncUnaryCall AsyncUnaryCall( - TRequest request, - ClientInterceptorContext context, - AsyncUnaryCallContinuation continuation) + TRequest request, + ClientInterceptorContext context, + AsyncUnaryCallContinuation continuation) { var call = continuation(request, context); @@ -28,8 +29,9 @@ public class ErrorHandlerInterceptor : Interceptor } catch (Exception ex) { - GlobalConstants.ConstSnackbar.Add(ex.Message, severity: MudBlazor.Severity.Error); + var message = ex.Message.ExtractUserFriendlyMessage(); + GlobalConstants.ConstSnackbar?.Add(message, Severity.Error); throw; } } -} \ No newline at end of file +} diff --git a/src/BackOffice/Common/Utilities/GlobalConstants.cs b/src/BackOffice/Common/Utilities/GlobalConstants.cs index 7d48134..b2a4788 100644 --- a/src/BackOffice/Common/Utilities/GlobalConstants.cs +++ b/src/BackOffice/Common/Utilities/GlobalConstants.cs @@ -8,6 +8,5 @@ public static class GlobalConstants public const string DateTimeFormat = "MMM dd, yyyy - HH:mm"; public static string JwtTokenKey = "AuthToken"; public const string SuccessMsg = "با موفقیت انجام شد"; - public static ISnackbar ConstSnackbar; - -} \ No newline at end of file + public static ISnackbar? ConstSnackbar { get; set; } +} diff --git a/src/BackOffice/Common/Utilities/RouteConstance.cs b/src/BackOffice/Common/Utilities/RouteConstance.cs index 5906b3b..55b63ca 100644 --- a/src/BackOffice/Common/Utilities/RouteConstance.cs +++ b/src/BackOffice/Common/Utilities/RouteConstance.cs @@ -11,4 +11,6 @@ public static class RouteConstance public const string UserOrder = "/UserOrderPage/"; public const string UserRole = "/UserRolePage/"; public const string UserAddress = "/UserAddressPage/"; + public const string Products = "/ProductsPage/"; + public const string Category = "/CategoryPage/"; } diff --git a/src/BackOffice/Pages/AutoComplete/ProductsAutoComplete.razor b/src/BackOffice/Pages/AutoComplete/ProductsAutoComplete.razor new file mode 100644 index 0000000..87ca4de --- /dev/null +++ b/src/BackOffice/Pages/AutoComplete/ProductsAutoComplete.razor @@ -0,0 +1,15 @@ +@using BackOffice.BFF.Products.Protobuf.Protos.Products + + + diff --git a/src/BackOffice/Pages/AutoComplete/ProductsAutoComplete.razor.cs b/src/BackOffice/Pages/AutoComplete/ProductsAutoComplete.razor.cs new file mode 100644 index 0000000..f19ae7b --- /dev/null +++ b/src/BackOffice/Pages/AutoComplete/ProductsAutoComplete.razor.cs @@ -0,0 +1,79 @@ +using BackOffice.BFF.Products.Protobuf.Protos.Products; +using Microsoft.AspNetCore.Components; + +namespace BackOffice.Pages.AutoComplete; + +public partial class ProductsAutoComplete +{ + [Inject] public ProductsContract.ProductsContractClient ProductsService { get; set; } = default!; + + [Parameter] public long? Value { get; set; } + [Parameter] public string? Label { get; set; } = "محصول"; + [Parameter] public EventCallback ValueChanged { get; set; } + + private List _items = new(); + private GetAllProductsByFilterResponseModel? _item; + + protected override async void OnParametersSet() + { + await base.OnParametersSetAsync(); + + if (Value.HasValue && Value.Value > 0) + { + var getAll = await ProductsService.GetAllProductsByFilterAsync(new GetAllProductsByFilterRequest + { + Filter = new GetAllProductsByFilterFilter + { + Id = Value.Value + }, + PaginationState = new PaginationState + { + PageNumber = 1, + PageSize = 1 + } + }); + + if (getAll?.Models != null && getAll.Models.Any()) + { + _item = getAll.Models.First(); + StateHasChanged(); + } + } + } + + private async Task OnSelected(GetAllProductsByFilterResponseModel? model) + { + if (model == null) + { + _item = null; + await ValueChanged.InvokeAsync(null); + } + else + { + _item = model; + await ValueChanged.InvokeAsync(model.Id); + } + } + + private async Task> Search(string value, CancellationToken token) + { + var request = new GetAllProductsByFilterRequest + { + Filter = new GetAllProductsByFilterFilter + { + Title = string.IsNullOrWhiteSpace(value) ? null : value + }, + PaginationState = new PaginationState + { + PageNumber = 1, + PageSize = 9 + } + }; + + var getAll = await ProductsService.GetAllProductsByFilterAsync(request); + _items = getAll?.Models?.ToList() ?? new List(); + + return _items; + } +} + diff --git a/src/BackOffice/Pages/Category/CategoryMainPage.razor b/src/BackOffice/Pages/Category/CategoryMainPage.razor new file mode 100644 index 0000000..9a05073 --- /dev/null +++ b/src/BackOffice/Pages/Category/CategoryMainPage.razor @@ -0,0 +1,83 @@ +@attribute [Route(RouteConstance.Category)] + +@using BackOffice.BFF.Category.Protobuf.Protos.Category +@using BackOffice.Common.BaseComponents +@using DataModel = BackOffice.BFF.Category.Protobuf.Protos.Category.GetAllCategoryByFilterResponseModel + + + + + + + + + + + + + + + + + مدیریت دستهبندیها + + + + افزودن + + + + + + + + + + + + @(context.Item.IsActive ? "فعال" : "غیرفعال") + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/BackOffice/Pages/Category/CategoryMainPage.razor.cs b/src/BackOffice/Pages/Category/CategoryMainPage.razor.cs new file mode 100644 index 0000000..7324188 --- /dev/null +++ b/src/BackOffice/Pages/Category/CategoryMainPage.razor.cs @@ -0,0 +1,120 @@ +using BackOffice.BFF.Category.Protobuf.Protos.Category; +using BackOffice.Common.BaseComponents; +using Mapster; +using Microsoft.AspNetCore.Components; +using MudBlazor; +using DataModel = BackOffice.BFF.Category.Protobuf.Protos.Category.GetAllCategoryByFilterResponseModel; + +namespace BackOffice.Pages.Category; + +public partial class CategoryMainPage +{ + [Inject] public CategoryContract.CategoryContractClient CategoryContract { get; set; } = default!; + + private BasePageComponent _basePage = default!; + private MudDataGrid _gridData = default!; + private GetAllCategoryByFilterRequest _request = new() { Filter = new() }; + + private async Task> ServerReload(GridState state) + { + _request.Filter ??= new(); + _request.PaginationState ??= new(); + _request.PaginationState.PageNumber = state.Page + 1; + _request.PaginationState.PageSize = state.PageSize; + + var result = await CategoryContract.GetAllCategoryByFilterAsync(_request); + if (result != null && result.Models != null && result.Models.Any()) + { + return new GridData + { + Items = result.Models.ToList(), + TotalItems = (int)result.MetaData.TotalCount + }; + } + + return new GridData(); + } + + public async Task CreateNew() + { + var dialog = await DialogService.ShowAsync( + "افزودن دستهبندی", + new DialogParameters + { + { x => x.Model, new CreateNewCategoryRequest { IsActive = true } } + }, + new DialogOptions { CloseButton = true, FullWidth = true, MaxWidth = MaxWidth.Small }); + + var result = await dialog.Result; + if (!result.Canceled) + { + await ReloadData(); + Snackbar.Add("دستهبندی با موفقیت ایجاد شد.", Severity.Success); + } + } + + public async Task Update(DataModel model) + { + var getResponse = await CategoryContract.GetCategoryAsync(new GetCategoryRequest { Id = model.Id }); + + var editModel = getResponse.Adapt(); + + var dialog = await DialogService.ShowAsync( + "ویرایش دستهبندی", + new DialogParameters + { + { x => x.EditModel, editModel } + }, + new DialogOptions { CloseButton = true, FullWidth = true, MaxWidth = MaxWidth.Small }); + + var result = await dialog.Result; + if (!result.Canceled) + { + await ReloadData(); + Snackbar.Add("دستهبندی با موفقیت ویرایش شد.", Severity.Success); + } + } + + private async Task OnDelete(DataModel model) + { + var options = new DialogOptions { CloseOnEscapeKey = true, MaxWidth = MaxWidth.Small }; + bool? result = await DialogService.ShowMessageBox( + "اخطار", + "آیا از حذف این دستهبندی مطمئن هستید؟", + yesText: "حذف", + cancelText: "لغو", + options: options); + + if (result == true) + { + await CategoryContract.DeleteCategoryAsync(new DeleteCategoryRequest { Id = model.Id }); + await ReloadData(); + Snackbar.Add("دستهبندی با موفقیت حذف شد.", Severity.Success); + } + + StateHasChanged(); + } + + private async Task ReloadData() + { + if (_gridData != null) + { + await _gridData.ReloadServerData(); + } + } + + public async Task OnFilterSubmit() + { + _basePage.IsFiltered = true; + StateHasChanged(); + await ReloadData(); + } + + public async Task OnFilterCleared() + { + _basePage.IsFiltered = false; + StateHasChanged(); + _request = new GetAllCategoryByFilterRequest { Filter = new GetAllCategoryByFilterFilter() }; + await ReloadData(); + } +} diff --git a/src/BackOffice/Pages/Category/CreateOrUpdateCategoryDialog.razor b/src/BackOffice/Pages/Category/CreateOrUpdateCategoryDialog.razor new file mode 100644 index 0000000..7484e35 --- /dev/null +++ b/src/BackOffice/Pages/Category/CreateOrUpdateCategoryDialog.razor @@ -0,0 +1,48 @@ +@using BackOffice.BFF.Category.Protobuf.Protos.Category +@using Microsoft.AspNetCore.Components.Forms + + + + + + + + + + + + + + + + + + + لغو + ثبت + + diff --git a/src/BackOffice/Pages/Category/CreateOrUpdateCategoryDialog.razor.cs b/src/BackOffice/Pages/Category/CreateOrUpdateCategoryDialog.razor.cs new file mode 100644 index 0000000..aba60fd --- /dev/null +++ b/src/BackOffice/Pages/Category/CreateOrUpdateCategoryDialog.razor.cs @@ -0,0 +1,86 @@ +using BackOffice.BFF.Category.Protobuf.Protos.Category; +using Microsoft.AspNetCore.Components; +using MudBlazor; + +namespace BackOffice.Pages.Category; + +public partial class CreateOrUpdateCategoryDialog +{ + [CascadingParameter] private IMudDialogInstance MudDialog { get; set; } = default!; + [Inject] public CategoryContract.CategoryContractClient CategoryContract { get; set; } = default!; + + [Parameter] public CreateNewCategoryRequest? Model { get; set; } + [Parameter] public UpdateCategoryRequest? EditModel { get; set; } + + private bool IsEdit => EditModel != null; + + private string Name { get; set; } = string.Empty; + private string Title { get; set; } = string.Empty; + private string? Description { get; set; } + private long? ParentId { get; set; } + private bool IsActive { get; set; } = true; + private int SortOrder { get; set; } + + protected override void OnInitialized() + { + if (IsEdit && EditModel != null) + { + Name = EditModel.Name; + Title = EditModel.Title; + Description = EditModel.Description; + ParentId = EditModel.ParentId; + IsActive = EditModel.IsActive; + SortOrder = EditModel.SortOrder; + } + else if (Model != null) + { + Name = Model.Name; + Title = Model.Title; + Description = Model.Description; + ParentId = Model.ParentId; + IsActive = Model.IsActive; + SortOrder = Model.SortOrder; + } + } + + private async Task Save() + { + if (IsEdit && EditModel != null) + { + var request = new UpdateCategoryRequest + { + Id = EditModel.Id, + Name = Name, + Title = Title, + Description = Description, + ParentId = ParentId, + IsActive = IsActive, + SortOrder = SortOrder + }; + + await CategoryContract.UpdateCategoryAsync(request); + } + else + { + var request = new CreateNewCategoryRequest + { + Name = Name, + Title = Title, + Description = Description, + ParentId = ParentId, + IsActive = IsActive, + SortOrder = SortOrder + }; + + await CategoryContract.CreateNewCategoryAsync(request); + } + + MudDialog.Close(DialogResult.Ok(true)); + } + + private void Cancel() + { + MudDialog.Cancel(); + } +} + diff --git a/src/BackOffice/Pages/Login/LoginPage.razor.cs b/src/BackOffice/Pages/Login/LoginPage.razor.cs index 6f7c8d0..d796916 100644 --- a/src/BackOffice/Pages/Login/LoginPage.razor.cs +++ b/src/BackOffice/Pages/Login/LoginPage.razor.cs @@ -15,7 +15,6 @@ public partial class LoginPage [Inject] public OtpContract.OtpContractClient OtpContract { get; set; } private async Task OnSubmitClick() { - Console.WriteLine(OtpContract == null); await _form.Validate(); if (!_form.IsValid) return; @@ -31,11 +30,12 @@ public partial class LoginPage } catch (Exception ex) { - Snackbar.Add(message: ex.Message, severity: Severity.Error, null); + var message = ex.Message.ExtractUserFriendlyMessage(); + Snackbar.Add(message: message, severity: Severity.Error); } _isLoading = false; StateHasChanged(); } -} \ No newline at end of file +} diff --git a/src/BackOffice/Pages/Login/VerifyCodePage.razor.cs b/src/BackOffice/Pages/Login/VerifyCodePage.razor.cs index 2e78f39..975d015 100644 --- a/src/BackOffice/Pages/Login/VerifyCodePage.razor.cs +++ b/src/BackOffice/Pages/Login/VerifyCodePage.razor.cs @@ -45,7 +45,8 @@ public partial class VerifyCodePage } catch (Exception ex) { - Snackbar.Add(message: ex.Message, severity: Severity.Error, null); + var message = ex.Message.ExtractUserFriendlyMessage(); + Snackbar.Add(message: message, severity: Severity.Error); } _isLoading = false; @@ -68,7 +69,8 @@ public partial class VerifyCodePage } catch (Exception ex) { - Snackbar.Add(message: ex.Message, severity: Severity.Error, null); + var message = ex.Message.ExtractUserFriendlyMessage(); + Snackbar.Add(message: message, severity: Severity.Error); } _isLoading = false; @@ -94,4 +96,4 @@ public partial class VerifyCodePage } }), null, 1000, 1000); } -} \ No newline at end of file +} diff --git a/src/BackOffice/Pages/Package/Components/CreateDialog.razor.cs b/src/BackOffice/Pages/Package/Components/CreateDialog.razor.cs index 2e3612e..6a5a847 100644 --- a/src/BackOffice/Pages/Package/Components/CreateDialog.razor.cs +++ b/src/BackOffice/Pages/Package/Components/CreateDialog.razor.cs @@ -11,7 +11,7 @@ public partial class CreateDialog { [Inject] public PackageContract.PackageContractClient PackageContract { get; set; } [Parameter] public CreateNewPackageRequest Model { get; set; } = new(); - [CascadingParameter] MudDialogInstance MudDialog { get; set; } + [CascadingParameter] IMudDialogInstance MudDialog { get; set; } private bool _isLoading = false; private string _srcImage = ""; @@ -69,9 +69,8 @@ public partial class CreateDialog Submit(); } - catch (RpcException ex) + catch (RpcException) { - Console.WriteLine(ex); Snackbar.Add("خطای سرور", Severity.Error); } @@ -80,4 +79,4 @@ public partial class CreateDialog } void Submit() => MudDialog.Close(DialogResult.Ok(true)); void Cancel() => MudDialog.Cancel(); -} \ No newline at end of file +} diff --git a/src/BackOffice/Pages/Package/Components/UpdateDialog.razor.cs b/src/BackOffice/Pages/Package/Components/UpdateDialog.razor.cs index 31ce4f9..464d7d0 100644 --- a/src/BackOffice/Pages/Package/Components/UpdateDialog.razor.cs +++ b/src/BackOffice/Pages/Package/Components/UpdateDialog.razor.cs @@ -15,7 +15,7 @@ namespace BackOffice.Pages.Package.Components public UpdatePackageRequest Model { get; set; } = new(); [CascadingParameter] - MudDialogInstance MudDialog { get; set; } + IMudDialogInstance MudDialog { get; set; } private string _srcImage = ""; private IBrowserFile? _imageFile; private long _maxAllowedSize = (1024 * 1024) * 200; diff --git a/src/BackOffice/Pages/Package/PackageMainPage.razor b/src/BackOffice/Pages/Package/PackageMainPage.razor index e5e43ad..ea3741c 100644 --- a/src/BackOffice/Pages/Package/PackageMainPage.razor +++ b/src/BackOffice/Pages/Package/PackageMainPage.razor @@ -5,7 +5,7 @@ @using BackOffice.Common.BaseComponents @using DataModel = BackOffice.BFF.Package.Protobuf.Protos.Package.GetAllPackageByFilterResponseModel - + @@ -93,4 +93,3 @@ - diff --git a/src/BackOffice/Pages/Package/PackageMainPage.razor.cs b/src/BackOffice/Pages/Package/PackageMainPage.razor.cs index cf7cfc4..05496db 100644 --- a/src/BackOffice/Pages/Package/PackageMainPage.razor.cs +++ b/src/BackOffice/Pages/Package/PackageMainPage.razor.cs @@ -82,16 +82,16 @@ public partial class PackageMainPage public async Task OnFilterSubmit() { - _basePage.IsFilterd = true; + _basePage.IsFiltered = true; StateHasChanged(); ReLoadData(); } - public async Task OnFilterCleard() + public async Task OnFilterCleared() { - _basePage.IsFilterd = false; + _basePage.IsFiltered = false; StateHasChanged(); _request = new() { Filter = new() { } }; ReLoadData(); } -} \ No newline at end of file +} diff --git a/src/BackOffice/Pages/Products/Components/CreateDialog.razor b/src/BackOffice/Pages/Products/Components/CreateDialog.razor new file mode 100644 index 0000000..ea53519 --- /dev/null +++ b/src/BackOffice/Pages/Products/Components/CreateDialog.razor @@ -0,0 +1,87 @@ +@using BackOffice.BFF.Products.Protobuf.Protos.Products +@using Microsoft.AspNetCore.Components.Forms +@using Tizzani.MudBlazor.HtmlEditor +@using BackOffice.Common.BaseComponents + + + + + + @if (!string.IsNullOrWhiteSpace(_mainImagePreview)) + { + + } + else + { + + تصویری انتخاب نشده است + + } + + + + + انتخاب تصویر اصلی + + + + @if (context != null) + { + + + @context.Name + + + } + else + { + فایلی انتخاب نشده + } + + + + + + + + + + + + + + + + + + + + + + + تصویر بندانگشتی + + + + انتخاب تصویر بندانگشتی (اختیاری) + + + + + + + + لغو + ثبت + + diff --git a/src/BackOffice/Pages/Products/Components/CreateDialog.razor.cs b/src/BackOffice/Pages/Products/Components/CreateDialog.razor.cs new file mode 100644 index 0000000..3080a9e --- /dev/null +++ b/src/BackOffice/Pages/Products/Components/CreateDialog.razor.cs @@ -0,0 +1,91 @@ +using BackOffice.BFF.Products.Protobuf.Protos.Products; +using Google.Protobuf; +using Microsoft.AspNetCore.Components; +using Microsoft.AspNetCore.Components.Forms; +using MudBlazor; + +namespace BackOffice.Pages.Products.Components; + +public partial class CreateDialog +{ + [CascadingParameter] IMudDialogInstance MudDialog { get; set; } = default!; + [Inject] public ProductsContract.ProductsContractClient ProductsContract { get; set; } = default!; + [Parameter] public CreateNewProductsRequest Model { get; set; } = default!; + + private IBrowserFile? _mainImageFile; + private IBrowserFile? _thumbnailImageFile; + private readonly long _maxAllowedSize = (1024 * 1024) * 5; + private bool _isLoading; + private string? _mainImagePreview; + + private async Task OnMainImageSelected(IBrowserFile? file) + { + _mainImageFile = file; + if (file != null) + { + var buffer = new byte[file.Size]; + await file.OpenReadStream(_maxAllowedSize).ReadAsync(buffer); + _mainImagePreview = $"data:{file.ContentType};base64," + Convert.ToBase64String(buffer); + StateHasChanged(); + } + } + + private async Task OnThumbnailSelected(IBrowserFile? file) + { + _thumbnailImageFile = file; + } + + public async void CallCreateMethod() + { + _isLoading = true; + StateHasChanged(); + + if (string.IsNullOrWhiteSpace(Model.Title)) + { + Snackbar.Add("لطفا عنوان را وارد کنید!", Severity.Warning); + _isLoading = false; + StateHasChanged(); + return; + } + + if (_mainImageFile != null) + { + var buffer = new byte[_mainImageFile.Size]; + await _mainImageFile.OpenReadStream(_maxAllowedSize).ReadAsync(buffer); + Model.ImageFile = new ImageFileModel + { + File = ByteString.CopyFrom(buffer), + Mime = _mainImageFile.ContentType, + FileName = Path.GetFileNameWithoutExtension(_mainImageFile.Name) + }; + } + + if (_thumbnailImageFile != null) + { + var buffer = new byte[_thumbnailImageFile.Size]; + await _thumbnailImageFile.OpenReadStream(_maxAllowedSize).ReadAsync(buffer); + Model.ThumbnailFile = new ImageFileModel + { + File = ByteString.CopyFrom(buffer), + Mime = _thumbnailImageFile.ContentType, + FileName = Path.GetFileNameWithoutExtension(_thumbnailImageFile.Name) + }; + } + + try + { + await ProductsContract.CreateNewProductsAsync(Model); + Submit(); + } + catch + { + Snackbar.Add("خطای سرور", Severity.Error); + } + + _isLoading = false; + StateHasChanged(); + } + + void Submit() => MudDialog.Close(DialogResult.Ok(true)); + void Cancel() => MudDialog.Cancel(); +} diff --git a/src/BackOffice/Pages/Products/Components/GalleryDialog.razor b/src/BackOffice/Pages/Products/Components/GalleryDialog.razor new file mode 100644 index 0000000..a821e9b --- /dev/null +++ b/src/BackOffice/Pages/Products/Components/GalleryDialog.razor @@ -0,0 +1,83 @@ +@using BackOffice.BFF.Products.Protobuf.Protos.Products +@using Microsoft.AspNetCore.Components.Forms + + + + + گالری تصاویر - @ProductTitle + + + + + + + + + + انتخاب تصویر + + + + @if (context != null) + { + @context.Name + } + else + { + فایلی انتخاب نشده + } + + + + @if (!string.IsNullOrWhiteSpace(_previewImage)) + { + + + + } + + افزودن به گالری + + + + + @if (Items?.Count > 0) + { + + @foreach (var item in Items) + { + + + + @item.Title + + + + + + } + + } + else + { + هنوز تصویری برای این محصول ثبت نشده است. + } + + + + + + بستن + + diff --git a/src/BackOffice/Pages/Products/Components/GalleryDialog.razor.cs b/src/BackOffice/Pages/Products/Components/GalleryDialog.razor.cs new file mode 100644 index 0000000..186724e --- /dev/null +++ b/src/BackOffice/Pages/Products/Components/GalleryDialog.razor.cs @@ -0,0 +1,129 @@ +using BackOffice.BFF.Products.Protobuf.Protos.Products; +using Google.Protobuf; +using Microsoft.AspNetCore.Components; +using Microsoft.AspNetCore.Components.Forms; +using MudBlazor; + +namespace BackOffice.Pages.Products.Components; + +public partial class GalleryDialog +{ + [CascadingParameter] IMudDialogInstance MudDialog { get; set; } = default!; + [Inject] public ProductsContract.ProductsContractClient ProductsContract { get; set; } = default!; + + [Parameter] public long ProductId { get; set; } + [Parameter] public string ProductTitle { get; set; } = string.Empty; + + private List Items { get; set; } = new List(); + private IBrowserFile? _file; + private string? _title; + private readonly long _maxAllowedSize = (1024 * 1024) * 5; + private bool _isUploading; + private string? _previewImage; + private byte[]? _imageBuffer; + + protected override async Task OnInitializedAsync() + { + if (string.IsNullOrWhiteSpace(_title)) + _title = ProductTitle; + + await LoadGallery(); + } + + private async Task LoadGallery() + { + var response = await ProductsContract.GetProductGalleryAsync(new GetProductGalleryRequest + { + ProductId = ProductId + }); + + Items = response?.Items? + .Select(x => new GalleryItemViewModel + { + ProductGalleryId = x.ProductGalleryId, + ProductImageId = x.ProductImageId, + Title = x.Title, + ImagePath = x.ImagePath, + ImageThumbnailPath = x.ImageThumbnailPath + }) + .ToList() ?? new List(); + StateHasChanged(); + } + + private async Task OnImageSelected(IBrowserFile? file) + { + _file = file; + _previewImage = null; + _imageBuffer = null; + + if (file != null) + { + var buffer = new byte[file.Size]; + await file.OpenReadStream(_maxAllowedSize).ReadAsync(buffer); + _imageBuffer = buffer; + _previewImage = $"data:{file.ContentType};base64," + Convert.ToBase64String(buffer); + StateHasChanged(); + } + } + + private async Task AddImage() + { + if (_file == null) + return; + + _isUploading = true; + StateHasChanged(); + + if (_imageBuffer == null) + { + var buffer = new byte[_file.Size]; + await _file.OpenReadStream(_maxAllowedSize).ReadAsync(buffer); + _imageBuffer = buffer; + } + + var request = new AddProductImageRequest + { + ProductId = ProductId, + Title = _title ?? string.Empty, + ImageFile = new ImageFileModel + { + File = ByteString.CopyFrom(_imageBuffer), + Mime = _file.ContentType, + FileName = Path.GetFileNameWithoutExtension(_file.Name) + } + }; + + await ProductsContract.AddProductImageAsync(request); + + _file = null; + _title = string.Empty; + _previewImage = null; + _imageBuffer = null; + + await LoadGallery(); + + _isUploading = false; + StateHasChanged(); + } + + private async Task RemoveImage(GalleryItemViewModel item) + { + await ProductsContract.RemoveProductImageAsync(new RemoveProductImageRequest + { + ProductGalleryId = item.ProductGalleryId + }); + + await LoadGallery(); + } + + private void Close() => MudDialog.Close(); + + private class GalleryItemViewModel + { + public long ProductGalleryId { get; set; } + public long ProductImageId { get; set; } + public string Title { get; set; } + public string ImagePath { get; set; } + public string ImageThumbnailPath { get; set; } + } +} diff --git a/src/BackOffice/Pages/Products/Components/ImagePreviewDialog.razor b/src/BackOffice/Pages/Products/Components/ImagePreviewDialog.razor new file mode 100644 index 0000000..75b3577 --- /dev/null +++ b/src/BackOffice/Pages/Products/Components/ImagePreviewDialog.razor @@ -0,0 +1,25 @@ +@using BackOffice.Common.BaseComponents + + + + + @Title + @if (!string.IsNullOrWhiteSpace(ImageUrl)) + { + + } + else + { + تصویری برای نمایش وجود ندارد. + } + + + + بستن + + + diff --git a/src/BackOffice/Pages/Products/Components/ImagePreviewDialog.razor.cs b/src/BackOffice/Pages/Products/Components/ImagePreviewDialog.razor.cs new file mode 100644 index 0000000..fd4694c --- /dev/null +++ b/src/BackOffice/Pages/Products/Components/ImagePreviewDialog.razor.cs @@ -0,0 +1,15 @@ +using Microsoft.AspNetCore.Components; +using MudBlazor; + +namespace BackOffice.Pages.Products.Components; + +public partial class ImagePreviewDialog +{ + [CascadingParameter] IMudDialogInstance MudDialog { get; set; } = default!; + + [Parameter] public string ImageUrl { get; set; } = string.Empty; + [Parameter] public string Title { get; set; } = string.Empty; + + private void Close() => MudDialog.Close(); +} + diff --git a/src/BackOffice/Pages/Products/Components/UpdateDialog.razor b/src/BackOffice/Pages/Products/Components/UpdateDialog.razor new file mode 100644 index 0000000..307542b --- /dev/null +++ b/src/BackOffice/Pages/Products/Components/UpdateDialog.razor @@ -0,0 +1,73 @@ +@using BackOffice.BFF.Products.Protobuf.Protos.Products +@using Microsoft.AspNetCore.Components.Forms +@using Tizzani.MudBlazor.HtmlEditor +@using BackOffice.Common.BaseComponents + + + + + + @if (!string.IsNullOrWhiteSpace(_mainImagePreview)) + { + + } + else + { + + تصویری انتخاب نشده است + + } + + + + + تغییر تصویر اصلی + + + + + + + + + + + + + + + + + + + + + + + + تصویر بندانگشتی + + + + تغییر تصویر بندانگشتی (اختیاری) + + + + + + + + لغو + ویرایش + + diff --git a/src/BackOffice/Pages/Products/Components/UpdateDialog.razor.cs b/src/BackOffice/Pages/Products/Components/UpdateDialog.razor.cs new file mode 100644 index 0000000..231982b --- /dev/null +++ b/src/BackOffice/Pages/Products/Components/UpdateDialog.razor.cs @@ -0,0 +1,97 @@ +using BackOffice.BFF.Products.Protobuf.Protos.Products; +using Google.Protobuf; +using Microsoft.AspNetCore.Components; +using Microsoft.AspNetCore.Components.Forms; +using MudBlazor; + +namespace BackOffice.Pages.Products.Components; + +public partial class UpdateDialog +{ + [CascadingParameter] IMudDialogInstance MudDialog { get; set; } = default!; + [Inject] public ProductsContract.ProductsContractClient ProductsContract { get; set; } = default!; + [Parameter] public UpdateProductsRequest Model { get; set; } = default!; + + private IBrowserFile? _mainImageFile; + private IBrowserFile? _thumbnailImageFile; + private readonly long _maxAllowedSize = (1024 * 1024) * 5; + private bool _isLoading; + private string? _mainImagePreview; + + private async Task OnMainImageSelected(IBrowserFile? file) + { + _mainImageFile = file; + if (file != null) + { + var buffer = new byte[file.Size]; + await file.OpenReadStream(_maxAllowedSize).ReadAsync(buffer); + _mainImagePreview = $"data:{file.ContentType};base64," + Convert.ToBase64String(buffer); + StateHasChanged(); + } + } + + private async Task OnThumbnailSelected(IBrowserFile? file) + { + _thumbnailImageFile = file; + } + + protected override Task OnInitializedAsync() + { + _mainImagePreview = Model.ImagePath; + return base.OnInitializedAsync(); + } + + public async void CallUpdateMethod() + { + _isLoading = true; + StateHasChanged(); + + if (string.IsNullOrWhiteSpace(Model.Title)) + { + Snackbar.Add("لطفا عنوان را وارد کنید!", Severity.Warning); + _isLoading = false; + StateHasChanged(); + return; + } + + if (_mainImageFile != null) + { + var buffer = new byte[_mainImageFile.Size]; + await _mainImageFile.OpenReadStream(_maxAllowedSize).ReadAsync(buffer); + Model.ImageFile = new ImageFileModel + { + File = ByteString.CopyFrom(buffer), + Mime = _mainImageFile.ContentType, + FileName = Path.GetFileNameWithoutExtension(_mainImageFile.Name) + }; + } + + if (_thumbnailImageFile != null) + { + var buffer = new byte[_thumbnailImageFile.Size]; + await _thumbnailImageFile.OpenReadStream(_maxAllowedSize).ReadAsync(buffer); + Model.ThumbnailFile = new ImageFileModel + { + File = ByteString.CopyFrom(buffer), + Mime = _thumbnailImageFile.ContentType, + FileName = Path.GetFileNameWithoutExtension(_thumbnailImageFile.Name) + }; + } + + try + { + await ProductsContract.UpdateProductsAsync(Model); + Submit(); + } + catch + { + Snackbar.Add("خطای سرور", Severity.Error); + } + + _isLoading = false; + StateHasChanged(); + } + + void Submit() => MudDialog.Close(DialogResult.Ok(true)); + void Cancel() => MudDialog.Cancel(); +} diff --git a/src/BackOffice/Pages/Products/ProductsMainPage.razor b/src/BackOffice/Pages/Products/ProductsMainPage.razor new file mode 100644 index 0000000..acdb396 --- /dev/null +++ b/src/BackOffice/Pages/Products/ProductsMainPage.razor @@ -0,0 +1,85 @@ +@attribute [Route(RouteConstance.Products)] + +@using BackOffice.BFF.Products.Protobuf.Protos.Products +@using BackOffice.Common.BaseComponents +@using BackOffice.Pages.Products.Components +@using DataModel = BackOffice.BFF.Products.Protobuf.Protos.Products.GetAllProductsByFilterResponseModel + + + + + + + + + + + + + + + مدیریت محصولات + + + افزودن + + + + + + + + + + OpenImagePreview(context.Item.ImagePath, context.Item.Title))"> + + + + @if (string.IsNullOrWhiteSpace(context.Item.Title)) + { + - + } + else + { + + + @(context.Item.Title.Truncate(20, true)) + + + } + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/BackOffice/Pages/Products/ProductsMainPage.razor.cs b/src/BackOffice/Pages/Products/ProductsMainPage.razor.cs new file mode 100644 index 0000000..8d4424e --- /dev/null +++ b/src/BackOffice/Pages/Products/ProductsMainPage.razor.cs @@ -0,0 +1,123 @@ +using BackOffice.BFF.Products.Protobuf.Protos.Products; +using BackOffice.Common.BaseComponents; +using BackOffice.Pages.Products.Components; +using Mapster; +using Microsoft.AspNetCore.Components; +using MudBlazor; +using DataModel = BackOffice.BFF.Products.Protobuf.Protos.Products.GetAllProductsByFilterResponseModel; + +namespace BackOffice.Pages.Products; + +public partial class ProductsMainPage +{ + [Inject] public ProductsContract.ProductsContractClient ProductsContract { get; set; } = default!; + + private bool _isLoading = true; + private MudDataGrid _gridData; + private BasePageComponent _basePage; + private GetAllProductsByFilterRequest _request = new() { Filter = new() }; + + private async Task> ServerReload(GridState state) + { + _request.Filter ??= new(); + _request.PaginationState ??= new(); + _request.PaginationState.PageNumber = state.Page + 1; + _request.PaginationState.PageSize = state.PageSize; + + var result = await ProductsContract.GetAllProductsByFilterAsync(_request); + if (result != null && result.Models != null && result.Models.Any()) + { + return new GridData { Items = result.Models.ToList(), TotalItems = (int)result.MetaData.TotalCount }; + } + + return new GridData(); + } + + public async Task Update(DataModel model) + { + var parameters = new DialogParameters { { x => x.Model, model.Adapt() } }; + + var dialog = await DialogService.ShowAsync("ویرایش محصول", parameters, new DialogOptions { CloseButton = true, FullWidth = true, MaxWidth = MaxWidth.Small }); + var result = await dialog.Result; + + if (!result.Canceled) + { + ReLoadData(); + Snackbar.Add("عملیات با موفقیت انجام شد", Severity.Success); + } + } + + private async Task OnDelete(DataModel model) + { + var options = new DialogOptions { CloseOnEscapeKey = true, MaxWidth = MaxWidth.Small }; + bool? result = await DialogService.ShowMessageBox( + "اخطار", + "آیا از حذف این مورد مطمئن هستید؟", + yesText: "حذف", cancelText: "لغو", + options: options); + if (result == true) + { + await ProductsContract.DeleteProductsAsync(new DeleteProductsRequest { Id = model.Id }); + ReLoadData(); + } + StateHasChanged(); + } + + public async void ReLoadData() + { + if (_gridData != null) + await _gridData.ReloadServerData(); + } + + public async Task CreateNew() + { + var dialog = await DialogService.ShowAsync("افزودن محصول", new DialogParameters { { x => x.Model, new CreateNewProductsRequest() } }, new DialogOptions { CloseButton = true, FullWidth = true, MaxWidth = MaxWidth.Small }); + var result = await dialog.Result; + if (!result.Canceled) + { + ReLoadData(); + Snackbar.Add("عملیات با موفقیت انجام شد", Severity.Success); + } + } + + public async Task OnFilterSubmit() + { + _basePage.IsFiltered = true; + StateHasChanged(); + ReLoadData(); + } + + public async Task OnFilterCleared() + { + _basePage.IsFiltered = false; + StateHasChanged(); + _request = new GetAllProductsByFilterRequest { Filter = new() }; + ReLoadData(); + } + + public async Task OpenGallery(DataModel model) + { + var parameters = new DialogParameters + { + { x => x.ProductId, model.Id }, + { x => x.ProductTitle, model.Title } + }; + await DialogService.ShowAsync("گالری تصاویر", parameters, + new DialogOptions { CloseButton = true, FullWidth = true, MaxWidth = MaxWidth.Medium }); + } + + public async Task OpenImagePreview(string imagePath, string title) + { + if (string.IsNullOrWhiteSpace(imagePath)) + return; + + var parameters = new DialogParameters + { + { x => x.ImageUrl, imagePath }, + { x => x.Title, title } + }; + + await DialogService.ShowAsync("پیشنمایش تصویر محصول", parameters, + new DialogOptions { CloseButton = true, MaxWidth = MaxWidth.Medium, FullWidth = true }); + } +} diff --git a/src/BackOffice/Pages/Role/Components/CreateDialog.razor.cs b/src/BackOffice/Pages/Role/Components/CreateDialog.razor.cs index aa0aa4e..2c0c776 100644 --- a/src/BackOffice/Pages/Role/Components/CreateDialog.razor.cs +++ b/src/BackOffice/Pages/Role/Components/CreateDialog.razor.cs @@ -11,7 +11,7 @@ public partial class CreateDialog { [Inject] public RoleContract.RoleContractClient RoleContract { get; set; } [Parameter] public CreateNewRoleRequest Model { get; set; } = new(); - [CascadingParameter] MudDialogInstance MudDialog { get; set; } + [CascadingParameter] IMudDialogInstance MudDialog { get; set; } private bool _isLoading = false; private string _srcImage = ""; @@ -53,9 +53,8 @@ public partial class CreateDialog Submit(); } - catch (RpcException ex) + catch (RpcException) { - Console.WriteLine(ex); Snackbar.Add("خطای سرور", Severity.Error); } @@ -64,4 +63,4 @@ public partial class CreateDialog } void Submit() => MudDialog.Close(DialogResult.Ok(true)); void Cancel() => MudDialog.Cancel(); -} \ No newline at end of file +} diff --git a/src/BackOffice/Pages/Role/Components/UpdateDialog.razor.cs b/src/BackOffice/Pages/Role/Components/UpdateDialog.razor.cs index 152800c..82236e0 100644 --- a/src/BackOffice/Pages/Role/Components/UpdateDialog.razor.cs +++ b/src/BackOffice/Pages/Role/Components/UpdateDialog.razor.cs @@ -15,7 +15,7 @@ namespace BackOffice.Pages.Role.Components public UpdateRoleRequest Model { get; set; } = new(); [CascadingParameter] - MudDialogInstance MudDialog { get; set; } + IMudDialogInstance MudDialog { get; set; } public async void CallUpdateMethod() { diff --git a/src/BackOffice/Pages/Role/RoleMainPage.razor b/src/BackOffice/Pages/Role/RoleMainPage.razor index 7057408..a8b0f09 100644 --- a/src/BackOffice/Pages/Role/RoleMainPage.razor +++ b/src/BackOffice/Pages/Role/RoleMainPage.razor @@ -5,7 +5,7 @@ @using BackOffice.Common.BaseComponents @using DataModel = BackOffice.BFF.Role.Protobuf.Protos.Role.GetAllRoleByFilterResponseModel - + @@ -57,4 +57,3 @@ - diff --git a/src/BackOffice/Pages/Role/RoleMainPage.razor.cs b/src/BackOffice/Pages/Role/RoleMainPage.razor.cs index 0fdb165..506af20 100644 --- a/src/BackOffice/Pages/Role/RoleMainPage.razor.cs +++ b/src/BackOffice/Pages/Role/RoleMainPage.razor.cs @@ -81,16 +81,16 @@ public partial class RoleMainPage } public async Task OnFilterSubmit() { - _basePage.IsFilterd = true; + _basePage.IsFiltered = true; StateHasChanged(); ReLoadData(); } - public async Task OnFilterCleard() + public async Task OnFilterCleared() { - _basePage.IsFilterd = false; + _basePage.IsFiltered = false; StateHasChanged(); _request = new() { Filter = new() { } }; ReLoadData(); } -} \ No newline at end of file +} diff --git a/src/BackOffice/Pages/User/Components/UserRoleDialog.razor.cs b/src/BackOffice/Pages/User/Components/UserRoleDialog.razor.cs index e212b36..7fc56cf 100644 --- a/src/BackOffice/Pages/User/Components/UserRoleDialog.razor.cs +++ b/src/BackOffice/Pages/User/Components/UserRoleDialog.razor.cs @@ -8,7 +8,7 @@ namespace BackOffice.Pages.User.Components; public partial class UserRoleDialog { [Parameter] public long UserId { get; set; } - [CascadingParameter] MudDialogInstance MudDialog { get; set; } + [CascadingParameter] IMudDialogInstance MudDialog { get; set; } MudForm _form; private bool _isLoading; private bool _isFirstLoad = true; diff --git a/src/BackOffice/Pages/User/UserMainPage.razor b/src/BackOffice/Pages/User/UserMainPage.razor index b8ea98e..aec662d 100644 --- a/src/BackOffice/Pages/User/UserMainPage.razor +++ b/src/BackOffice/Pages/User/UserMainPage.razor @@ -5,7 +5,7 @@ @using BackOffice.Common.BaseComponents @using DataModel = BackOffice.BFF.User.Protobuf.Protos.User.GetAllUserByFilterResponseModel - + @@ -64,4 +64,3 @@ - diff --git a/src/BackOffice/Pages/User/UserMainPage.razor.cs b/src/BackOffice/Pages/User/UserMainPage.razor.cs index 9dda340..53ec385 100644 --- a/src/BackOffice/Pages/User/UserMainPage.razor.cs +++ b/src/BackOffice/Pages/User/UserMainPage.razor.cs @@ -51,30 +51,6 @@ public partial class UserMainPage if (_gridData != null) await _gridData.ReloadServerData(); } - private async Task Searchfirstname(string? firstname) - { - _request.Filter ??= new(); - _request.Filter.FirstName = string.IsNullOrWhiteSpace(firstname) ? default : firstname; - ReLoadData(); - } - private async Task Searchlastname(string? lastname) - { - _request.Filter ??= new(); - _request.Filter.LastName = string.IsNullOrWhiteSpace(lastname) ? default : lastname; - ReLoadData(); - } - private async Task SearchNationalCode(string? nationalcode) - { - _request.Filter ??= new(); - _request.Filter.NationalCode = string.IsNullOrWhiteSpace(nationalcode) ? default : nationalcode; - ReLoadData(); - } - private async Task SearchMobile(string? mobile) - { - _request.Filter ??= new(); - _request.Filter.Mobile = string.IsNullOrWhiteSpace(mobile) ? default : mobile; - ReLoadData(); - } public async Task OnclickUserRoleDialog(long userid) { var dialog = await DialogService.ShowAsync($"نقش های کاربر", new DialogParameters() { { x => x.UserId, userid } }, new DialogOptions() { CloseButton = true, FullWidth = true, MaxWidth = MaxWidth.Small }); @@ -87,16 +63,16 @@ public partial class UserMainPage } public async Task OnFilterSubmit() { - _basePage.IsFilterd = true; + _basePage.IsFiltered = true; StateHasChanged(); ReLoadData(); } - public async Task OnFilterCleard() + public async Task OnFilterCleared() { - _basePage.IsFilterd = false; + _basePage.IsFiltered = false; StateHasChanged(); _request = new() { Filter = new() { } }; ReLoadData(); } -} \ No newline at end of file +} diff --git a/src/BackOffice/Pages/UserAddress/Components/CreateDialog.razor.cs b/src/BackOffice/Pages/UserAddress/Components/CreateDialog.razor.cs index d2142c7..ae95782 100644 --- a/src/BackOffice/Pages/UserAddress/Components/CreateDialog.razor.cs +++ b/src/BackOffice/Pages/UserAddress/Components/CreateDialog.razor.cs @@ -11,7 +11,7 @@ public partial class CreateDialog { [Inject] public UserAddressContract.UserAddressContractClient UserAddressContract { get; set; } [Parameter] public CreateNewUserAddressRequest Model { get; set; } = new(); - [CascadingParameter] MudDialogInstance MudDialog { get; set; } + [CascadingParameter] IMudDialogInstance MudDialog { get; set; } private bool _isLoading = false; @@ -49,9 +49,8 @@ public partial class CreateDialog Submit(); } - catch (RpcException ex) + catch (RpcException) { - Console.WriteLine(ex); Snackbar.Add("خطای سرور", Severity.Error); } @@ -60,4 +59,4 @@ public partial class CreateDialog } void Submit() => MudDialog.Close(DialogResult.Ok(true)); void Cancel() => MudDialog.Cancel(); -} \ No newline at end of file +} diff --git a/src/BackOffice/Pages/UserAddress/Components/UpdateDialog.razor.cs b/src/BackOffice/Pages/UserAddress/Components/UpdateDialog.razor.cs index 212edaa..0c884a4 100644 --- a/src/BackOffice/Pages/UserAddress/Components/UpdateDialog.razor.cs +++ b/src/BackOffice/Pages/UserAddress/Components/UpdateDialog.razor.cs @@ -15,7 +15,7 @@ namespace BackOffice.Pages.UserAddress.Components public UpdateUserAddressRequest Model { get; set; } = new(); [CascadingParameter] - MudDialogInstance MudDialog { get; set; } + IMudDialogInstance MudDialog { get; set; } public async void CallUpdateMethod() { _isLoading = true; diff --git a/src/BackOffice/Pages/UserAddress/UserAddressMainPage.razor b/src/BackOffice/Pages/UserAddress/UserAddressMainPage.razor index ecb2f13..2a34c45 100644 --- a/src/BackOffice/Pages/UserAddress/UserAddressMainPage.razor +++ b/src/BackOffice/Pages/UserAddress/UserAddressMainPage.razor @@ -5,7 +5,7 @@ @using BackOffice.Common.BaseComponents @using DataModel = BackOffice.BFF.UserAddress.Protobuf.Protos.UserAddress.GetAllUserAddressByFilterResponseModel - + @@ -54,4 +54,3 @@ - diff --git a/src/BackOffice/Pages/UserAddress/UserAddressMainPage.razor.cs b/src/BackOffice/Pages/UserAddress/UserAddressMainPage.razor.cs index 4525371..d2e4d53 100644 --- a/src/BackOffice/Pages/UserAddress/UserAddressMainPage.razor.cs +++ b/src/BackOffice/Pages/UserAddress/UserAddressMainPage.razor.cs @@ -83,16 +83,16 @@ public partial class UserAddressMainPage } public async Task OnFilterSubmit() { - _basePage.IsFilterd = true; + _basePage.IsFiltered = true; StateHasChanged(); ReLoadData(); } - public async Task OnFilterCleard() + public async Task OnFilterCleared() { - _basePage.IsFilterd = false; + _basePage.IsFiltered = false; StateHasChanged(); _request = new() { Filter = new() { } }; ReLoadData(); } -} \ No newline at end of file +} diff --git a/src/BackOffice/Pages/UserOrder/UserOrderMainPage.razor b/src/BackOffice/Pages/UserOrder/UserOrderMainPage.razor index c8a477d..60cd716 100644 --- a/src/BackOffice/Pages/UserOrder/UserOrderMainPage.razor +++ b/src/BackOffice/Pages/UserOrder/UserOrderMainPage.razor @@ -5,7 +5,7 @@ @using BackOffice.Common.BaseComponents @using DataModel = BackOffice.BFF.UserOrder.Protobuf.Protos.UserOrder.GetAllUserOrderByFilterResponseModel - + @@ -65,4 +65,3 @@ - diff --git a/src/BackOffice/Pages/UserOrder/UserOrderMainPage.razor.cs b/src/BackOffice/Pages/UserOrder/UserOrderMainPage.razor.cs index 26c7ad9..dc6108a 100644 --- a/src/BackOffice/Pages/UserOrder/UserOrderMainPage.razor.cs +++ b/src/BackOffice/Pages/UserOrder/UserOrderMainPage.razor.cs @@ -56,16 +56,16 @@ public partial class UserOrderMainPage } public async Task OnFilterSubmit() { - _basePage.IsFilterd = true; + _basePage.IsFiltered = true; StateHasChanged(); ReLoadData(); } - public async Task OnFilterCleard() + public async Task OnFilterCleared() { - _basePage.IsFilterd = false; + _basePage.IsFiltered = false; StateHasChanged(); _request = new() { Filter = new() { } }; ReLoadData(); } -} \ No newline at end of file +} diff --git a/src/BackOffice/Pages/UserRole/Components/CreateDialog.razor.cs b/src/BackOffice/Pages/UserRole/Components/CreateDialog.razor.cs index 4bd7a3d..5b3e918 100644 --- a/src/BackOffice/Pages/UserRole/Components/CreateDialog.razor.cs +++ b/src/BackOffice/Pages/UserRole/Components/CreateDialog.razor.cs @@ -11,7 +11,7 @@ public partial class CreateDialog { [Inject] public UserRoleContract.UserRoleContractClient UserRoleContract { get; set; } [Parameter] public CreateNewUserRoleRequest Model { get; set; } = new(); - [CascadingParameter] MudDialogInstance MudDialog { get; set; } + [CascadingParameter] IMudDialogInstance MudDialog { get; set; } private bool _isLoading = false; diff --git a/src/BackOffice/Pages/UserRole/Components/UpdateDialog.razor.cs b/src/BackOffice/Pages/UserRole/Components/UpdateDialog.razor.cs index baebcdd..9b40628 100644 --- a/src/BackOffice/Pages/UserRole/Components/UpdateDialog.razor.cs +++ b/src/BackOffice/Pages/UserRole/Components/UpdateDialog.razor.cs @@ -15,7 +15,7 @@ namespace BackOffice.Pages.UserRole.Components public UpdateUserRoleRequest Model { get; set; } = new(); [CascadingParameter] - MudDialogInstance MudDialog { get; set; } + IMudDialogInstance MudDialog { get; set; } public async void CallUpdateMethod() { diff --git a/src/BackOffice/Pages/UserRole/UserRoleMainPage.razor b/src/BackOffice/Pages/UserRole/UserRoleMainPage.razor index dbfe885..f6ad58d 100644 --- a/src/BackOffice/Pages/UserRole/UserRoleMainPage.razor +++ b/src/BackOffice/Pages/UserRole/UserRoleMainPage.razor @@ -5,7 +5,7 @@ @using BackOffice.Common.BaseComponents @using DataModel = BackOffice.BFF.UserRole.Protobuf.Protos.UserRole.GetAllUserRoleByFilterResponseModel - + @@ -53,4 +53,3 @@ - diff --git a/src/BackOffice/Pages/UserRole/UserRoleMainPage.razor.cs b/src/BackOffice/Pages/UserRole/UserRoleMainPage.razor.cs index b397833..1026c09 100644 --- a/src/BackOffice/Pages/UserRole/UserRoleMainPage.razor.cs +++ b/src/BackOffice/Pages/UserRole/UserRoleMainPage.razor.cs @@ -83,16 +83,16 @@ public partial class UserRoleMainPage } public async Task OnFilterSubmit() { - _basePage.IsFilterd = true; + _basePage.IsFiltered = true; StateHasChanged(); ReLoadData(); } - public async Task OnFilterCleard() + public async Task OnFilterCleared() { - _basePage.IsFilterd = false; + _basePage.IsFiltered = false; StateHasChanged(); _request = new() { Filter = new() { } }; ReLoadData(); } -} \ No newline at end of file +} diff --git a/src/BackOffice/Shared/CustomMudTheme.cs b/src/BackOffice/Shared/CustomMudTheme.cs new file mode 100644 index 0000000..31a1efa --- /dev/null +++ b/src/BackOffice/Shared/CustomMudTheme.cs @@ -0,0 +1,104 @@ +using MudBlazor; + +namespace BackOffice.Shared; + +public static class CustomMudTheme +{ + public static MudTheme CustomMudBlazorTheme { get; set; } = new() + { + PaletteLight = new PaletteLight() + { + Primary = "#0380C0", + Secondary = "#5E4DF9", + PrimaryContrastText = "#FFFFFF", + AppbarText = Colors.Gray.Darken4, + Background = "#F5F5F5", + AppbarBackground = "#F5F5F5", + TextPrimary = Colors.Gray.Darken3, + Surface = "#FFFFFF", + Divider = "#B2BFCB", + DrawerBackground = "#FFFFFF", + DrawerText = Colors.Gray.Darken3, + DrawerIcon = Colors.Gray.Darken2, + }, + LayoutProperties = new LayoutProperties + { + DefaultBorderRadius = "12px", + AppbarHeight = "56px", + DrawerWidthLeft = "260px", + DrawerWidthRight = "260px" + }, + Typography = new Typography + { + Default = new DefaultTypography() + { + FontFamily = new[] { "Vazir", "Tahoma", "Segoe UI", "Arial", "sans-serif" } + }, + H1 = new H1Typography() + { + FontFamily = new[] { "Vazir" }, FontSize = "2rem", LineHeight = "1.70", FontWeight = "800", + LetterSpacing = "normal" + }, + H2 = new H2Typography() + { + FontFamily = new[] { "Vazir" }, FontSize = "1.875rem", LineHeight = "1.65", FontWeight = "800", + LetterSpacing = "normal" + }, + H3 = new H3Typography() + { + FontFamily = new[] { "Vazir" }, FontSize = "1.5rem", LineHeight = "1.60", FontWeight = "800", + LetterSpacing = "normal" + }, + H4 = new H4Typography() + { + FontFamily = new[] { "Vazir" }, FontSize = "1.25rem", LineHeight = "1.55", FontWeight = "800", + LetterSpacing = "normal" + }, + H5 = new H5Typography() + { + FontFamily = new[] { "Vazir" }, FontSize = "1.125rem", LineHeight = "1.50", FontWeight = "800", + LetterSpacing = "normal" + }, + H6 = new H6Typography() + { + FontFamily = new[] { "Vazir" }, FontSize = "1rem", LineHeight = "1.45", FontWeight = "800", + LetterSpacing = "normal" + }, + Subtitle1 = new Subtitle1Typography() + { + FontFamily = new[] { "Vazir" }, FontSize = "1rem", LineHeight = "1.62", FontWeight = "500", + LetterSpacing = "normal" + }, + Subtitle2 = new Subtitle2Typography() + { + FontFamily = new[] { "Vazir" }, FontSize = "0.875rem", LineHeight = "1.60", FontWeight = "500", + LetterSpacing = "normal" + }, + Body1 = new Body1Typography() + { + FontFamily = new[] { "Vazir" }, FontSize = "1rem", LineHeight = "1.85", FontWeight = "400", + LetterSpacing = "normal" + }, + Body2 = new Body2Typography() + { + FontFamily = new[] { "Vazir" }, FontSize = "0.875rem", LineHeight = "1.80", FontWeight = "400", + LetterSpacing = "normal" + }, + Caption = new CaptionTypography() + { + FontFamily = new[] { "Vazir" }, FontSize = "0.75rem", LineHeight = "1.60", FontWeight = "400", + LetterSpacing = "normal" + }, + Overline = new OverlineTypography() + { + FontFamily = new[] { "Vazir" }, FontSize = "0.75rem", LineHeight = "1.60", FontWeight = "500", + LetterSpacing = "normal" + }, + Button = new ButtonTypography() + { + FontFamily = new[] { "Vazir" }, FontSize = "0.875rem", LineHeight = "1.60", FontWeight = "600", + LetterSpacing = "normal", TextTransform = "none" + } + } + }; +} diff --git a/src/BackOffice/Shared/EmptyLayout.razor b/src/BackOffice/Shared/EmptyLayout.razor index 6c9e2bb..ca0e3bc 100644 --- a/src/BackOffice/Shared/EmptyLayout.razor +++ b/src/BackOffice/Shared/EmptyLayout.razor @@ -1,7 +1,7 @@ @inherits LayoutComponentBase - + @@ -13,24 +13,3 @@ - - -@code { - - - MudTheme CustomTheme = new MudTheme() - { - Typography = new Typography() - { - Default = new Default() - { - FontFamily = new[] { "IRANSans" } - - } - }, - LayoutProperties = new() - { - DrawerWidthRight = "250px" - }, - }; -} \ No newline at end of file diff --git a/src/BackOffice/Shared/MainLayout.razor b/src/BackOffice/Shared/MainLayout.razor index 541ea1d..dca9a7c 100644 --- a/src/BackOffice/Shared/MainLayout.razor +++ b/src/BackOffice/Shared/MainLayout.razor @@ -4,14 +4,14 @@ @attribute [AllowAnonymous] - + - + - مدیریت + پنل مدیریت فرصت @@ -33,4 +33,3 @@ - diff --git a/src/BackOffice/Shared/MainLayout.razor.cs b/src/BackOffice/Shared/MainLayout.razor.cs index 6aa7911..fa1153e 100644 --- a/src/BackOffice/Shared/MainLayout.razor.cs +++ b/src/BackOffice/Shared/MainLayout.razor.cs @@ -1,27 +1,24 @@ +using BackOffice.Common.Utilities; using Microsoft.AspNetCore.Components; using Microsoft.AspNetCore.Components.Authorization; using MudBlazor; namespace BackOffice.Shared; + public partial class MainLayout { - bool _drawerOpen = false; - private string Details { get; set; } - + private bool _drawerOpen; + + protected override void OnInitialized() + { + base.OnInitialized(); + GlobalConstants.ConstSnackbar = Snackbar; + } + void DrawerToggle() { _drawerOpen = !_drawerOpen; } - MudTheme CustomTheme = new MudTheme() - { - Typography = new Typography() - { - Default = new Default() - { - FontFamily = new[] { "IRANSans" } - - } - } - }; -} \ No newline at end of file + +} diff --git a/src/BackOffice/Shared/NavMenu.razor b/src/BackOffice/Shared/NavMenu.razor index a89e1eb..4409b79 100644 --- a/src/BackOffice/Shared/NavMenu.razor +++ b/src/BackOffice/Shared/NavMenu.razor @@ -1,19 +1,57 @@ -@using BackOffice.BFF.Package.Protobuf.Protos.Package -@using Microsoft.AspNetCore.Components.Authorization +@using Microsoft.AspNetCore.Components.Authorization + + + منوی اصلی + + + + داشبورد + - - داشبورد - مدیریت پکیج - مدیریت کاربر - مدیریت نقش - + + مدیریت پکیج + + + + مدیریت محصول + + + + مدیریت دستهبندی + + + + مدیریت کاربر + + + + مدیریت نقش + - - - خروج از حساب + + + + + خروج از حساب + @@ -24,4 +62,4 @@ await LocalStorageService.RemoveItemAsync("AuthToken"); Navigation.NavigateTo("/Login"); } -} \ No newline at end of file +} diff --git a/src/BackOffice/Shared/NavMenu.razor.css b/src/BackOffice/Shared/NavMenu.razor.css index e69de29..893825d 100644 --- a/src/BackOffice/Shared/NavMenu.razor.css +++ b/src/BackOffice/Shared/NavMenu.razor.css @@ -0,0 +1,33 @@ +.nav-menu { + padding: 1.5rem 0.5rem 1.5rem 0.5rem; +} + +.nav-menu__title { + padding: 0 0.75rem 0.25rem 0.75rem; + color: rgba(0, 0, 0, 0.6); +} + +.nav-menu .mud-divider { + opacity: 0.7; +} + +.nav-menu .mud-nav-link { + margin: 0.15rem 0.4rem; + border-radius: 999px; + padding-inline: 0.75rem 1rem; + font-weight: 500; +} + +.nav-menu .mud-nav-link .mud-nav-link-text { + margin-right: 0.5rem; +} + +.nav-menu .mud-nav-link.mud-nav-link-active { + background-color: rgba(3, 128, 192, 0.12); + color: #0380C0; + font-weight: 600; +} + +.nav-menu .mud-nav-link.mud-nav-link-active .mud-nav-link-icon { + color: #0380C0; +} diff --git a/src/BackOffice/_Imports.razor b/src/BackOffice/_Imports.razor index acf87fb..9bc8f09 100644 --- a/src/BackOffice/_Imports.razor +++ b/src/BackOffice/_Imports.razor @@ -11,6 +11,7 @@ @using MudBlazor @using Mapster @using DateTimeConverterCL +@using Microsoft.AspNetCore.Components.Authorization @attribute [Authorize(Roles = "Administrator, Admin, Author")] @@ -20,3 +21,4 @@ @inject IJSRuntime jsRuntime @inject NavigationManager Navigation @inject ILocalStorageService LocalStorageService +@inject AuthenticationStateProvider AuthenticationStateProvider diff --git a/src/BackOffice/wwwroot/css/app.css b/src/BackOffice/wwwroot/css/app.css index 46042f4..01a6133 100644 --- a/src/BackOffice/wwwroot/css/app.css +++ b/src/BackOffice/wwwroot/css/app.css @@ -9,6 +9,17 @@ url('../fonts/ttf/IRANSansWeb(FaNum)_Medium.ttf') format('truetype'); } +html, body { + margin: 0; + padding: 0; + height: 100%; + background-color: #f5f5f5; +} + +body { + font-family: "Vazir", "IRANSans", Tahoma, "Segoe UI", Arial, sans-serif; +} + .loading-progress { position: relative; display: block; @@ -79,4 +90,4 @@ h1:focus { .blazor-error-boundary::after { content: "An error has occurred." - } \ No newline at end of file + }