Lets start with a sample Razor component that renders the UI:
Component.razor
Forename
Surname
Username
@foreach (var user in Users)
{
@user.Forename
@user.Surname
@user.Username
}
Lets look at the ‘code behind’ (ooof I’m old) AKA a partial class that represents the logic of the component (view)
Component.razor.cs
public partial class Component
{
private string _url = String.Empty;
IEnumerable Users { get; set; }
//The Configuration object needs to be injected via dependency injection
protected async override void OnInitializedAsync()
{
_url = Configuration.AllClientSettings().BaseServiceURI;
Users = (await http.CreateClient("ClientSettings").GetFromJsonAsync>($"{_URL}/api/lookup/User"))
.OrderBy(t => t.Username)
.ToList();
}
Looks pretty normal right?
The main issue is the way the method OnInitializedAsync() executes in Blazor. It actually executes TWICE, sort of confusing but there is a good reason!
A few things are going on here:
- Component.razor is initially rendered statically as part of the page, whilst async methods execute in the background.
- Now we’re awaiting the Users IEnumerable<UserModel>, this data may not be accessible on the initial page render since it’s a task that needs to be awaited.
- Once the browser has rendered the component the OnInitializedAsync() method is executed again.
Guess what happens when you try to enumerate an IEnumerable on the first render? It’s null (possibly) due to the nature of asynchronous programming, hence the error!
How to fix this behavior?
The solution is pretty simple, wait for the IEnumerable to be available.
@if(Users is not null)
{
foreach (var user in Users)
{
@user.Forename
@user.Surname
@user.Username
}
}
else
{
Hold tight... loading.
}
What's the good reason for this?
The reason for the OnInitalizedAsync() method executing twice is to allow developers to insert a loading message or load certain static assets before all the razor components are rendered, especially if they’re populated via asynchronous methods. This way the user knows something is happening and they’re waiting for the data/page to load.
Read more here
Hope this helps, I've fallen victim to this myself!
Want to learn more?
In Memory State Management with Blazor Server
Creating your first In Memory State Container with Blazor Server Web technology has come along way in the relatively recent past. With the surge in popularity of SPAs and Javascript Frameworks a lot of developers…
Read moreHow to add HealthChecks to a ASP.net Core application
Adding HealthChecks to your application can drastically reduce troubleshooting time Why would you want to monitor your applications health? Health checks can test an applications dependencies, such as databases and external service endpoints, to confirm…
Read moreHow can I dynamically render components using MudBlazor radio buttons?
You need to have a MudRadioGroup to contain your MudRadio buttons. On the MudRadioGroup you have the @bind-SelectedOption event callback exposed. You can set a switch statement on that to dynamically render your other components. Parent Component: TestComponent.razor Sample Project https://try.mudblazor.com/snippet/wOmRucPSpMDbtnAw
Read more