Realizzando rich web application con Blazor, molto spesso può essere necessario esportare dei dati in un file. Tramite Blazor, possiamo facilmente generare questi file sul browser dell'utente - e quindi senza scomodare il server - per poi consentirne il download.
Concentriamoci per prima cosa su quest'ultimo aspetto: per eseguire il download di un file da browser, a partire da uno stream di dati, abbiamo bisogno di utilizzare del codice JavaScript. Per esempio possiamo creare un modulo come il seguente, all'interno di wwwroot, usando il codice descritto nella documentazione ufficiale di Microsoft (https://learn.microsoft.com/en-us/aspnet/core/blazor/file-downloads?view=aspnetcore-7.0#download-from-a-stream):
export async function downloadFileFromStream(fileName, contentStreamReference) {
const arrayBuffer = await contentStreamReference.arrayBuffer();
const blob = new Blob([arrayBuffer]);
const url = URL.createObjectURL(blob);
const anchorElement = document.createElement('a');
anchorElement.href = url;
anchorElement.download = fileName ?? '';
anchorElement.click();
anchorElement.remove();
URL.revokeObjectURL(url);
}Si tratta di uno snipped JavaScript che accetta uno stream in input che utilizza per costruire poi un oggetto di tipo Blob. Il link a questo oggetto viene poi associato a un link invisibile, il cui evento click avvierà il download dal browser.
A questo punto, possiamo creare un componente, che chiameremo FileSaver, che rappresenterà il nostro wrapper C# sulla funzione che abbiamo visto in alto.
public class FileSaver
{
private Lazy<Task<IJSObjectReference>> _downloadModule;
public FileSaver(IJSRuntime js)
{
_downloadModule = new Lazy<Task<IJSObjectReference>>(() => js.InvokeAsync<IJSObjectReference>("import",
"./js/downloadFileFromStream.js").AsTask());
}
public async Task SaveAsAsync(string fileName, Stream stream)
{
var js = await _downloadModule.Value;
using var streamRef = new DotNetStreamReference(stream: stream);
await js.InvokeVoidAsync("downloadFileFromStream", fileName, streamRef);
}
public async ValueTask DisposeAsync()
{
if (_downloadModule.IsValueCreated)
{
await (await _downloadModule.Value).DisposeAsync();
}
}
}Questa classe, come prima cosa, sfrutta un oggetto Lazy<T> per implementare la logica di inizializzazione con JavaScript, in cui viene referenziato il modulo che abbiamo creato nello step precedente.
Poi espone un metodo SaveAsAsync, che non fa altro che invocare la funzione downloadFileFromStream, passando lo stream C#.
Ovviamente, implementa anche l'interfaccia IAsyncDisposable per distruggere la reference a JavaScript.
Dopo aver registrato questo servizio nell'IoC container
public static async Task Main(string[] args)
{
// ... altro codice qui ...
builder.Services.AddSingleton<FileSaver>();
await builder.Build().RunAsync();
}possiamo finalmente utilizzarlo in una delle nostre pagine.
Per esempio, immaginiamo di voler esportare il WeatherForecast della pagina FetchData in formato CSV.
Dopo aver referenziato il package CsvHelper
dotnet add package csvhelper
non dobbiamo far altro che aggiungere un apposito button al markup:
.. altro codice qui ..
<button class="btn btn-primary" @onclick="DownloadFileFromStream">
Download forecast data in CSV
</button>Il metodo DownloadFileFromStream non farà altro che generare un MemoryStream con il contenuto desiderato, e inviarlo al FileSaver che abbiamo creato:
private async Task DownloadFileFromStream()
{
using var stream = new MemoryStream();
using var writer = new StreamWriter(stream, leaveOpen:true);
using (var csv = new CsvWriter(writer, CultureInfo.InvariantCulture))
{
csv.WriteRecords(forecasts);
}
stream.Position = 0;
await this.FileSaver.SaveAsAsync("forecasts.csv", stream);
}Commenti
Per inserire un commento, devi avere un account.
Fai il login e torna a questa pagina, oppure registrati alla nostra community.
Approfondimenti
Centralizzare gli endpoint AI Foundry con Azure API Management
Utilizzare noopener e noreferrer nei link HTML
Usare il metodo nameof con un tipo generico in C# 14
Gestire gli errori nelle Promise JavaScript con try()
DevSecOps per .NET: dalla teoria alla pratica
Impostare il tipo di supporto dei campi Json in Entity Framework con Sql Server
Evidenziare una porzione di testo in un pagina dopo una navigazione
Eseguire i pre-commit hook di git con dependabot
Proteggere l'endpoint dell'agente A2A delle Logic App
Dallo sviluppo locale ad Azure con .NET Aspire
Mappare una complex property di una entity su un campo JSON
I più letti di oggi
- Global Azure 2026 - ASPItalia.com - Milano
- L'agenda di #GlobalAzure 2026 by ASPItalia.com è pronta: da #AKS a #AIFoundry, passando per #MCP, #Fabric e tanto altro.Ci vediamo il 16 aprile a Milano! https://aspit.co/globalazure-26
- Future Dev Day - Milano
- Il nuovo persistent state in Blazor
- Eseguire i pre-commit hook di git con dependabot




