In this blog post, I will show you how to show the deterministic progress bar in Blazor Web Assembly. To achieve the goal, we don’t need any third-party library. C# already has one interface, IProgress
, that exposes one method, Report
, that sends the data to the caller.
Let’s see the following method. I am making API calls to JSON placeholder in a loop in this method. I have injected the IProgress<int>
interface in the GetPosts, and then I am calculating the progress value and calling the report method.
private async Task GetPosts(IEnumerable<int> ids, IProgress<int> progress = null)
{
using var client = new HttpClient();
var tasks = new List<Task<HttpResponseMessage>>();
tasks = ids.Select(async x =>
{
var todoTask = await client.GetAsync("https://jsonplaceholder.typicode.com/todos/" + x);
await Task.Delay(1000);
if (progress == null) return todoTask;
eventCount++;
var percentage = (double)eventCount / tasks.Count();
var p = percentage * 100;
var pint = (int)Math.Round(p, 0);
progress.Report(pint);
return todoTask;
}
).ToList();
var allResult = await Task.WhenAll(tasks);
foreach (var element in allResult)
{
var content = await element.Content.ReadAsStringAsync();
}
}
}
Let’s see how to show the progress bar in the UI. Here I am creating the instance of the Progress class and then passing the action method in the body where I get the progress value. Once I receive the progress value, I re-render the UI by calling the StateHasChanged
method.
<div class="progress">
<div class="progress-bar bg-success"
role="progressbar" style="width: @(progressValue)%" aria-valuenow="@progressValue" aria-valuemin="0" aria-valuemax="100"></div>
</div>
@code {
int eventCount = 0;
int progressValue = 0;
protected override async Task OnInitializedAsync()
{
var progress = new Progress<int>(x =>
{
progressValue = x;
StateHasChanged();
});
await GetPosts(new List<int>() { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11 }, progress);
}
Complete Code
@page "/progress"
@using System.Net.Http
<div class="progress">
<div class="progress-bar bg-success"
role="progressbar" style="width: @(progressValue)%" aria-valuenow="@progressValue" aria-valuemin="0" aria-valuemax="100"></div>
</div>
@code {
int eventCount = 0;
int progressValue = 0;
protected override async Task OnInitializedAsync()
{
var progress = new Progress<int>(x =>
{
progressValue = x;
StateHasChanged();
});
await GetPosts(new List<int>() { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11 }, progress);
}
private async Task GetPosts(IEnumerable<int> ids, IProgress<int> progress = null)
{
using var client = new HttpClient();
var tasks = new List<Task<HttpResponseMessage>>();
tasks = ids.Select(async x =>
{
var todoTask = await client.GetAsync("https://jsonplaceholder.typicode.com/todos/" + x);
await Task.Delay(1000);
if (progress == null) return todoTask;
eventCount++;
var percentage = (double)eventCount / tasks.Count();
var p = percentage * 100;
var pint = (int)Math.Round(p, 0);
progress.Report(pint);
return todoTask;
}
).ToList();
var allResult = await Task.WhenAll(tasks);
foreach (var element in allResult)
{
var content = await element.Content.ReadAsStringAsync();
}
}
}
The progress bar does not update synchronously, the one-second delay between each task does not show.
ReplyDelete