--- title: ASP.NET Core Blazor cascading values and parameters author: guardrex description: Learn how to flow data from an ancestor Razor component to descendent components. monikerRange: '>= aspnetcore-3.1' ms.author: riande ms.custom: mvc ms.date: 11/08/2022 uid: blazor/components/cascading-values-and-parameters --- # ASP.NET Core Blazor cascading values and parameters [!INCLUDE[](~/includes/not-latest-version.md)] This article explains how to flow data from an ancestor Razor component to descendent components. *Cascading values and parameters* provide a convenient way to flow data down a component hierarchy from an ancestor component to any number of descendent components. Unlike [Component parameters](xref:blazor/components/index#component-parameters), cascading values and parameters don't require an attribute assignment for each descendent component where the data is consumed. Cascading values and parameters also allow components to coordinate with each other across a component hierarchy. > [!NOTE] > The code examples in this article adopt [nullable reference types (NRTs) and .NET compiler null-state static analysis](xref:migration/50-to-60#nullable-reference-types-nrts-and-net-compiler-null-state-static-analysis), which are supported in ASP.NET Core 6.0 or later. When targeting ASP.NET Core 5.0 or earlier, remove the null type designation (`?`) from the `CascadingType?`, `@ActiveTab?`, `RenderFragment?`, `ITab?`, `TabSet?`, and `string?` types in the article's examples. ## `CascadingValue` component An ancestor component provides a cascading value using the Blazor framework's [`CascadingValue`](xref:Microsoft.AspNetCore.Components.CascadingValue%601) component, which wraps a subtree of a component hierarchy and supplies a single value to all of the components within its subtree. The following example demonstrates the flow of theme information down the component hierarchy of a layout component to provide a CSS style class to buttons in child components. The following `ThemeInfo` C# class is placed in a folder named `UIThemeClasses` and specifies the theme information. > [!NOTE] > For the examples in this section, the app's namespace is `BlazorSample`. When experimenting with the code in your own sample app, change the app's namespace to your sample app's namespace. `UIThemeClasses/ThemeInfo.cs`: :::moniker range=">= aspnetcore-7.0" :::code language="csharp" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/UIThemeClasses/ThemeInfo.cs"::: :::moniker-end :::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" :::code language="csharp" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/UIThemeClasses/ThemeInfo.cs"::: :::moniker-end :::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" :::code language="csharp" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/UIThemeClasses/ThemeInfo.cs"::: :::moniker-end :::moniker range="< aspnetcore-5.0" :::code language="csharp" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/UIThemeClasses/ThemeInfo.cs"::: :::moniker-end The following [layout component](xref:blazor/components/layouts) specifies theme information (`ThemeInfo`) as a cascading value for all components that make up the layout body of the property. `ButtonClass` is assigned a value of [`btn-success`](https://getbootstrap.com/docs/5.0/components/buttons/), which is a Bootstrap button style. Any descendent component in the component hierarchy can use the `ButtonClass` property through the `ThemeInfo` cascading value. `Shared/MainLayout.razor`: :::moniker range=">= aspnetcore-7.0" :::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Shared/MainLayout.razor"::: :::moniker-end :::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" :::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Shared/MainLayout.razor"::: :::moniker-end :::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" :::code language="razor" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/Shared/MainLayout.razor"::: :::moniker-end :::moniker range="< aspnetcore-5.0" :::code language="razor" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/Shared/MainLayout.razor"::: :::moniker-end ## `[CascadingParameter]` attribute To make use of cascading values, descendent components declare cascading parameters using the [`[CascadingParameter]` attribute](xref:Microsoft.AspNetCore.Components.CascadingParameterAttribute). Cascading values are bound to cascading parameters **by type**. Cascading multiple values of the same type is covered in the [Cascade multiple values](#cascade-multiple-values) section later in this article. The following component binds the `ThemeInfo` cascading value to a cascading parameter, optionally using the same name of `ThemeInfo`. The parameter is used to set the CSS class for the **`Increment Counter (Themed)`** button. `Pages/ThemedCounter.razor`: :::moniker range=">= aspnetcore-7.0" :::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Pages/ThemedCounter.razor"::: :::moniker-end :::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" :::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Pages/ThemedCounter.razor"::: :::moniker-end :::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" :::code language="razor" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/Pages/ThemedCounter.razor"::: :::moniker-end :::moniker range="< aspnetcore-5.0" :::code language="razor" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/Pages/ThemedCounter.razor"::: :::moniker-end :::moniker range=">= aspnetcore-6.0" Similar to a regular component parameter, components accepting a cascading parameter are rerendered when the cascading value is changed. For instance, configuring a different theme instance causes the `ThemedCounter` component from the [`CascadingValue` component](#cascadingvalue-component) section to rerender: `Shared/MainLayout.razor`: ```razor
@Body
@code { private ThemeInfo theme = new() { ButtonClass = "btn-success" }; private void ChangeToDarkTheme() { theme = new() { ButtonClass = "btn-darkmode-success" }; } } ``` can be used to indicate that a cascading parameter doesn't change after initialization. :::moniker-end ## Cascade multiple values To cascade multiple values of the same type within the same subtree, provide a unique string to each [`CascadingValue`](xref:Microsoft.AspNetCore.Components.CascadingValue%601) component and their corresponding [`[CascadingParameter]` attributes](xref:Microsoft.AspNetCore.Components.CascadingParameterAttribute). In the following example, two [`CascadingValue`](xref:Microsoft.AspNetCore.Components.CascadingValue%601) components cascade different instances of `CascadingType`: ```razor ... @code { private CascadingType? parentCascadeParameter1; [Parameter] public CascadingType? ParentCascadeParameter2 { get; set; } } ``` In a descendant component, the cascaded parameters receive their cascaded values from the ancestor component by : ```razor @code { [CascadingParameter(Name = "CascadeParam1")] protected CascadingType? ChildCascadeParameter1 { get; set; } [CascadingParameter(Name = "CascadeParam2")] protected CascadingType? ChildCascadeParameter2 { get; set; } } ``` ## Pass data across a component hierarchy Cascading parameters also enable components to pass data across a component hierarchy. Consider the following UI tab set example, where a tab set component maintains a series of individual tabs. > [!NOTE] > For the examples in this section, the app's namespace is `BlazorSample`. When experimenting with the code in your own sample app, change the namespace to your sample app's namespace. Create an `ITab` interface that tabs implement in a folder named `UIInterfaces`. `UIInterfaces/ITab.cs`: ```csharp using Microsoft.AspNetCore.Components; namespace BlazorSample.UIInterfaces; public interface ITab { RenderFragment ChildContent { get; } } ``` > [!NOTE] > For more information on , see . The following `TabSet` component maintains a set of tabs. The tab set's `Tab` components, which are created later in this section, supply the list items (`
  • ...
  • `) for the list (`
      ...
    `). Child `Tab` components aren't explicitly passed as parameters to the `TabSet`. Instead, the child `Tab` components are part of the child content of the `TabSet`. However, the `TabSet` still needs a reference each `Tab` component so that it can render the headers and the active tab. To enable this coordination without requiring additional code, the `TabSet` component *can provide itself as a cascading value* that is then picked up by the descendent `Tab` components. `Shared/TabSet.razor`: ```razor @using BlazorSample.UIInterfaces @code { [Parameter] public RenderFragment? ChildContent { get; set; } public ITab? ActiveTab { get; private set; } public void AddTab(ITab tab) { if (ActiveTab is null) { SetActiveTab(tab); } } public void SetActiveTab(ITab tab) { if (ActiveTab != tab) { ActiveTab = tab; StateHasChanged(); } } } ``` Descendent `Tab` components capture the containing `TabSet` as a cascading parameter. The `Tab` components add themselves to the `TabSet` and coordinate to set the active tab. `Shared/Tab.razor`: ```razor @using BlazorSample.UIInterfaces @implements ITab
  • @Title
  • @code { [CascadingParameter] public TabSet? ContainerTabSet { get; set; } [Parameter] public string? Title { get; set; } [Parameter] public RenderFragment? ChildContent { get; set; } private string? TitleCssClass => ContainerTabSet?.ActiveTab == this ? "active" : null; protected override void OnInitialized() { ContainerTabSet?.AddTab(this); } private void ActivateTab() { ContainerTabSet?.SetActiveTab(this); } } ``` The following `ExampleTabSet` component uses the `TabSet` component, which contains three `Tab` components. `Pages/ExampleTabSet.razor`: ```razor @page "/example-tab-set"

    Greetings from the first tab!

    Hello from the second tab!

    @if (showThirdTab) {

    Welcome to the disappearing third tab!

    Toggle this tab from the first tab.

    }
    @code { private bool showThirdTab; } ```