6.9 KiB
title | author | description | monikerRange | ms.author | ms.custom | ms.date | uid |
---|---|---|---|---|---|---|---|
Work with images in ASP.NET Core Blazor | guardrex | Learn how to work with images in ASP.NET Core Blazor apps. | >= aspnetcore-6.0 | riande | mvc | 11/08/2022 | blazor/images |
Work with images in ASP.NET Core Blazor
This article describes common scenarios for working with images in Blazor apps.
Dynamically set an image source
The following example demonstrates how to dynamically set an image's source with a C# field.
For the example in this section:
-
Obtain three images from any source or right-click each of the following images to save them locally. Name the images
image1.png
,image2.png
, andimage3.png
. -
Place the images in a new folder named
images
in the app's web root (wwwroot
). The use of theimages
folder is only for demonstration purposes. You can organize images in any folder layout that you prefer, including serving the images directly from thewwwroot
folder.
In the following ShowImage1
component:
- The image's source (
src
) is dynamically set to the value ofimageSource
in C#. - The
ShowImage
method updates theimageSource
field based on an imageid
argument passed to the method. - Rendered buttons call the
ShowImage
method with an image argument for each of the three available images in theimages
folder. The file name is composed using the argument passed to the method and matches one of the three images in theimages
folder.
Pages/ShowImage1.razor
:
:::moniker range=">= aspnetcore-7.0"
:::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Pages/images/ShowImage1.razor":::
:::moniker-end
:::moniker range="< aspnetcore-7.0"
:::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Pages/images/ShowImage1.razor":::
:::moniker-end
The preceding example uses a C# field to hold the image's source data, but you can also use a C# property to hold the data.
[!NOTE] Avoid using a loop variable directly in a lambda expression, such as
i
in the precedingfor
loop example. Otherwise, the same variable is used by all lambda expressions, which results in use of the same value in all lambdas. Capture the variable's value in a local variable. In the preceding example:
- The loop variable
i
is assigned toimageId
.imageId
is used in the lambda expression.Alternatively, use a
foreach
loop with xref:System.Linq.Enumerable.Range%2A?displayProperty=nameWithType, which doesn't suffer from the preceding problem:@foreach (var imageId in Enumerable.Range(1,3)) { <button @onclick="() => ShowImage(imageId)"> Image @imageId </button> }
For more information, see xref:blazor/components/event-handling#lambda-expressions.
Stream image data
An image can be directly sent to the client using Blazor's streaming interop features instead of hosting the image at a public URL.
The example in this section streams image source data using JavaScript (JS) interop. The following setImage
JS function accepts the <img>
tag id
and data stream for the image. The function performs the following steps:
- Reads the provided stream into an
ArrayBuffer
. - Creates a
Blob
to wrap theArrayBuffer
. - Creates an object URL to serve as the address for the image to be shown.
- Updates the
<img>
element with the specifiedimageElementId
with the object URL just created. - To prevent memory leaks, the function calls
revokeObjectURL
to dispose of the object URL when the component is finished working with an image.
<script>
window.setImage = async (imageElementId, imageStream) => {
const arrayBuffer = await imageStream.arrayBuffer();
const blob = new Blob([arrayBuffer]);
const url = URL.createObjectURL(blob);
const image = document.getElementById(imageElementId);
image.onload = () => {
URL.revokeObjectURL(url);
}
image.src = url;
}
</script>
[!NOTE] For general guidance on JS location and our recommendations for production apps, see xref:blazor/js-interop/index#javascript-location.
The following ShowImage2
component:
- Injects services for an xref:System.Net.Http.HttpClient?displayProperty=fullName and xref:Microsoft.JSInterop.IJSRuntime?displayProperty=fullName.
- Includes an
<img>
tag to display an image. - Has a
GetImageStreamAsync
C# method to retrieve a xref:System.IO.Stream for an image. A production app may dynamically generate an image based on the specific user or retrieve an image from storage. The following example retrieves the .NET avatar for thedotnet
GitHub repository. - Has a
SetImageAsync
method that's triggered on the button's selection by the user.SetImageAsync
performs the following steps:- Retrieves the xref:System.IO.Stream from
GetImageStreamAsync
. - Wraps the xref:System.IO.Stream in a xref:Microsoft.JSInterop.DotNetStreamReference, which allows streaming the image data to the client.
- Invokes the
setImage
JavaScript function, which accepts the data on the client.
- Retrieves the xref:System.IO.Stream from
[!NOTE] Blazor Server apps use a dedicated xref:System.Net.Http.HttpClient service to make requests, so no action is required by the developer in Blazor Server apps to register an xref:System.Net.Http.HttpClient service. Blazor WebAssembly apps have a default xref:System.Net.Http.HttpClient service registration when the app is created from a Blazor WebAssembly project template. If an xref:System.Net.Http.HttpClient service registration isn't present in
Program.cs
of a Blazor WebAssembly app, provide one by addingbuilder.Services.AddHttpClient();
. For more information, see xref:fundamentals/http-requests.
Pages/ShowImage2.razor
:
:::moniker range=">= aspnetcore-7.0"
:::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Pages/images/ShowImage2.razor":::
:::moniker-end
:::moniker range="< aspnetcore-7.0"
:::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Pages/images/ShowImage2.razor":::
:::moniker-end