From 050c28168e0c1b3eb44c61c0ff79833c0a49dae4 Mon Sep 17 00:00:00 2001 From: Luke Latham <1622880+guardrex@users.noreply.github.com> Date: Fri, 1 Dec 2023 08:27:19 -0500 Subject: [PATCH] Improve AdditionalAssemblies coverage (#31131) --- aspnetcore/blazor/components/render-modes.md | 11 +-- aspnetcore/blazor/fundamentals/routing.md | 71 +++++++++++++++++--- aspnetcore/migration/70-80.md | 11 ++- 3 files changed, 69 insertions(+), 24 deletions(-) diff --git a/aspnetcore/blazor/components/render-modes.md b/aspnetcore/blazor/components/render-modes.md index af7403551c..d6cca04b5a 100644 --- a/aspnetcore/blazor/components/render-modes.md +++ b/aspnetcore/blazor/components/render-modes.md @@ -532,16 +532,9 @@ The following component results in a runtime error when the component is rendere > :::no-loc text="Cannot create a component of type 'BlazorSample.Components.SharedMessage' because its render mode 'Microsoft.AspNetCore.Components.Web.InteractiveWebAssemblyRenderMode' is not supported by Interactive Server rendering."::: -## Discover components from additional assemblies for Static Server rendering +## Discover components from additional assemblies -Configure additional assemblies to discover routable Razor components for Static Server rendering using the method chained to . - -The following example includes the assembly of the `DifferentAssemblyCounter` component: - -```csharp -app.MapRazorComponents() - .AddAdditionalAssemblies(typeof(DifferentAssemblyCounter).Assembly); -``` +Additional assemblies must be disclosed to the Blazor framework to discover routable Razor components in referenced projects. For more information, see . ## Closure of circuits when there are no remaining Interactive Server components diff --git a/aspnetcore/blazor/fundamentals/routing.md b/aspnetcore/blazor/fundamentals/routing.md index cf97559009..9c2e16ebd0 100644 --- a/aspnetcore/blazor/fundamentals/routing.md +++ b/aspnetcore/blazor/fundamentals/routing.md @@ -19,6 +19,20 @@ This article explains how to manage Blazor app request routing and how to use th > [!IMPORTANT] > Code examples throughout this article show methods called on `Navigation`, which is an injected in classes and components. +:::moniker range=">= aspnetcore-8.0" + +## Static versus interactive routing + +*This section applies to Blazor Web Apps.* + +If [prerendering isn't disabled](xref:blazor/components/render-modes#prerendering), the Blazor router (`Router` component, `` in `Routes.razor`) performs static routing to components during static server-side rendering (static SSR). This type of routing is called *static routing*. + +When an interactive render mode is assigned to the `Routes` component, the Blazor router becomes interactive after static SSR with static routing on the server. This type of routing is called *interactive routing*. + +Static routers use endpoint routing and the HTTP request path to determine which component to render. When the router becomes interactive, it uses the document's URL (the URL in the browser's address bar) to determine which component to render. This means that the interactive router can dynamically change which component is rendered if the document's URL dynamically changes to another valid internal URL, and it can do so without performing an HTTP request to fetch new page content. + +:::moniker-end + ## Route templates :::moniker range=">= aspnetcore-8.0" @@ -135,30 +149,69 @@ Arbitrary items are supported as content of the `` tags, such as other ## Route to components from multiple assemblies -Use the parameter to specify additional assemblies for the component to consider when searching for routable components. Additional assemblies are scanned in addition to the assembly specified to `AppAssembly`. In the following example, `Component1` is a routable component defined in a referenced [component class library](xref:blazor/components/class-libraries). The following example results in routing support for `Component1`. +:::moniker range=">= aspnetcore-8.0" -:::moniker range=">= aspnetcore-6.0" +*This section applies to Blazor Web Apps.* + +Use the component's parameter and the endpoint convention builder to discover routable components in additional assemblies. The following subsections explain when and how to use each API. + +### Static routing + +To discover routable components from additional assemblies for static server-side rendering (static SSR), even if the router later becomes interactive for interactive rendering, the assemblies must be disclosed to the Blazor framework. Call the method with the additional assemblies chained to in the server project's `Program` file. + +The following example includes the routable components in the `BlazorSample.Client` project's assembly using the project's `_Imports.razor` file: + +```csharp +app.MapRazorComponents() + .AddAdditionalAssemblies(typeof(BlazorSample.Client._Imports).Assembly); +``` + +> [!NOTE] +> The preceding guidance also applies in [component class library](xref:blazor/components/class-libraries) scenarios. Additional important guidance for class libraries and static SSR is found in . + +### Interactive routing + +An interactive render mode can be assigned to the `Routes` component (`Routes.razor`) that makes the Blazor router become interactive after static SSR and static routing on the server. For example, `` assigns the Interactive Server render mode to the `Routes` component. The `Router` component inherits the Interactive Server render mode from the `Routes` component. The router becomes interactive after static routing on the server. + +If the `Routes` component is defined in the server project, the parameter of the `Router` component should include the `.Client` project's assembly. This allows the router to work correctly when rendered interactively. + +In the following example, the `Routes` component is in the server project, and the `_Imports.razor` file of the `BlazorSample.Client` project indicates the assembly to search for routable components: ```razor - @* ... Router component elements ... *@ + AppAssembly="..." + AdditionalAssemblies="new[] { typeof(BlazorSample.Client._Imports).Assembly }"> + ... ``` +Additional assemblies are scanned in addition to the assembly specified to . + +> [!NOTE] +> The preceding guidance also applies in [component class library](xref:blazor/components/class-libraries) scenarios. + +Alterntively, routable components only exist in the `.Client` project with global Interactive WebAssembly or Auto rendering applied, and the `Routes` component is defined in the `.Client` project, not the server project. In this case, there aren't external assemblies with routable components, so it isn't necessary to specify a value for . + :::moniker-end -:::moniker range="< aspnetcore-6.0" +:::moniker range="< aspnetcore-8.0" + +*This section applies to Blazor Server apps.* + +Use the component's parameter and the endpoint convention builder to discover routable components in additional assemblies. + +In the following example, `Component1` is a routable component defined in a referenced [component class library](xref:blazor/components/class-libraries) named `ComponentLibrary`: ```razor - @* ... Router component elements ... *@ + AppAssembly="..." + AdditionalAssemblies="new[] { typeof(ComponentLibrary.Component1).Assembly }"> + ... ``` +Additional assemblies are scanned in addition to the assembly specified to . + :::moniker-end ## Route parameters diff --git a/aspnetcore/migration/70-80.md b/aspnetcore/migration/70-80.md index 7dba5ccc06..63772026f2 100644 --- a/aspnetcore/migration/70-80.md +++ b/aspnetcore/migration/70-80.md @@ -89,6 +89,7 @@ The following migration scenarios are covered: * [Migrate the `` component to cascading authentication state services](#migrate-the-cascadingauthenticationstate-component-to-cascading-authentication-state-services) * [*New article*: HTTP caching issues during migration](#new-article-on-http-caching-issues) * [*New article*: New article on class libraries with static server-side rendering (static SSR)](#new-article-on-class-libraries-with-static-server-side-rendering-static-ssr) +* [Discover components from additional assemblies](#discover-components-from-additional-assemblies) For guidance on adding Blazor support to an ASP.NET Core app, see . @@ -431,14 +432,8 @@ Updated configuration guidance appears in the following locations: * [Temporary redirection URL validity duration](xref:blazor/security/server/index?view=aspnetcore-8.0&preserve-view=true#temporary-redirection-url-validity-duration): Covers control of the lifetime of data protection validity for temporary redirection URLs emitted by Blazor server-side rendering. * [Detailed errors](xref:blazor/fundamentals/handle-errors?view=aspnetcore-8.0&preserve-view=true#detailed-errors-for-razor-component-server-side-rendering): Covers enabling detailed errors for Razor component server-side rendering. * [Prerendering configuration](xref:blazor/components/render-modes?view=aspnetcore-8.0&preserve-view=true#prerendering): Prerendering is enabled by default for Blazor Web Apps. Follow this link for guidance on how to disable prerendering if you have special circumstances that require an app to disable prerendering. - - - ### Drop Blazor Server with Yarp routing workaround If you previously followed the guidance in for migrating a Blazor Server app with Yarp to .NET 6 or .NET 7, you can reverse the workaround steps that you took when following the article's guidance. Routing and deep linking for Blazor Server with Yarp work correctly in .NET 8. @@ -524,6 +519,10 @@ We've added a new article that discusses component library authorship in Razor c For more information, see . +### Discover components from additional assemblies + +When migrating from a Blazor Server app to a Blazor Web App, access the guidance in if the app uses routable components from additional assemblies, such as component class libraries. + ## Docker ### Update Docker images