Closed danroth27 closed 4 months ago
@MackinnonBuck please add a comment documenting the ReconnectionUI work that you've done.
@javiercn please add a comment with the Static Assets compression work that you've done. Also the new APIs for getting the current RenderMode please.
@halter73 please add a comment documenting the AuthenticationStateProvider work.
The following changes have been made to the default Blazor Server reconnection experience:
Blazor.start({
circuit: {
reconnectionOptions: {
retryIntervalMilliseconds: (previousAttempts, maxRetries) => previousAttempts >= maxRetries ? null : previousAttempts * 1000,
},
},
});
We've introduced an API to make easier to answer 3 questions many component authors have. Making some scenarios more straightforward. These questions are:
ComponentBase (and per extension your components) offer a new Platform property (soon to be renamed RendererInfo) that exposes a `(Name, and IsInteractive) properties.
Static
, Server
, WebAssembly
or WebView
.Platform.IsInteractive
indicates whether the platform supports interactivity. This is true for all implementations except Static
.Finally, the other property ComponentBase exposes is AssignedRenderMode
. This property exposes the render mode value defined in the component hierarchy (if any) via the render-mode
attribute on a root component or the [RenderMode]
attribute. Its values can be InteractiveServer
, InteractiveAuto
or InteractiveWebassembly
.
These values are most useful during prerendering, as they allow you to know where the component will transition to after prerendering, and you might use that to render different content.
For example, if you create a Form component and the form is going to be rendered interactively, you might choose to disable the inputs during prerendering, until the component becomes interactive and enables them.
Alternatively, if the component is not going to be rendered in an interactive context, you might render markup to support performing any action through regular web mechanics.
A big part of delivering performant web applications involves optimizing asset delivery to the browser. This process entails many aspects, among some of them are:
To this effect, we are introducing a new functionality called MapStaticAssets
. MapStaticAssets
works by combining work done at build/publish time to gather information about all the static resources in your application with a runtime library that is capable of processing that information and using it to better serve the files to the browser.
MapStaticAssets
can replace UseStaticFiles
in most situations, however, its optimized for serving the assets that your app has knowledge off at build and publish time. This means that if your app serves assets from other locations on disk, or embedded resources, etc. then using UseStaticFiles
is still the right choice.
What are the things that MapStaticAssets does that UseStaticFiles doesn't?
What are some of the gains that we can expect:
This is the default Razor Pages template (did we mention that this works for all UI flavors, Blazor, MVC, Razor Pages or your own custom framework?)
File | Original | Compressed | % Reduction |
---|---|---|---|
bootstrap.min.css | 163 | 17.5 | 89.26% |
jquery.js | 89.6 | 28 | 68.75% |
bootstrap.min.js | 78.5 | 20 | 74.52% |
For a total of 331.1 KB uncompressed vs 65.5 KB compressed.
This is the reduction that you get if you use the Fluent UI Blazor components library:
File | Original | Compressed | % Reduction |
---|---|---|---|
fluent.js | 384 | 73 | 80.99% |
fluent.css | 94 | 11 | 88.30% |
For a total of 478 KB uncompressed to 84 KB compressed.
Or, if you use other popular Blazor component libraries like MudBlazor:
File | Original | Compressed | Reduction |
---|---|---|---|
MudBlazor.min.css | 541 | 37.5 | 93.07% |
MudBlazor.min.js | 47.4 | 9.2 | 80.59% |
For a total of 588.4 KB to 46.7 KB
The best part is that all of this happens automatically for the app after using MapStaticAssets
. When you decide to bring in a new library or copy some new JS/CSS library to your project, you don't have to do anything. It'll get optimized as part of the build and served to the browser faster, which is especially important for mobile environments with lower bandwidth or spotty connections.
You might be wondering, if I have turn on dynamic compression on the server, how do I benefit from this? To begin, this approach is simpler, because there is no server specific configuration for you to figure out.
Then, even in this situation, you still benefit from leveraging MapStaticAssets instead of letting your IIS or other server do the compression for you. This is for two reasons:
We are providing a new solution template in .NET 9 starting in Preview 5 to make it easier to create .NET MAUI native and Blazor web client apps that share the same UI. This allows you to create client apps that can target Android, iOS, Mac, Windows and Web maximizing code reuse.
The template allows you to choose a Blazor interactive render mode for the web app and will create the appropriate projects for you, including a Blazor Web App and a .NET MAUI Blazor Hybrid app. It will then wire them up to use a shared Razor Class Library that has all of the UI components and pages. It also includes sample code that shows you how to use service injection to provide different interface implementations for the Blazor Hybrid and Blazor Web App. In .NET 8 this is a manual process documented here: Build a .NET MAUI Blazor Hybrid app with a Blazor Web App.
To get started, install the .NET 9 SDK (Preview 5 or higher) then install the .NET MAUI workload which contains the template.
dotnet workload install maui
You can then create the template from the commandline like this:
dotnet new maui-blazor-web
Alternative, you can use the template from Visual Studio:
TODO: Show screen capture of the template in Visual Studio
Note: Currently Blazor hybrid apps do not ignore the Blazor rendering modes if they are defined at the per page/component level and will throw an error. This work will be completed in a subsequent preview of .NET 9. For more details see #51235.
When you create a new Blazor Web App project with authentication using "Individual Accounts" and you enable WebAssembly-based interactivity, the project will include a custom AuthenticationStateProvider
in both the server and client projects to flow the user's authentication state to the browser. Authenticating on the server rather than the client allows the app to access authentication state during prerendering and before the WebAssembly runtime is initialized. The custom AuthenticationStateProvider
implementations use the PersistentComponentState
service to serialize the authentication state into HTML comments and then read it back from WebAssembly to create a new AuthenticationState
instance. This works well if you've started from the Blazor Web App project template and selected the "Individual Accounts" option, but it's a lot of code to implement yourself or copy if you're trying to add authentication to an existing project.
To make this easier, .NET 9 Preview 5 introduces AddAuthenticationStateSerialization
and AddAuthenticationStateDeserialization
APIs that can be called in the server and client projects:
// Server Program.cs
builder.Services.AddRazorComponents()
.AddInteractiveWebAssemblyComponents()
.AddAuthenticationStateSerialization();
// Client Program.cs
builder.Services.AddAuthorizationCore();
builder.Services.AddCascadingAuthenticationState();
builder.Services.AddAuthenticationStateDeserialization();
By default, this will only serialize the server-side name and role claims for access in the browser. If you want to include all claims, you can write the following on the server instead:
// Server Program.cs
builder.Services.AddRazorComponents()
.AddInteractiveWebAssemblyComponents()
.AddAuthenticationStateSerialization(options => options.SerializeAllClaims = true);
The Blazor Web App project template has been updated to use these APIs, so you can try out using the new template.
Progress report:
I'll report back here later to confirm today's goals are achieved.
Progress report:
All reference article content merged and cross-linked. What's New content for Blazor merged. 🍻
PR list ...
~I'm working on the Blazor-specific MapStaticAssets
coverage.~ Done! Just waiting to hear if it will be reviewed or not.
~FYI (I noticed in passing): The main doc set's Static Files article hasn't been updated for the new middleware:~
~https://learn.microsoft.com/en-us/aspnet/core/fundamentals/static-files?view=aspnetcore-9.0~
Nevermind! That's covered by an open issue.
Description
Update the "What's new in ASP.NET Core 9.0" for .NET 9 Preview 5
Page URL
https://learn.microsoft.com/en-us/aspnet/core/release-notes/aspnetcore-9.0?view=aspnetcore-8.0
Content source URL
https://github.com/dotnet/AspNetCore.Docs/blob/main/aspnetcore/release-notes/aspnetcore-9.0.md
Document ID
4e75ad25-2c3f-b28e-6a91-ac79a9c683b6
Article author
@Rick-Anderson @tdykstra @guardrex