AugusteVN.Razor.State
1.0.5
See the version list below for details.
dotnet add package AugusteVN.Razor.State --version 1.0.5
NuGet\Install-Package AugusteVN.Razor.State -Version 1.0.5
<PackageReference Include="AugusteVN.Razor.State" Version="1.0.5" />
paket add AugusteVN.Razor.State --version 1.0.5
#r "nuget: AugusteVN.Razor.State, 1.0.5"
// Install AugusteVN.Razor.State as a Cake Addin #addin nuget:?package=AugusteVN.Razor.State&version=1.0.5 // Install AugusteVN.Razor.State as a Cake Tool #tool nuget:?package=AugusteVN.Razor.State&version=1.0.5
State management in Blazor / Razor
Helper classes and components for state management in Blazor / Razor.
Persist Component State
Blazor has an unfortunate double rendering issue when pre-rendering is enabled.
The PesistState.cs
helps prevent double HTTP calls that occur on the component's lifecycle methods which get triggered twice.
Usage Example
Inheritance
public class BlogHttpClient : StateProvider
{
private readonly HttpClient _http;
public BlogHttpClient(HttpClient http, PersistentComponentState appState) : base(appState)
{
_http = http;
}
public async Task<BlogPostResponse?> FetchSelectedBlogPost(GetBlogPostForSlugRequest request)
{
return await FetchAsync<BlogPostResponse?>(nameof(GetBlogPostForSlug), () => GetBlogPostForSlug(request));
}
private async Task<BlogPostResponse?> GetBlogPostForSlug(GetBlogPostForSlugRequest request)
{
try
{
return await _http.GetFromJsonAsync<BlogPostResponse>($"{ApiRoutes.BlogPosts}/{request.Slug}");
}
catch (Exception e)
{
_logger.LogWarning("Could not get blog post with slug: {slug}, {@errorMessage}", request.Slug, e.Message);
return null;
}
}
}
Composition
builder.Services.AddSingleton<StateProvider>();
public class BlogHttpClient
{
private readonly HttpClient _http;
private readonly PersistentComponentState _appState;
private readonly StateProvider _stateProvider;
public BlogHttpClient(HttpClient http, PersistentComponentState appState, StateProvider persistState)
{
_http = http;
_appState = appState;
_stateProvider = stateProvider;
}
public async Task<BlogPostResponse?> FetchSelectedBlogPost(GetBlogPostForSlugRequest request)
{
return await stateProvider.FetchAsync<BlogPostResponse?>(_appState, nameof(GetBlogPostForSlug), () => GetBlogPostForSlug(request));
}
private async Task<BlogPostResponse?> GetBlogPostForSlug(GetBlogPostForSlugRequest request)
{
try
{
return await _http.GetFromJsonAsync<BlogPostResponse>($"{ApiRoutes.BlogPosts}/{request.Slug}");
}
catch (Exception e)
{
_logger.LogWarning("Could not get blog post with slug: {slug}, {@errorMessage}", request.Slug, e.Message);
return null;
}
}
INotifyPropertyChanged
Notify whenever the value of a property in our global state changes.
public class BlogHttpClient : StateProvider
{
private BlogPostResponse? _selectedBlogPost;
public BlogPostResponse? SelectedBlogPost {
get => _selectedBlogPost;
set => SetField(ref _selectedBlogPost, value);
}
private readonly HttpClient _http;
public BlogHttpClient(HttpClient http, PersistentComponentState appState) : base(appState)
{
_http = http;
}
public async Task InitSelectedBlogPost(GetBlogPostForSlugRequest request)
{
SelectedBlogPost = await FetchAsync<BlogPostResponse?>(nameof(GetBlogPostForSlug), () => GetBlogPostForSlug(request));
}
private async Task<BlogPostResponse?> GetBlogPostForSlug(GetBlogPostForSlugRequest request)
{
try
{
return await _http.GetFromJsonAsync<BlogPostResponse>($"{ApiRoutes.BlogPosts}/{request.Slug}");
}
catch (Exception e)
{
_logger.LogWarning("Could not get blog post with slug: {slug}, {@errorMessage}", request.Slug, e.Message);
return null;
}
}
}
ExampleComponent.razor
@implements IDisposable
@inject BlogHttpClient BlogHttpClient
@code {
protected override void OnInitialized()
{
BlogHttpClient.PropertyChanged += HandleStateChanged;
}
private void HandleStateChanged(object? sender, PropertyChangedEventArgs e)
{
if (e.PropertyName == nameof(BlogHttpClient.SelectedBlogPost))
{
Console.WriteLine(BlogHttpClient.SelectedBlogPost.Title);
}
}
public void Dispose()
{
State.PropertyChanged -= HandleStateChanged;
}
}
To see it in action, watch: Prevent this Blazor Prerendering ISSUE C# .NET 8.
Brought to you by: kiss-code.com.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net8.0 is compatible. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
-
net8.0
- Microsoft.AspNetCore.Components.Web (>= 8.0.0)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Rename PersistStateProvider to StateProvider.