Added BlazorAppRadzenNet8SerilogLogging
This commit is contained in:
2
src/BlazorAppRadzenNet8SerilogLogging/.gitignore
vendored
Normal file
2
src/BlazorAppRadzenNet8SerilogLogging/.gitignore
vendored
Normal file
@@ -0,0 +1,2 @@
|
||||
!BlazorAppRadzenNet8SerilogLogging/Components/Pages/Log
|
||||
/BlazorAppRadzenNet8SerilogLogging/LogsFolder/logs*
|
||||
@@ -0,0 +1,25 @@
|
||||
|
||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||
# Visual Studio Version 17
|
||||
VisualStudioVersion = 17.8.34601.278
|
||||
MinimumVisualStudioVersion = 10.0.40219.1
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "BlazorAppRadzenNet8SerilogLogging", "BlazorAppRadzenNet8SerilogLogging\BlazorAppRadzenNet8SerilogLogging.csproj", "{A0D005D1-56D6-4540-B107-42D5188042A1}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|Any CPU = Debug|Any CPU
|
||||
Release|Any CPU = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||
{A0D005D1-56D6-4540-B107-42D5188042A1}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{A0D005D1-56D6-4540-B107-42D5188042A1}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{A0D005D1-56D6-4540-B107-42D5188042A1}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{A0D005D1-56D6-4540-B107-42D5188042A1}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
EndGlobalSection
|
||||
GlobalSection(ExtensibilityGlobals) = postSolution
|
||||
SolutionGuid = {EDF75B28-8549-46F9-A08B-A83715AEB2C4}
|
||||
EndGlobalSection
|
||||
EndGlobal
|
||||
@@ -0,0 +1,29 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk.Web">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net8.0</TargetFramework>
|
||||
<Nullable>enable</Nullable>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Mapster" Version="7.4.0" />
|
||||
<PackageReference Include="Microsoft.EntityFrameworkCore.InMemory" Version="8.0.5" />
|
||||
<PackageReference Include="Microsoft.EntityFrameworkCore.Sqlite" Version="8.0.5" />
|
||||
<PackageReference Include="Microsoft.AspNetCore.Diagnostics.EntityFrameworkCore" Version="8.0.5" />
|
||||
<PackageReference Include="Radzen.Blazor" Version="4.31.5" />
|
||||
<PackageReference Include="Serilog" Version="3.1.1" />
|
||||
<PackageReference Include="Serilog.AspNetCore" Version="8.0.1" />
|
||||
<PackageReference Include="Serilog.Sinks.Console" Version="5.0.1" />
|
||||
<PackageReference Include="Serilog.Sinks.File" Version="5.0.0" />
|
||||
<PackageReference Include="Serilog.Sinks.Seq" Version="7.0.1" />
|
||||
<PackageReference Include="Serilog.Sinks.SQLite" Version="6.0.0" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<None Update="LogsFolder\readme.txt">
|
||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||
</None>
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
||||
@@ -0,0 +1,26 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<base href="/" />
|
||||
<link rel="stylesheet" href="bootstrap/bootstrap.min.css" />
|
||||
|
||||
<link rel="stylesheet" href="_content/Radzen.Blazor/css/standard-base.css">
|
||||
|
||||
@* <link rel="stylesheet" href="app.css" /> *@
|
||||
@* <link rel="stylesheet" href="BlazorAppRadzenNet8SerilogLogging.styles.css" /> *@
|
||||
|
||||
<link rel="icon" type="image/png" href="favicon.png" />
|
||||
<HeadOutlet @rendermode="InteractiveServer" />
|
||||
</head>
|
||||
|
||||
<body>
|
||||
<Routes @rendermode="InteractiveServer" />
|
||||
|
||||
<script src="_content/Radzen.Blazor/Radzen.Blazor.js"></script>
|
||||
<script src="_framework/blazor.web.js"></script>
|
||||
</body>
|
||||
|
||||
</html>
|
||||
@@ -0,0 +1,65 @@
|
||||
@inherits LayoutComponentBase
|
||||
|
||||
@inject IJSRuntime JSRuntime
|
||||
@inject NavigationManager NavigationManager
|
||||
@inject DialogService DialogService
|
||||
@inject ContextMenuService ContextMenuService
|
||||
@inject TooltipService TooltipService
|
||||
@inject NotificationService NotificationService
|
||||
|
||||
<RadzenDialog />
|
||||
<RadzenNotification />
|
||||
<RadzenTooltip />
|
||||
<RadzenContextMenu />
|
||||
|
||||
|
||||
@* <RadzenComponents /> *@
|
||||
|
||||
|
||||
|
||||
<RadzenLayout style="grid-template-areas: 'rz-sidebar rz-header' 'rz-sidebar rz-body';">
|
||||
<RadzenHeader>
|
||||
<RadzenRow JustifyContent="JustifyContent.Start" AlignItems="AlignItems.Center" Gap="0">
|
||||
<RadzenColumn Size="5">
|
||||
<RadzenSidebarToggle Click="@SidebarToggleClick"></RadzenSidebarToggle>
|
||||
</RadzenColumn>
|
||||
<RadzenColumn Size="7">
|
||||
<RadzenStack AlignItems="AlignItems.Center" Orientation="Orientation.Horizontal" JustifyContent="JustifyContent.End"></RadzenStack>
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
</RadzenHeader>
|
||||
<RadzenBody Expanded="@sidebarExpanded">
|
||||
<RadzenRow class="rz-mx-auto rz-px-4 rz-pt-2 rz-pt-md-4 rz-pt-lg-6 rz-pt-xl-12 rz-pb-2 rz-pb-lg-12" Style="max-width: 1440px;">
|
||||
<RadzenColumn Size="12">
|
||||
@Body
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
</RadzenBody>
|
||||
<RadzenSidebar Expanded="@sidebarExpanded" style="z-index: 2">
|
||||
<RadzenStack AlignItems="Radzen.AlignItems.Center" class="rz-py-4 rz-py-lg-6" Style="padding: var(--rz-panel-menu-item-padding); border-bottom: var(--rz-panel-menu-item-border);">
|
||||
|
||||
@* <RadzenImage Path="images/logo.png" style="width: 48px; height: 48px;"></RadzenImage>
|
||||
*@
|
||||
<RadzenText Text="appname" TextStyle="Radzen.Blazor.TextStyle.Subtitle1" class="rz-mb-0" style="color: var(--rz-sidebar-color);" />
|
||||
</RadzenStack>
|
||||
|
||||
<NavMenu />
|
||||
|
||||
<RadzenStack AlignItems="Radzen.AlignItems.Center" Gap="0" class="rz-py-4 rz-py-lg-6" Style="padding: var(--rz-panel-menu-item-padding);">
|
||||
<RadzenText Text="appname v1.0.0" TextStyle="Radzen.Blazor.TextStyle.Caption" style="color: var(--rz-text-disabled-color);" TagName="Radzen.Blazor.TagName.P" TextAlign="Radzen.TextAlign.Center" />
|
||||
<RadzenText Text="Copyright Ⓒ 2024" TextStyle="Radzen.Blazor.TextStyle.Caption" class="rz-mb-0" style="color: var(--rz-text-disabled-color);" TagName="Radzen.Blazor.TagName.P" TextAlign="Radzen.TextAlign.Center" />
|
||||
</RadzenStack>
|
||||
|
||||
</RadzenSidebar>
|
||||
</RadzenLayout>
|
||||
|
||||
|
||||
@code {
|
||||
bool sidebarExpanded = true;
|
||||
|
||||
void SidebarToggleClick()
|
||||
{
|
||||
sidebarExpanded = !sidebarExpanded;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,8 @@
|
||||
|
||||
<RadzenPanelMenu>
|
||||
|
||||
<RadzenPanelMenuItem Text="Home" Path="/" />
|
||||
<RadzenPanelMenuItem Text="Posts" Path="/BlogPost" />
|
||||
<RadzenPanelMenuItem Text="Logs" Path="/Log" />
|
||||
|
||||
</RadzenPanelMenu>
|
||||
@@ -0,0 +1,77 @@
|
||||
@page "/BlogPost/Create"
|
||||
|
||||
<RadzenText Text="Create" TextStyle="TextStyle.H5" />
|
||||
|
||||
@if (blogPostViewModel == null)
|
||||
{
|
||||
<p>Loading...</p>
|
||||
}
|
||||
else
|
||||
{
|
||||
|
||||
<RadzenStack>
|
||||
<RadzenFieldset Text="New Post">
|
||||
<RadzenStack Gap="2rem">
|
||||
<EditForm Context="editFormNewPost" Model="@blogPostViewModel" OnValidSubmit="HandleValidSubmit">
|
||||
<DataAnnotationsValidator />
|
||||
<ValidationSummary />
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Title" for="title" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<InputText id="title" class="form-control" placeholder="Title"
|
||||
@bind-Value="blogPostViewModel.Title" />
|
||||
<ValidationMessage For="@(() => blogPostViewModel.Title)" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Content" for="content" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<InputText id="content" class="form-control" placeholder="Content"
|
||||
@bind-Value="blogPostViewModel.Content" />
|
||||
<ValidationMessage For="@(() => blogPostViewModel.Content)" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenButton Text="Save" Icon="save"
|
||||
ButtonType="ButtonType.Submit" ButtonStyle="ButtonStyle.Success" />
|
||||
|
||||
</EditForm>
|
||||
|
||||
</RadzenStack>
|
||||
</RadzenFieldset>
|
||||
</RadzenStack>
|
||||
|
||||
<RadzenButton Text="Back" Icon="arrow_back" Class="rz-mt-2"
|
||||
Click="NavigatetoBlogPostIndex"
|
||||
ButtonStyle="ButtonStyle.Primary" />
|
||||
|
||||
}
|
||||
|
||||
@code {
|
||||
private BlogPostViewModel? blogPostViewModel;
|
||||
|
||||
protected override void OnInitialized()
|
||||
{
|
||||
blogPostViewModel = new();
|
||||
}
|
||||
|
||||
protected async Task HandleValidSubmit()
|
||||
{
|
||||
if (blogPostViewModel == null) return;
|
||||
|
||||
var blogPost = Mapper.Map<BlogPostViewModel, BlogPost>(blogPostViewModel);
|
||||
bool result = await BlogPostService.AddBlogPostAsync(blogPost);
|
||||
if (result)
|
||||
NavigationManager.NavigateTo("/BlogPost/");
|
||||
|
||||
}
|
||||
|
||||
private void NavigatetoBlogPostIndex() => NavigationManager.NavigateTo("/BlogPost");
|
||||
|
||||
}
|
||||
@@ -0,0 +1,79 @@
|
||||
@page "/BlogPost/Delete/{id:int}"
|
||||
|
||||
<h3>Delete</h3>
|
||||
|
||||
@if (blogPostViewModel == null)
|
||||
{
|
||||
<p>Loading...</p>
|
||||
}
|
||||
else
|
||||
{
|
||||
|
||||
<RadzenStack>
|
||||
<RadzenFieldset Text="Post Delete">
|
||||
<RadzenStack>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Id" for="id" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<RadzenTextBox id="id" class="form-control" placeholder="Id" ReadOnly=true
|
||||
Value="@blogPostViewModel.Id.ToString()" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Title" for="title" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<RadzenTextBox id="title" class="form-control" placeholder="Title" ReadOnly=true
|
||||
Value="@blogPostViewModel.Title" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
</RadzenStack>
|
||||
|
||||
<RadzenButton Text="REMOVE" Icon="delete_forever" ButtonStyle="ButtonStyle.Danger"
|
||||
Click="RemoveButtonClick" />
|
||||
|
||||
</RadzenFieldset>
|
||||
</RadzenStack>
|
||||
|
||||
<RadzenButton Text="Back" Icon="arrow_back" Class="rz-mt-2"
|
||||
Click="NavigatetoBlogPostIndex"
|
||||
ButtonStyle="ButtonStyle.Primary" />
|
||||
|
||||
|
||||
}
|
||||
|
||||
@code {
|
||||
|
||||
[Parameter]
|
||||
public int id { get; set; }
|
||||
|
||||
BlogPostViewModel? blogPostViewModel;
|
||||
|
||||
protected override async Task OnInitializedAsync()
|
||||
{
|
||||
if (blogPostViewModel == null)
|
||||
{
|
||||
var blogPost = await BlogPostService.GetBlogPostByIdAsync(id);
|
||||
if (blogPost == null)
|
||||
return;
|
||||
|
||||
blogPostViewModel = Mapper.Map<BlogPost, BlogPostViewModel>(blogPost);
|
||||
}
|
||||
}
|
||||
|
||||
private async void RemoveButtonClick()
|
||||
{
|
||||
bool result = await BlogPostService.DeleteBlogPostByIdAsync(id);
|
||||
if (result)
|
||||
NavigationManager.NavigateTo("/BlogPost");
|
||||
}
|
||||
|
||||
private void NavigatetoBlogPostIndex() => NavigationManager.NavigateTo("/BlogPost");
|
||||
|
||||
}
|
||||
@@ -0,0 +1,78 @@
|
||||
@page "/BlogPost/Detail/{id:int}"
|
||||
|
||||
<h3>Detail</h3>
|
||||
|
||||
@if (blogPostViewModel == null)
|
||||
{
|
||||
<p>Loading...</p>
|
||||
}
|
||||
else
|
||||
{
|
||||
|
||||
<RadzenStack>
|
||||
<RadzenFieldset Text="Post Detail">
|
||||
<RadzenStack>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Id" for="id" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<RadzenTextBox id="id" class="form-control" placeholder="Id" ReadOnly=true
|
||||
Value="@blogPostViewModel.Id.ToString()" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Title" for="title" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<RadzenTextBox id="title" class="form-control" placeholder="Title" ReadOnly=true
|
||||
Value="@blogPostViewModel.Title" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Content" for="content" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<RadzenTextBox id="content" class="form-control" placeholder="Content" ReadOnly=true
|
||||
Value="@blogPostViewModel.Content" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
</RadzenStack>
|
||||
</RadzenFieldset>
|
||||
</RadzenStack>
|
||||
|
||||
<RadzenButton Text="Back" Icon="arrow_back" Class="rz-mt-2"
|
||||
Click="NavigatetoBlogPostIndex"
|
||||
ButtonStyle="ButtonStyle.Primary" />
|
||||
|
||||
}
|
||||
|
||||
@code {
|
||||
|
||||
[Parameter]
|
||||
public int id { get; set; }
|
||||
|
||||
BlogPostViewModel? blogPostViewModel;
|
||||
|
||||
protected override async Task OnInitializedAsync()
|
||||
{
|
||||
if (blogPostViewModel == null)
|
||||
{
|
||||
var blogPost = await BlogPostService.GetBlogPostByIdAsync(id);
|
||||
if (blogPost == null)
|
||||
return;
|
||||
|
||||
blogPostViewModel = Mapper.Map<BlogPost, BlogPostViewModel>(blogPost);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private void NavigatetoBlogPostIndex() => NavigationManager.NavigateTo("/BlogPost");
|
||||
|
||||
}
|
||||
@@ -0,0 +1,105 @@
|
||||
@page "/BlogPost/Edit/{id:int}"
|
||||
|
||||
<h3>Edit</h3>
|
||||
|
||||
@if (blogPostViewModel == null)
|
||||
{
|
||||
<p>Loading...</p>
|
||||
}
|
||||
else
|
||||
{
|
||||
|
||||
<RadzenStack>
|
||||
<RadzenFieldset Text="Edit Post">
|
||||
<RadzenStack Gap="2rem">
|
||||
<EditForm Context="editFormEdit" Model="@blogPostViewModel" OnValidSubmit="HandleValidSubmit">
|
||||
<DataAnnotationsValidator />
|
||||
<ValidationSummary />
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Title" for="title" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<InputText id="title" class="form-control" placeholder="Title"
|
||||
@bind-Value="blogPostViewModel.Title" />
|
||||
<ValidationMessage For="@(() => blogPostViewModel.Title)" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Content" for="content" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<InputText id="content" class="form-control" placeholder="Content"
|
||||
@bind-Value="blogPostViewModel.Content" />
|
||||
<ValidationMessage For="@(() => blogPostViewModel.Content)" />
|
||||
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenButton Text="Save" Icon="save"
|
||||
ButtonType="ButtonType.Submit" ButtonStyle="ButtonStyle.Success" />
|
||||
|
||||
</EditForm>
|
||||
|
||||
</RadzenStack>
|
||||
</RadzenFieldset>
|
||||
</RadzenStack>
|
||||
|
||||
<RadzenButton Text="Back" Icon="arrow_back" Class="rz-mt-2"
|
||||
Click="NavigatetoBlogPostIndex"
|
||||
ButtonStyle="ButtonStyle.Primary" />
|
||||
|
||||
}
|
||||
|
||||
@code {
|
||||
|
||||
[Parameter]
|
||||
public int id { get; set; }
|
||||
|
||||
BlogPostViewModel? blogPostViewModel;
|
||||
|
||||
protected override async Task OnInitializedAsync()
|
||||
{
|
||||
if (blogPostViewModel == null)
|
||||
{
|
||||
var blogPost = await BlogPostService.GetBlogPostByIdAsync(id);
|
||||
if (blogPost == null)
|
||||
return;
|
||||
|
||||
blogPostViewModel = Mapper.Map<BlogPost, BlogPostViewModel>(blogPost);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void OnPaste(HtmlEditorPasteEventArgs args)
|
||||
{
|
||||
}
|
||||
|
||||
void OnChange(string html)
|
||||
{
|
||||
}
|
||||
|
||||
void OnInput(string html)
|
||||
{
|
||||
}
|
||||
|
||||
void OnExecute(HtmlEditorExecuteEventArgs args)
|
||||
{
|
||||
}
|
||||
|
||||
protected async Task HandleValidSubmit()
|
||||
{
|
||||
if (blogPostViewModel == null) return;
|
||||
|
||||
var blogPost = Mapper.Map<BlogPostViewModel, BlogPost>(blogPostViewModel);
|
||||
bool result = await BlogPostService.UpdateBlogPostAsync(id, blogPost);
|
||||
if (result)
|
||||
NavigationManager.NavigateTo("/BlogPost/");
|
||||
}
|
||||
|
||||
private void NavigatetoBlogPostIndex() => NavigationManager.NavigateTo("/BlogPost");
|
||||
|
||||
}
|
||||
@@ -0,0 +1,81 @@
|
||||
@page "/BlogPost"
|
||||
|
||||
<PageTitle>Posts</PageTitle>
|
||||
|
||||
<RadzenRow>
|
||||
|
||||
<RadzenColumn SizeSM="12" SizeMD="12" SizeLG="4">
|
||||
|
||||
<RadzenStack Orientation="Orientation.Horizontal" AlignItems="AlignItems.Center">
|
||||
<RadzenText Text="Posts" TextStyle="TextStyle.H5" />
|
||||
<RadzenButton Text="Create" Icon="add_circle_outline"
|
||||
Click="NavigatetoCreate"
|
||||
ButtonStyle="ButtonStyle.Success" class="rz-mb-2 rz-p-2" />
|
||||
</RadzenStack>
|
||||
|
||||
</RadzenColumn>
|
||||
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenDataGrid KeyProperty="Id" IsLoading="@isLoading" ShowPagingSummary=true
|
||||
Count="@totalCount" Data="@blogPosts" LoadData="@LoadData"
|
||||
FilterPopupRenderMode="PopupRenderMode.OnDemand"
|
||||
FilterCaseSensitivity="FilterCaseSensitivity.CaseInsensitive"
|
||||
FilterMode="FilterMode.Advanced" AllowSorting="true" AllowFiltering="true"
|
||||
AllowPaging="true" PageSize="@itemPageSize" PagerHorizontalAlign="HorizontalAlign.Center"
|
||||
TItem="BlogPostViewModel" ColumnWidth="200px">
|
||||
<Columns>
|
||||
<RadzenDataGridColumn TItem="BlogPostViewModel" Property="Id" Filterable="false" Title="Id" Frozen="true" Width="30px" MinWidth="30px" TextAlign="TextAlign.Center" />
|
||||
|
||||
<RadzenDataGridColumn TItem="BlogPostViewModel" Property="Title" Title="Title" />
|
||||
<RadzenDataGridColumn TItem="BlogPostViewModel" Property="Content" Title="Content" />
|
||||
|
||||
<RadzenDataGridColumn TItem="BlogPostViewModel" Context="blogPost" Filterable="false" Sortable="false" Width="150px" TextAlign="TextAlign.Center">
|
||||
<Template Context="blogPost">
|
||||
|
||||
<RadzenRow JustifyContent="JustifyContent.Center">
|
||||
<RadzenButton Icon="pageview" ButtonStyle="ButtonStyle.Info" Variant="Variant.Flat" Size="ButtonSize.Medium"
|
||||
Click="@(args => NavigatetoDetail(blogPost.Id))" @onclick:stopPropagation="true">
|
||||
</RadzenButton>
|
||||
<RadzenButton Icon="edit" ButtonStyle="ButtonStyle.Warning" Variant="Variant.Flat" Size="ButtonSize.Medium"
|
||||
Click="@(args => NavigatetoEdit(blogPost.Id))" @onclick:stopPropagation="true">
|
||||
</RadzenButton>
|
||||
<RadzenButton Icon="delete_forever" ButtonStyle="ButtonStyle.Danger" Variant="Variant.Flat" Size="ButtonSize.Medium"
|
||||
Click="@(args => NavigatetoDelete(blogPost.Id))" @onclick:stopPropagation="true">
|
||||
</RadzenButton>
|
||||
</RadzenRow>
|
||||
|
||||
</Template>
|
||||
</RadzenDataGridColumn>
|
||||
</Columns>
|
||||
</RadzenDataGrid>
|
||||
|
||||
|
||||
@code {
|
||||
|
||||
const int itemPageSize = 10;
|
||||
private bool isLoading;
|
||||
private int totalCount;
|
||||
private IEnumerable<BlogPostViewModel>? blogPosts;
|
||||
|
||||
private async Task LoadData(LoadDataArgs args)
|
||||
{
|
||||
isLoading = true;
|
||||
|
||||
var result = await BlogPostService.GetBlogPostsAsync(filter: args.Filter, top: args.Top, skip: args.Skip, orderby: args.OrderBy, count: true);
|
||||
|
||||
blogPosts = Mapper.Map<IEnumerable<BlogPost>, IEnumerable<BlogPostViewModel>>(result.Result);
|
||||
totalCount = result.TotalCount;
|
||||
|
||||
isLoading = false;
|
||||
}
|
||||
|
||||
private void NavigatetoCreate() => NavigationManager.NavigateTo("/BlogPost/Create");
|
||||
private void NavigatetoDetail(int id) => NavigationManager.NavigateTo($"/BlogPost/Detail/{id}");
|
||||
private void NavigatetoEdit(int id) => NavigationManager.NavigateTo($"/BlogPost/Edit/{id}");
|
||||
private void NavigatetoDelete(int id) => NavigationManager.NavigateTo($"/BlogPost/Delete/{id}");
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
@@ -0,0 +1,7 @@
|
||||
@using BlazorAppRadzenNet8SerilogLogging.Data;
|
||||
@using BlazorAppRadzenNet8SerilogLogging.Models;
|
||||
@using BlazorAppRadzenNet8SerilogLogging.Services;
|
||||
@using BlazorAppRadzenNet8SerilogLogging.ViewModels;
|
||||
|
||||
@inject NavigationManager NavigationManager
|
||||
@inject BlogPostService BlogPostService
|
||||
@@ -0,0 +1,36 @@
|
||||
@page "/Error"
|
||||
@using System.Diagnostics
|
||||
|
||||
<PageTitle>Error</PageTitle>
|
||||
|
||||
<h1 class="text-danger">Error.</h1>
|
||||
<h2 class="text-danger">An error occurred while processing your request.</h2>
|
||||
|
||||
@if (ShowRequestId)
|
||||
{
|
||||
<p>
|
||||
<strong>Request ID:</strong> <code>@RequestId</code>
|
||||
</p>
|
||||
}
|
||||
|
||||
<h3>Development Mode</h3>
|
||||
<p>
|
||||
Swapping to <strong>Development</strong> environment will display more detailed information about the error that occurred.
|
||||
</p>
|
||||
<p>
|
||||
<strong>The Development environment shouldn't be enabled for deployed applications.</strong>
|
||||
It can result in displaying sensitive information from exceptions to end users.
|
||||
For local debugging, enable the <strong>Development</strong> environment by setting the <strong>ASPNETCORE_ENVIRONMENT</strong> environment variable to <strong>Development</strong>
|
||||
and restarting the app.
|
||||
</p>
|
||||
|
||||
@code{
|
||||
[CascadingParameter]
|
||||
private HttpContext? HttpContext { get; set; }
|
||||
|
||||
private string? RequestId { get; set; }
|
||||
private bool ShowRequestId => !string.IsNullOrEmpty(RequestId);
|
||||
|
||||
protected override void OnInitialized() =>
|
||||
RequestId = Activity.Current?.Id ?? HttpContext?.TraceIdentifier;
|
||||
}
|
||||
@@ -0,0 +1,7 @@
|
||||
@page "/"
|
||||
|
||||
<PageTitle>Home</PageTitle>
|
||||
|
||||
<h1>Hello, world!</h1>
|
||||
|
||||
Welcome to your new app.
|
||||
@@ -0,0 +1,69 @@
|
||||
@page "/Log/Delete/{id:int}"
|
||||
|
||||
<h3>Delete</h3>
|
||||
|
||||
@if (logViewModel == null)
|
||||
{
|
||||
<p>Loading...</p>
|
||||
}
|
||||
else
|
||||
{
|
||||
|
||||
<RadzenStack>
|
||||
<RadzenFieldset Text="Log Delete">
|
||||
<RadzenStack>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Id" for="id" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<RadzenTextBox id="id" class="form-control" placeholder="Id" ReadOnly=true
|
||||
Value="@logViewModel.id.ToString()" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
</RadzenStack>
|
||||
|
||||
<RadzenButton Text="REMOVE" Icon="delete_forever" ButtonStyle="ButtonStyle.Danger"
|
||||
Click="RemoveButtonClick" />
|
||||
|
||||
</RadzenFieldset>
|
||||
</RadzenStack>
|
||||
|
||||
<RadzenButton Text="Back" Icon="arrow_back" Class="rz-mt-2"
|
||||
Click="NavigatetoLogIndex"
|
||||
ButtonStyle="ButtonStyle.Primary" />
|
||||
|
||||
|
||||
}
|
||||
|
||||
@code {
|
||||
|
||||
[Parameter]
|
||||
public int id { get; set; }
|
||||
|
||||
LogViewModel? logViewModel;
|
||||
|
||||
protected override async Task OnInitializedAsync()
|
||||
{
|
||||
if (logViewModel == null)
|
||||
{
|
||||
var log = await LoggerService.GetLogByIdAsync(id);
|
||||
if (log == null)
|
||||
return;
|
||||
|
||||
logViewModel = Mapper.Map<Log, LogViewModel>(log);
|
||||
}
|
||||
}
|
||||
|
||||
private async void RemoveButtonClick()
|
||||
{
|
||||
bool result = await LoggerService.DeleteLogByIdAsync(id);
|
||||
if (result)
|
||||
NavigationManager.NavigateTo("/Log");
|
||||
}
|
||||
|
||||
private void NavigatetoLogIndex() => NavigationManager.NavigateTo("/Log");
|
||||
|
||||
}
|
||||
@@ -0,0 +1,108 @@
|
||||
@page "/Log/Detail/{id:int}"
|
||||
|
||||
<h3>Detail</h3>
|
||||
|
||||
@if (logViewModel == null)
|
||||
{
|
||||
<p>Loading...</p>
|
||||
}
|
||||
else
|
||||
{
|
||||
|
||||
<RadzenStack>
|
||||
<RadzenFieldset Text="Log Detail">
|
||||
<RadzenStack>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Id" for="id" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<RadzenTextBox id="id" class="form-control" placeholder="Id" ReadOnly=true
|
||||
Value="@logViewModel.id.ToString()" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Timestamp" for="timestamp" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<RadzenTextBox id="timestamp" class="form-control" placeholder="Timestamp" ReadOnly=true
|
||||
Value="@logViewModel.Timestamp.ToString("yyyy-MM-dd HH:mm:ss")" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Level" for="level" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<RadzenTextBox id="level" class="form-control" placeholder="Level" ReadOnly=true
|
||||
Value="@logViewModel.Level" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Exception" for="exception" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<RadzenTextArea ReadOnly="true" Rows="5" id="exception" class="form-control" placeholder="Exception"
|
||||
Value="@logViewModel.Exception" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="RenderedMessage" for="renderedMessage" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<RadzenTextArea ReadOnly="true" Rows="5" id="renderedMessage" class="form-control" placeholder="RenderedMessage"
|
||||
Value="@logViewModel.RenderedMessage" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenRow AlignItems="AlignItems.Center" Class="rz-mb-2">
|
||||
<RadzenColumn SizeSM="12" SizeMD="2" SizeLG="2">
|
||||
<RadzenLabel Text="Properties" for="properties" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn SizeSM="12" SizeMD="10" SizeLG="10">
|
||||
<RadzenTextArea ReadOnly="true" Rows="5" id="properties" class="form-control" placeholder="Properties"
|
||||
Value="@logViewModel.Properties" />
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
|
||||
</RadzenStack>
|
||||
</RadzenFieldset>
|
||||
</RadzenStack>
|
||||
|
||||
<RadzenButton Text="Back" Icon="arrow_back" Class="rz-mt-2"
|
||||
Click="NavigatetoLogIndex"
|
||||
ButtonStyle="ButtonStyle.Primary" />
|
||||
|
||||
}
|
||||
|
||||
@code {
|
||||
|
||||
[Parameter]
|
||||
public int id { get; set; }
|
||||
|
||||
LogViewModel? logViewModel;
|
||||
|
||||
protected override async Task OnInitializedAsync()
|
||||
{
|
||||
if (logViewModel == null)
|
||||
{
|
||||
var log = await LoggerService.GetLogByIdAsync(id);
|
||||
if (log == null)
|
||||
return;
|
||||
|
||||
logViewModel = Mapper.Map<Log, LogViewModel>(log);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private void NavigatetoLogIndex() => NavigationManager.NavigateTo("/Log");
|
||||
|
||||
}
|
||||
@@ -0,0 +1,102 @@
|
||||
@page "/Log"
|
||||
|
||||
@inject DialogService DialogService
|
||||
|
||||
<PageTitle>Logs</PageTitle>
|
||||
|
||||
<RadzenRow>
|
||||
|
||||
<RadzenColumn SizeSM="12" SizeMD="12" SizeLG="4">
|
||||
|
||||
<RadzenStack Orientation="Orientation.Horizontal" AlignItems="AlignItems.Center">
|
||||
<RadzenText Text="Logs" TextStyle="TextStyle.H5" />
|
||||
<RadzenButton Text="DELETE ALL LOGS" Icon="delete_forever"
|
||||
Click="DeleteAllLogs"
|
||||
ButtonStyle="ButtonStyle.Danger" class="rz-mb-2 rz-p-2" />
|
||||
</RadzenStack>
|
||||
|
||||
</RadzenColumn>
|
||||
|
||||
</RadzenRow>
|
||||
|
||||
<RadzenDataGrid KeyProperty="id" IsLoading="@isLoading" ShowPagingSummary=true
|
||||
Count="@totalCount" Data="@logs" LoadData="@LoadData"
|
||||
FilterPopupRenderMode="PopupRenderMode.OnDemand"
|
||||
FilterCaseSensitivity="FilterCaseSensitivity.CaseInsensitive"
|
||||
FilterMode="FilterMode.Advanced" AllowSorting="true" AllowFiltering="true"
|
||||
AllowPaging="true" PageSize="@itemPageSize" PagerHorizontalAlign="HorizontalAlign.Center"
|
||||
TItem="LogViewModel" ColumnWidth="200px">
|
||||
<Columns>
|
||||
<RadzenDataGridColumn TItem="LogViewModel" Property="id" Filterable="false" Title="Id" Frozen="true" Width="30px" MinWidth="30px" TextAlign="TextAlign.Center" />
|
||||
|
||||
<RadzenDataGridColumn TItem="LogViewModel" Property="Timestamp" Title="Timestamp" />
|
||||
<RadzenDataGridColumn TItem="LogViewModel" Property="Level" Title="Level" Context="log">
|
||||
<Template>
|
||||
<span class="text-@Helpers.LogEventLevelHelper.GetBootstrapUIClass(log.Level)">
|
||||
@log.Level
|
||||
</span>
|
||||
</Template>
|
||||
</RadzenDataGridColumn>
|
||||
|
||||
<RadzenDataGridColumn TItem="LogViewModel" Property="Exception" Title="Exception" />
|
||||
<RadzenDataGridColumn TItem="LogViewModel" Property="RenderedMessage" Title="RenderedMessage" />
|
||||
<RadzenDataGridColumn TItem="LogViewModel" Property="Properties" Title="Properties" />
|
||||
|
||||
<RadzenDataGridColumn TItem="LogViewModel" Context="log" Filterable="false" Sortable="false" Width="150px" TextAlign="TextAlign.Center">
|
||||
<Template Context="log">
|
||||
|
||||
<RadzenRow JustifyContent="JustifyContent.Center">
|
||||
<RadzenButton Icon="pageview" ButtonStyle="ButtonStyle.Info" Variant="Variant.Flat" Size="ButtonSize.Medium"
|
||||
Click="@(args => NavigatetoDetail(log.id))" @onclick:stopPropagation="true">
|
||||
</RadzenButton>
|
||||
<RadzenButton Icon="delete_forever" ButtonStyle="ButtonStyle.Danger" Variant="Variant.Flat" Size="ButtonSize.Medium"
|
||||
Click="@(args => NavigatetoDelete(log.id))" @onclick:stopPropagation="true">
|
||||
</RadzenButton>
|
||||
</RadzenRow>
|
||||
|
||||
</Template>
|
||||
</RadzenDataGridColumn>
|
||||
</Columns>
|
||||
</RadzenDataGrid>
|
||||
|
||||
|
||||
@code {
|
||||
|
||||
const int itemPageSize = 10;
|
||||
private bool isLoading;
|
||||
private int totalCount;
|
||||
private IEnumerable<LogViewModel>? logs;
|
||||
|
||||
private async Task LoadData(LoadDataArgs args)
|
||||
{
|
||||
isLoading = true;
|
||||
|
||||
var result = await LoggerService.GetLogsAsync(filter: args.Filter, top: args.Top, skip: args.Skip, orderby: args.OrderBy, count: true);
|
||||
|
||||
logs = Mapper.Map<IEnumerable<Log>, IEnumerable<LogViewModel>>(result.Result);
|
||||
totalCount = result.TotalCount;
|
||||
|
||||
isLoading = false;
|
||||
}
|
||||
|
||||
private async Task DeleteAllLogs()
|
||||
{
|
||||
var dialogResult = await DialogService.Confirm("Are you sure DELETE All Logs?", "Delete All Logs",
|
||||
new ConfirmOptions { OkButtonText = "Ok", CancelButtonText = "Cancel" });
|
||||
|
||||
if (dialogResult == true)
|
||||
{
|
||||
var deleteAllLogsResult = await LoggerService.DeleteAllLogsAsync();
|
||||
if (deleteAllLogsResult == true)
|
||||
NavigationManager.NavigateTo("/Log", true);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
private void NavigatetoDetail(int id) => NavigationManager.NavigateTo($"/Log/Detail/{id}");
|
||||
private void NavigatetoDelete(int id) => NavigationManager.NavigateTo($"/Log/Delete/{id}");
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
@@ -0,0 +1,7 @@
|
||||
@using BlazorAppRadzenNet8SerilogLogging.Data;
|
||||
@using BlazorAppRadzenNet8SerilogLogging.Models;
|
||||
@using BlazorAppRadzenNet8SerilogLogging.Services;
|
||||
@using BlazorAppRadzenNet8SerilogLogging.ViewModels;
|
||||
|
||||
@inject NavigationManager NavigationManager
|
||||
@inject LoggerService LoggerService
|
||||
@@ -0,0 +1,6 @@
|
||||
<Router AppAssembly="typeof(Program).Assembly">
|
||||
<Found Context="routeData">
|
||||
<RouteView RouteData="routeData" DefaultLayout="typeof(Layout.MainLayout)" />
|
||||
@* <FocusOnNavigate RouteData="routeData" Selector="h1" /> *@
|
||||
</Found>
|
||||
</Router>
|
||||
@@ -0,0 +1,16 @@
|
||||
@using System.Net.Http
|
||||
@using System.Net.Http.Json
|
||||
@using Microsoft.AspNetCore.Components.Forms
|
||||
@using Microsoft.AspNetCore.Components.Routing
|
||||
@using Microsoft.AspNetCore.Components.Web
|
||||
@using static Microsoft.AspNetCore.Components.Web.RenderMode
|
||||
@using Microsoft.AspNetCore.Components.Web.Virtualization
|
||||
@using Microsoft.JSInterop
|
||||
@using BlazorAppRadzenNet8SerilogLogging
|
||||
@using BlazorAppRadzenNet8SerilogLogging.Components
|
||||
|
||||
@using MapsterMapper
|
||||
@using Radzen
|
||||
@using Radzen.Blazor
|
||||
|
||||
@inject IMapper Mapper
|
||||
@@ -0,0 +1,19 @@
|
||||
using BlazorAppRadzenNet8SerilogLogging.Models;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
|
||||
namespace BlazorAppRadzenNet8SerilogLogging.Data;
|
||||
|
||||
public class ApplicationDbContext : DbContext
|
||||
{
|
||||
public ApplicationDbContext(DbContextOptions<ApplicationDbContext> options)
|
||||
: base(options)
|
||||
{
|
||||
}
|
||||
|
||||
public DbSet<BlogPost> BlogPosts => Set<BlogPost>();
|
||||
|
||||
protected override void OnModelCreating(ModelBuilder modelBuilder)
|
||||
{
|
||||
base.OnModelCreating(modelBuilder);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,19 @@
|
||||
using BlazorAppRadzenNet8SerilogLogging.Models;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
|
||||
namespace BlazorAppRadzenNet8SerilogLogging.Data;
|
||||
|
||||
public class ApplicationLoggerDbContext : DbContext
|
||||
{
|
||||
public ApplicationLoggerDbContext(DbContextOptions<ApplicationLoggerDbContext> options)
|
||||
: base(options)
|
||||
{
|
||||
}
|
||||
|
||||
public DbSet<Log> Logs => Set<Log>();
|
||||
|
||||
protected override void OnModelCreating(ModelBuilder modelBuilder)
|
||||
{
|
||||
base.OnModelCreating(modelBuilder);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,101 @@
|
||||
using BlazorAppRadzenNet8SerilogLogging.Models;
|
||||
|
||||
namespace BlazorAppRadzenNet8SerilogLogging.Data;
|
||||
|
||||
public class SeedData
|
||||
{
|
||||
private readonly ApplicationDbContext _context;
|
||||
|
||||
public SeedData(ApplicationDbContext context)
|
||||
{
|
||||
_context = context;
|
||||
}
|
||||
|
||||
public async Task CreateInitialData()
|
||||
{
|
||||
var posts = GetAllBlogPosts();
|
||||
await _context.BlogPosts.AddRangeAsync(posts);
|
||||
await _context.SaveChangesAsync();
|
||||
}
|
||||
|
||||
private static IEnumerable<BlogPost> GetAllBlogPosts()
|
||||
{
|
||||
List<BlogPost> posts = new();
|
||||
for (int i = 0; i < 50; i++)
|
||||
{
|
||||
BlogPost post = new() { Id = i + 1, Title = titles[i], Content = contents[i % 10] };
|
||||
posts.Add(post);
|
||||
}
|
||||
|
||||
return posts;
|
||||
}
|
||||
|
||||
private static readonly string[] titles = {
|
||||
"Introduction to Object-Oriented Programming",
|
||||
"Mastering Data Structures and Algorithms",
|
||||
"Building Web Applications with ASP.NET",
|
||||
"Creating Mobile Apps with Xamarin",
|
||||
"Exploring Artificial Intelligence and Machine Learning",
|
||||
"Understanding Functional Programming Concepts",
|
||||
"Developing Games with Unity",
|
||||
"Securing Web Applications from Cyber Attacks",
|
||||
"Optimizing Code Performance for Better Efficiency",
|
||||
"Implementing Design Patterns in Software Development",
|
||||
"Testing and Debugging Strategies for Reliable Software",
|
||||
"Working with Databases and SQL",
|
||||
"Building Responsive User Interfaces with HTML and CSS",
|
||||
"Exploring Cloud Computing and Serverless Architecture",
|
||||
"Developing Cross-Platform Applications with React Native",
|
||||
"Introduction to Internet of Things (IoT)",
|
||||
"Creating Scalable Microservices with Docker and Kubernetes",
|
||||
"Understanding Network Protocols and TCP/IP",
|
||||
"Building RESTful APIs with Node.js and Express",
|
||||
"Exploring Big Data Analytics and Apache Hadoop",
|
||||
"Mastering Version Control with Git and GitHub",
|
||||
"Developing Desktop Applications with WPF",
|
||||
"Securing Mobile Applications from Malicious Attacks",
|
||||
"Optimizing Database Performance with Indexing",
|
||||
"Implementing Continuous Integration and Deployment",
|
||||
"Testing Mobile Apps on Different Platforms",
|
||||
"Working with NoSQL Databases like MongoDB",
|
||||
"Building Progressive Web Apps with React",
|
||||
"Exploring Quantum Computing and Quantum Algorithms",
|
||||
"Introduction to Cybersecurity and Ethical Hacking",
|
||||
"Creating Chatbots with Natural Language Processing",
|
||||
"Understanding Software Development Life Cycle",
|
||||
"Developing Augmented Reality (AR) Applications",
|
||||
"Securing Web APIs with OAuth and JWT",
|
||||
"Optimizing Front-End Performance for Better User Experience",
|
||||
"Implementing Machine Learning Models with TensorFlow",
|
||||
"Testing Web Applications for Cross-Browser Compatibility",
|
||||
"Working with Blockchain Technology and Smart Contracts",
|
||||
"Building Real-Time Applications with SignalR",
|
||||
"Exploring Cryptography and Encryption Techniques",
|
||||
"Introduction to Agile Software Development",
|
||||
"Creating Voice User Interfaces with Amazon Alexa",
|
||||
"Understanding Web Accessibility and Inclusive Design",
|
||||
"Developing Natural Language Processing Applications",
|
||||
"Securing Cloud Infrastructure and Services",
|
||||
"Optimizing Backend Performance for Scalability",
|
||||
"Implementing Continuous Monitoring and Alerting",
|
||||
"Testing APIs with Postman and Swagger",
|
||||
"Working with Data Visualization Libraries like D3.js",
|
||||
"Building E-commerce Applications with Shopify",
|
||||
"Exploring Robotic Process Automation (RPA)",
|
||||
"Introduction to DevOps and CI/CD Pipelines"
|
||||
};
|
||||
|
||||
private static readonly string[] contents = new string[]
|
||||
{
|
||||
"Lorem ipsum dolor sit amet, consectetur t.",
|
||||
"Sed ut perspiciatis unde omnis iste natuccusantium doloremque laudantium.",
|
||||
"Nemo enim ipsam voluptatem quia voluptas aut fugit.",
|
||||
"Quis autem vel eum iure reprehenderit quesse quam nihil molestiae consequatur.",
|
||||
"At vero eos et accusamus et iusto odio d.",
|
||||
"Similique sunt in culpa qui officia de.",
|
||||
"Et harum quidem rerum facilis est et expio.",
|
||||
"Nam libero tempore, cum soluta nobis est.",
|
||||
"Omnis voluptas assumenda est, omnis dolo",
|
||||
"Temporibus autem quibusdam et aut offic"
|
||||
};
|
||||
}
|
||||
@@ -0,0 +1,32 @@
|
||||
using Serilog.Events;
|
||||
|
||||
namespace BlazorAppRadzenNet8SerilogLogging.Helpers;
|
||||
|
||||
public static class LogEventLevelHelper
|
||||
{
|
||||
public static string GetBootstrapUIClass(string value)
|
||||
{
|
||||
var level = StringtoEnum(value);
|
||||
return level switch
|
||||
{
|
||||
LogEventLevel.Verbose or LogEventLevel.Debug or LogEventLevel.Information => "info",
|
||||
LogEventLevel.Warning => "warning",
|
||||
LogEventLevel.Error or LogEventLevel.Fatal => "danger",
|
||||
_ => throw new Exception("not valid logeventlevel")
|
||||
};
|
||||
}
|
||||
|
||||
public static LogEventLevel StringtoEnum(string value)
|
||||
{
|
||||
return value switch
|
||||
{
|
||||
"Verbose" => LogEventLevel.Verbose,
|
||||
"Debug" => LogEventLevel.Debug,
|
||||
"Information" => LogEventLevel.Information,
|
||||
"Warning" => LogEventLevel.Warning,
|
||||
"Error" => LogEventLevel.Error,
|
||||
"Fatal" => LogEventLevel.Fatal,
|
||||
_ => throw new Exception("not valid logeventlevel")
|
||||
};
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1 @@
|
||||
logfolder
|
||||
@@ -0,0 +1,8 @@
|
||||
namespace BlazorAppRadzenNet8SerilogLogging.Models;
|
||||
|
||||
public class BlogPost
|
||||
{
|
||||
public int Id { get; set; }
|
||||
public string Title { get; set; } = string.Empty;
|
||||
public string Content { get; set; } = string.Empty;
|
||||
}
|
||||
@@ -0,0 +1,11 @@
|
||||
namespace BlazorAppRadzenNet8SerilogLogging.Models;
|
||||
|
||||
public class Log
|
||||
{
|
||||
public int id { get; set; }
|
||||
public DateTime Timestamp { get; set; }
|
||||
public string Level { get; set; } = string.Empty;
|
||||
public string Exception { get; set; } = string.Empty;
|
||||
public string RenderedMessage { get; set; } = string.Empty;
|
||||
public string Properties { get; set; } = string.Empty;
|
||||
}
|
||||
@@ -0,0 +1,127 @@
|
||||
using BlazorAppRadzenNet8SerilogLogging.Components;
|
||||
using BlazorAppRadzenNet8SerilogLogging.Data;
|
||||
using BlazorAppRadzenNet8SerilogLogging.Services;
|
||||
using BlazorAppRadzenNet8SerilogLogging.ViewModels;
|
||||
using Mapster;
|
||||
using MapsterMapper;
|
||||
using Microsoft.Data.Sqlite;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Radzen;
|
||||
using Serilog;
|
||||
using System.Reflection;
|
||||
|
||||
internal class Program
|
||||
{
|
||||
private static void Main(string[] args)
|
||||
{
|
||||
var builder = WebApplication.CreateBuilder(args);
|
||||
|
||||
var currentDir = Directory.GetCurrentDirectory();
|
||||
|
||||
// get connection string from configuration file (appsettings.json)
|
||||
string? sqliteLoggerConnectionString = builder.Configuration.GetConnectionString("SqliteLogger");
|
||||
SqliteConnectionStringBuilder sqliteLoggerConnectionStringBuilder = new SqliteConnectionStringBuilder(sqliteLoggerConnectionString);
|
||||
sqliteLoggerConnectionStringBuilder.DataSource = Path.Combine(currentDir, sqliteLoggerConnectionStringBuilder.DataSource);
|
||||
string sqliteDbFilePath = sqliteLoggerConnectionStringBuilder.DataSource;
|
||||
|
||||
// file logger path
|
||||
string serilogFileLoggerFilePath = Path.Combine(currentDir, "LogsFolder", "logs.log");
|
||||
|
||||
builder.Host.UseSerilog((ctx, lc) => lc
|
||||
.MinimumLevel.Information()
|
||||
//.WriteTo.Console(new JsonFormatter(), restrictedToMinimumLevel: Serilog.Events.LogEventLevel.Information)
|
||||
.WriteTo.Console(restrictedToMinimumLevel: Serilog.Events.LogEventLevel.Information)
|
||||
.WriteTo.Seq("http://localhost:5001", restrictedToMinimumLevel: Serilog.Events.LogEventLevel.Information)
|
||||
.WriteTo.File(serilogFileLoggerFilePath,
|
||||
restrictedToMinimumLevel: Serilog.Events.LogEventLevel.Verbose,
|
||||
rollingInterval: RollingInterval.Hour,
|
||||
encoding: System.Text.Encoding.UTF8)
|
||||
.WriteTo.SQLite(sqliteDbFilePath,
|
||||
tableName: "Logs",
|
||||
restrictedToMinimumLevel:
|
||||
builder.Environment.IsDevelopment() ? Serilog.Events.LogEventLevel.Information : Serilog.Events.LogEventLevel.Warning,
|
||||
storeTimestampInUtc: false,
|
||||
batchSize:
|
||||
builder.Environment.IsDevelopment() ? (uint)1 : (uint)100,
|
||||
retentionPeriod: new TimeSpan(0, 1, 0, 0, 0),
|
||||
maxDatabaseSize: 10)
|
||||
);
|
||||
|
||||
builder.Services.AddDbContext<ApplicationLoggerDbContext>(options =>
|
||||
options.UseSqlite(sqliteLoggerConnectionStringBuilder.ConnectionString)
|
||||
);
|
||||
|
||||
// Add services to the container.
|
||||
builder.Services.AddDbContext<ApplicationDbContext>(options =>
|
||||
options.UseInMemoryDatabase("ConnectionInMemory")
|
||||
);
|
||||
builder.Services.AddScoped<SeedData>();
|
||||
builder.Services.AddScoped<BlogPostService>();
|
||||
builder.Services.AddScoped<LoggerService>();
|
||||
|
||||
// Radzen Services
|
||||
builder.Services.AddScoped<DialogService>();
|
||||
builder.Services.AddScoped<NotificationService>();
|
||||
builder.Services.AddScoped<TooltipService>();
|
||||
builder.Services.AddScoped<ContextMenuService>();
|
||||
|
||||
// Add mapster mapper
|
||||
builder.Services.AddMapster();
|
||||
|
||||
builder.Services.CreateDatabase().GetAwaiter().GetResult();
|
||||
|
||||
builder.Services.AddDatabaseDeveloperPageExceptionFilter();
|
||||
|
||||
// Add services to the container.
|
||||
builder.Services.AddRazorComponents()
|
||||
.AddInteractiveServerComponents();
|
||||
|
||||
var app = builder.Build();
|
||||
|
||||
app.UseSerilogRequestLogging();
|
||||
|
||||
// Configure the HTTP request pipeline.
|
||||
if (!app.Environment.IsDevelopment())
|
||||
{
|
||||
app.UseExceptionHandler("/Error", createScopeForErrors: true);
|
||||
// The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
|
||||
app.UseHsts();
|
||||
}
|
||||
|
||||
app.UseHttpsRedirection();
|
||||
|
||||
app.UseStaticFiles();
|
||||
app.UseAntiforgery();
|
||||
|
||||
app.MapRazorComponents<App>()
|
||||
.AddInteractiveServerRenderMode();
|
||||
|
||||
app.Run();
|
||||
}
|
||||
}
|
||||
|
||||
public static class ServiceCollectionExtensions
|
||||
{
|
||||
public static async Task CreateDatabase(this IServiceCollection services)
|
||||
{
|
||||
using (IServiceScope tmp = services.BuildServiceProvider().CreateScope())
|
||||
{
|
||||
await using var _context = tmp.ServiceProvider.GetRequiredService<ApplicationDbContext>();
|
||||
var seedData = tmp.ServiceProvider.GetRequiredService<SeedData>();
|
||||
await seedData.CreateInitialData();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public static class MapsterConfiguration
|
||||
{
|
||||
public static void AddMapster(this IServiceCollection services)
|
||||
{
|
||||
var typeAdapterConfig = TypeAdapterConfig.GlobalSettings;
|
||||
Assembly applicationAssembly = typeof(BaseViewModel<,>).Assembly;
|
||||
typeAdapterConfig.Scan(applicationAssembly);
|
||||
|
||||
var mapperConfig = new Mapper(typeAdapterConfig);
|
||||
services.AddSingleton<IMapper>(mapperConfig);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,14 @@
|
||||
{
|
||||
"$schema": "http://json.schemastore.org/launchsettings.json",
|
||||
"profiles": {
|
||||
"https": {
|
||||
"commandName": "Project",
|
||||
"dotnetRunMessages": true,
|
||||
"launchBrowser": true,
|
||||
"applicationUrl": "https://localhost:5001;http://localhost:5000",
|
||||
"environmentVariables": {
|
||||
"ASPNETCORE_ENVIRONMENT": "Development"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,101 @@
|
||||
using BlazorAppRadzenNet8SerilogLogging.Data;
|
||||
using BlazorAppRadzenNet8SerilogLogging.Models;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Radzen;
|
||||
using System.Linq.Dynamic.Core;
|
||||
|
||||
namespace BlazorAppRadzenNet8SerilogLogging.Services;
|
||||
|
||||
public class BlogPostService
|
||||
{
|
||||
private readonly ILogger<LoggerService> _logger;
|
||||
private readonly ApplicationDbContext _context;
|
||||
|
||||
public BlogPostService(ILogger<LoggerService> logger, ApplicationDbContext context)
|
||||
{
|
||||
_logger = logger;
|
||||
_context = context;
|
||||
}
|
||||
|
||||
public async Task<BlogPost?> GetBlogPostByIdAsync(int id)
|
||||
{
|
||||
_logger.LogInformation($"Called GetBlogPostByIdAsync", id);
|
||||
return await _context.BlogPosts.FirstOrDefaultAsync(x => x.Id == id);
|
||||
}
|
||||
|
||||
public async Task<(IEnumerable<BlogPost> Result, int TotalCount)> GetBlogPostsAsync(string? filter = default, int? top = default, int? skip = default, string? orderby = default, string? expand = default, string? select = default, bool? count = default)
|
||||
{
|
||||
_logger.LogInformation($"Called GetBlogPostsAsync");
|
||||
|
||||
var query = _context.BlogPosts.AsQueryable();
|
||||
|
||||
if (!string.IsNullOrEmpty(filter))
|
||||
query = query.Where(filter);
|
||||
|
||||
if (!string.IsNullOrEmpty(orderby))
|
||||
query = query.OrderBy(orderby);
|
||||
|
||||
int totalCount = 0;
|
||||
if (count == true)
|
||||
totalCount = query.Count();
|
||||
|
||||
IEnumerable<BlogPost>? result;
|
||||
if (skip == null || top == null)
|
||||
result = await query.ToListAsync();
|
||||
else
|
||||
result = await query.Skip(skip.Value).Take(top.Value).ToListAsync();
|
||||
|
||||
return (result, totalCount);
|
||||
}
|
||||
|
||||
public async Task<bool> AddBlogPostAsync(BlogPost blogPost)
|
||||
{
|
||||
_logger.LogInformation($"Called AddBlogPostAsync");
|
||||
|
||||
try
|
||||
{
|
||||
await _context.BlogPosts.AddAsync(blogPost);
|
||||
await _context.SaveChangesAsync();
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public async Task<bool> UpdateBlogPostAsync(int id, BlogPost blogPost)
|
||||
{
|
||||
_logger.LogInformation($"Called UpdateBlogPostAsync");
|
||||
|
||||
try
|
||||
{
|
||||
var oldBlogPost = _context.BlogPosts.FirstOrDefault(x => x.Id == id);
|
||||
if (oldBlogPost == null) return false;
|
||||
|
||||
oldBlogPost.Title = blogPost.Title;
|
||||
oldBlogPost.Content = blogPost.Content;
|
||||
|
||||
await _context.SaveChangesAsync();
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public async Task<bool> DeleteBlogPostByIdAsync(int id)
|
||||
{
|
||||
_logger.LogInformation($"Called DeleteBlogPostByIdAsync", id);
|
||||
|
||||
var blogPost = await _context.BlogPosts.FirstOrDefaultAsync(x => x.Id == id);
|
||||
if (blogPost == null)
|
||||
return false;
|
||||
|
||||
_context.BlogPosts.Remove(blogPost);
|
||||
await _context.SaveChangesAsync();
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,73 @@
|
||||
using BlazorAppRadzenNet8SerilogLogging.Models;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Radzen;
|
||||
using System.Linq.Dynamic.Core;
|
||||
|
||||
namespace BlazorAppRadzenNet8SerilogLogging.Data;
|
||||
|
||||
public class LoggerService
|
||||
{
|
||||
private readonly ILogger<LoggerService> _logger;
|
||||
private readonly ApplicationLoggerDbContext _loggerDbContext;
|
||||
|
||||
public LoggerService(ILogger<LoggerService> logger, ApplicationLoggerDbContext loggerDbContext)
|
||||
{
|
||||
_logger = logger;
|
||||
_loggerDbContext = loggerDbContext;
|
||||
}
|
||||
|
||||
public async Task<Log?> GetLogByIdAsync(int id)
|
||||
{
|
||||
_logger.LogInformation($"Called GetLogByIdAsync", id);
|
||||
return await _loggerDbContext.Logs.FirstOrDefaultAsync(x => x.id == id);
|
||||
}
|
||||
|
||||
public async Task<(IEnumerable<Log> Result, int TotalCount)> GetLogsAsync(string? filter = default, int? top = default, int? skip = default, string? orderby = default, string? expand = default, string? select = default, bool? count = default)
|
||||
{
|
||||
_logger.LogInformation($"Called GetLogsAsync");
|
||||
|
||||
var query = _loggerDbContext.Logs.AsQueryable();
|
||||
|
||||
if (!string.IsNullOrEmpty(filter))
|
||||
query = query.Where(filter);
|
||||
|
||||
if (!string.IsNullOrEmpty(orderby))
|
||||
query = query.OrderBy(orderby);
|
||||
|
||||
int totalCount = 0;
|
||||
if (count == true)
|
||||
totalCount = query.Count();
|
||||
|
||||
IEnumerable<Log>? result;
|
||||
if (skip == null || top == null)
|
||||
result = await query.ToListAsync();
|
||||
else
|
||||
result = await query.Skip(skip.Value).Take(top.Value).ToListAsync();
|
||||
|
||||
return (result, totalCount);
|
||||
}
|
||||
|
||||
public async Task<bool> DeleteLogByIdAsync(int id)
|
||||
{
|
||||
_logger.LogInformation($"Called DeleteLogByIdAsync", id);
|
||||
|
||||
var log = await _loggerDbContext.Logs.FirstOrDefaultAsync(x => x.id == id);
|
||||
if (log == null)
|
||||
return false;
|
||||
|
||||
_loggerDbContext.Logs.Remove(log);
|
||||
await _loggerDbContext.SaveChangesAsync();
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public async Task<bool?> DeleteAllLogsAsync()
|
||||
{
|
||||
_logger.LogInformation($"Called DeleteAllLogsAsync");
|
||||
var all = await _loggerDbContext.Logs.ToListAsync();
|
||||
_loggerDbContext.Logs.RemoveRange(all); ;
|
||||
await _loggerDbContext.SaveChangesAsync();
|
||||
_logger.LogInformation($"Deleted All Logs.");
|
||||
return true;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,45 @@
|
||||
using Mapster;
|
||||
|
||||
namespace BlazorAppRadzenNet8SerilogLogging.ViewModels;
|
||||
|
||||
/// <summary>
|
||||
/// Model <-> ViewModel Mapping
|
||||
/// </summary>
|
||||
/// <typeparam name="TViewModel">ViewModel</typeparam>
|
||||
/// <typeparam name="TModel">Model</typeparam>
|
||||
public abstract class BaseViewModel<TViewModel, TModel> : IRegister
|
||||
where TViewModel : class, new()
|
||||
where TModel : class, new()
|
||||
{
|
||||
public TModel ToModel()
|
||||
{
|
||||
return this.Adapt<TModel>();
|
||||
}
|
||||
|
||||
public TModel ToModel(TModel model)
|
||||
{
|
||||
return (this as TViewModel).Adapt(model);
|
||||
}
|
||||
|
||||
public static TViewModel FromModel(TModel model)
|
||||
{
|
||||
return model.Adapt<TViewModel>();
|
||||
}
|
||||
|
||||
private TypeAdapterConfig Config { get; set; }
|
||||
|
||||
public virtual void AddCustomMappings()
|
||||
{ }
|
||||
|
||||
protected TypeAdapterSetter<TViewModel, TModel> SetCustomMappings()
|
||||
=> Config.ForType<TViewModel, TModel>();
|
||||
|
||||
protected TypeAdapterSetter<TModel, TViewModel> SetCustomMappingsInverse()
|
||||
=> Config.ForType<TModel, TViewModel>();
|
||||
|
||||
public void Register(TypeAdapterConfig config)
|
||||
{
|
||||
Config = config;
|
||||
AddCustomMappings();
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,18 @@
|
||||
using BlazorAppRadzenNet8SerilogLogging.Models;
|
||||
using System.ComponentModel.DataAnnotations;
|
||||
|
||||
namespace BlazorAppRadzenNet8SerilogLogging.ViewModels;
|
||||
|
||||
public class BlogPostViewModel : BaseViewModel<BlogPostViewModel, BlogPost>
|
||||
{
|
||||
public int Id { get; set; }
|
||||
|
||||
[Required(AllowEmptyStrings = false, ErrorMessage = "Title can not be empty")]
|
||||
public string Title { get; set; } = string.Empty;
|
||||
|
||||
[Required(AllowEmptyStrings = false, ErrorMessage = "Content can not be empty")]
|
||||
public string Content { get; set; } = string.Empty;
|
||||
|
||||
public string TitleShort { get; set; } = string.Empty;
|
||||
public string ContentShort { get; set; } = string.Empty;
|
||||
}
|
||||
@@ -0,0 +1,13 @@
|
||||
using BlazorAppRadzenNet8SerilogLogging.Models;
|
||||
|
||||
namespace BlazorAppRadzenNet8SerilogLogging.ViewModels;
|
||||
|
||||
public class LogViewModel : BaseViewModel<LogViewModel, Log>
|
||||
{
|
||||
public int id { get; set; }
|
||||
public DateTime Timestamp { get; set; }
|
||||
public string Level { get; set; } = string.Empty;
|
||||
public string Exception { get; set; } = string.Empty;
|
||||
public string RenderedMessage { get; set; } = string.Empty;
|
||||
public string Properties { get; set; } = string.Empty;
|
||||
}
|
||||
@@ -0,0 +1,11 @@
|
||||
{
|
||||
"ConnectionStrings": {
|
||||
"SqliteLogger": "Data Source=LogsFolder/logs.db;Cache=Shared"
|
||||
},
|
||||
"Logging": {
|
||||
"LogLevel": {
|
||||
"Default": "Information",
|
||||
"Microsoft.AspNetCore": "Warning"
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,12 @@
|
||||
{
|
||||
"ConnectionStrings": {
|
||||
"SqliteLogger": "Data Source=LogsFolder/logs.db;Cache=Shared"
|
||||
},
|
||||
"Logging": {
|
||||
"LogLevel": {
|
||||
"Default": "Information",
|
||||
"Microsoft.AspNetCore": "Warning"
|
||||
}
|
||||
},
|
||||
"AllowedHosts": "*"
|
||||
}
|
||||
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
Binary file not shown.
|
After Width: | Height: | Size: 1.1 KiB |
Reference in New Issue
Block a user