diff --git a/aspnetcore/blazor/components/index.md b/aspnetcore/blazor/components/index.md index dd9e46e3fa..f4a853ad05 100644 --- a/aspnetcore/blazor/components/index.md +++ b/aspnetcore/blazor/components/index.md @@ -971,188 +971,6 @@ You can factor out child components purely as a way of reusing rendering logic. For more information, see [Reuse rendering logic](xref:blazor/performance#define-reusable-renderfragments-in-code). -## Attribute splatting and arbitrary parameters - -Components can capture and render additional attributes in addition to the component's declared parameters. Additional attributes can be captured in a dictionary and then *splatted* onto an element when the component is rendered using the [`@attributes`][3] Razor directive attribute. This scenario is useful for defining a component that produces a markup element that supports a variety of customizations. For example, it can be tedious to define attributes separately for an `` that supports many parameters. - -In the following `Splat` component: - -* The first `` element (`id="useIndividualParams"`) uses individual component parameters. -* The second `` element (`id="useAttributesDict"`) uses attribute splatting. - -`Pages/Splat.razor`: - -:::moniker range=">= aspnetcore-7.0" - -:::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Pages/index/Splat.razor"::: - -:::moniker-end - -:::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" - -:::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Pages/index/Splat.razor"::: - -:::moniker-end - -:::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" - -:::code language="razor" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/Pages/index/Splat.razor"::: - -:::moniker-end - -:::moniker range="< aspnetcore-5.0" - -:::code language="razor" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/Pages/index/Splat.razor"::: - -:::moniker-end - -The rendered `` elements in the webpage are identical: - -```html - - - -``` - -To accept arbitrary attributes, define a [component parameter](#component-parameters) with the property set to `true`: - -```razor -@code { - [Parameter(CaptureUnmatchedValues = true)] - public Dictionary? InputAttributes { get; set; } -} -``` - -The property on [`[Parameter]`](xref:Microsoft.AspNetCore.Components.ParameterAttribute) allows the parameter to match all attributes that don't match any other parameter. A component can only define a single parameter with . The property type used with must be assignable from [`Dictionary`](xref:System.Collections.Generic.Dictionary%602) with string keys. Use of [`IEnumerable>`](xref:System.Collections.Generic.IEnumerable%601) or [`IReadOnlyDictionary`](xref:System.Collections.Generic.IReadOnlyDictionary%602) are also options in this scenario. - -The position of [`@attributes`][3] relative to the position of element attributes is important. When [`@attributes`][3] are splatted on the element, the attributes are processed from right to left (last to first). Consider the following example of a parent component that consumes a child component: - -`Shared/AttributeOrderChild1.razor`: - -:::moniker range=">= aspnetcore-7.0" - -:::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild1.razor"::: - -:::moniker-end - -:::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" - -:::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild1.razor"::: - -:::moniker-end - -:::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" - -:::code language="razor" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild1.razor"::: - -:::moniker-end - -:::moniker range="< aspnetcore-5.0" - -:::code language="razor" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild1.razor"::: - -:::moniker-end - -`Pages/AttributeOrderParent1.razor`: - -:::moniker range=">= aspnetcore-7.0" - -:::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent1.razor"::: - -:::moniker-end - -:::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" - -:::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent1.razor"::: - -:::moniker-end - -:::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" - -:::code language="razor" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent1.razor"::: - -:::moniker-end - -:::moniker range="< aspnetcore-5.0" - -:::code language="razor" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent1.razor"::: - -:::moniker-end - -The `AttributeOrderChild1` component's `extra` attribute is set to the right of [`@attributes`][3]. The `AttributeOrderParent1` component's rendered `
` contains `extra="5"` when passed through the additional attribute because the attributes are processed right to left (last to first): - -```html -
-``` - -In the following example, the order of `extra` and [`@attributes`][3] is reversed in the child component's `
`: - -`Shared/AttributeOrderChild2.razor`: - -:::moniker range=">= aspnetcore-7.0" - -:::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild2.razor"::: - -:::moniker-end - -:::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" - -:::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild2.razor"::: - -:::moniker-end - -:::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" - -:::code language="razor" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild2.razor"::: - -:::moniker-end - -:::moniker range="< aspnetcore-5.0" - -:::code language="razor" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild2.razor"::: - -:::moniker-end - -`Pages/AttributeOrderParent2.razor`: - -:::moniker range=">= aspnetcore-7.0" - -:::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent2.razor"::: - -:::moniker-end - -:::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" - -:::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent2.razor"::: - -:::moniker-end - -:::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" - -:::code language="razor" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent2.razor"::: - -:::moniker-end - -:::moniker range="< aspnetcore-5.0" - -:::code language="razor" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent2.razor"::: - -:::moniker-end - -The `
` in the parent component's rendered webpage contains `extra="10"` when passed through the additional attribute: - -```html -
-``` - ## Capture references to components Component references provide a way to reference a component instance for issuing commands. To capture a component reference: diff --git a/aspnetcore/blazor/components/splat-attributes-and-arbitrary-parameters.md b/aspnetcore/blazor/components/splat-attributes-and-arbitrary-parameters.md new file mode 100644 index 0000000000..48b03bfce7 --- /dev/null +++ b/aspnetcore/blazor/components/splat-attributes-and-arbitrary-parameters.md @@ -0,0 +1,195 @@ +--- +title: ASP.NET Core Blazor attribute splatting and arbitrary parameters +author: guardrex +description: Learn how components can capture and render additional attributes in addition to the component's declared parameters. +monikerRange: '>= aspnetcore-3.1' +ms.author: riande +ms.custom: mvc +ms.date: 05/16/2023 +uid: blazor/components/attribute-splatting +--- +# ASP.NET Core Blazor attribute splatting and arbitrary parameters + +Components can capture and render additional attributes in addition to the component's declared parameters. Additional attributes can be captured in a dictionary and then *splatted* onto an element when the component is rendered using the Razor directive attribute. This scenario is useful for defining a component that produces a markup element that supports a variety of customizations. For example, it can be tedious to define attributes separately for an `` that supports many parameters. + +## Attribute splatting + +In the following `Splat` component: + +* The first `` element (`id="useIndividualParams"`) uses individual component parameters. +* The second `` element (`id="useAttributesDict"`) uses attribute splatting. + +`Pages/Splat.razor`: + +:::moniker range=">= aspnetcore-7.0" + +:::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Pages/index/Splat.razor"::: + +:::moniker-end + +:::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" + +:::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Pages/index/Splat.razor"::: + +:::moniker-end + +:::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" + +:::code language="razor" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/Pages/index/Splat.razor"::: + +:::moniker-end + +:::moniker range="< aspnetcore-5.0" + +:::code language="razor" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/Pages/index/Splat.razor"::: + +:::moniker-end + +The rendered `` elements in the webpage are identical: + +```html + + + +``` + +## Arbitrary attributes + +To accept arbitrary attributes, define a [component parameter](xref:blazor/components/index#component-parameters) with the property set to `true`: + +```razor +@code { + [Parameter(CaptureUnmatchedValues = true)] + public Dictionary? InputAttributes { get; set; } +} +``` + +The property on [`[Parameter]`](xref:Microsoft.AspNetCore.Components.ParameterAttribute) allows the parameter to match all attributes that don't match any other parameter. A component can only define a single parameter with . The property type used with must be assignable from [`Dictionary`](xref:System.Collections.Generic.Dictionary%602) with string keys. Use of [`IEnumerable>`](xref:System.Collections.Generic.IEnumerable%601) or [`IReadOnlyDictionary`](xref:System.Collections.Generic.IReadOnlyDictionary%602) are also options in this scenario. + +The position of relative to the position of element attributes is important. When are splatted on the element, the attributes are processed from right to left (last to first). Consider the following example of a parent component that consumes a child component: + +`Shared/AttributeOrderChild1.razor`: + +:::moniker range=">= aspnetcore-7.0" + +:::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild1.razor"::: + +:::moniker-end + +:::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" + +:::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild1.razor"::: + +:::moniker-end + +:::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" + +:::code language="razor" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild1.razor"::: + +:::moniker-end + +:::moniker range="< aspnetcore-5.0" + +:::code language="razor" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild1.razor"::: + +:::moniker-end + +`Pages/AttributeOrderParent1.razor`: + +:::moniker range=">= aspnetcore-7.0" + +:::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent1.razor"::: + +:::moniker-end + +:::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" + +:::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent1.razor"::: + +:::moniker-end + +:::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" + +:::code language="razor" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent1.razor"::: + +:::moniker-end + +:::moniker range="< aspnetcore-5.0" + +:::code language="razor" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent1.razor"::: + +:::moniker-end + +The `AttributeOrderChild1` component's `extra` attribute is set to the right of . The `AttributeOrderParent1` component's rendered `
` contains `extra="5"` when passed through the additional attribute because the attributes are processed right to left (last to first): + +```html +
+``` + +In the following example, the order of `extra` and is reversed in the child component's `
`: + +`Shared/AttributeOrderChild2.razor`: + +:::moniker range=">= aspnetcore-7.0" + +:::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild2.razor"::: + +:::moniker-end + +:::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" + +:::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild2.razor"::: + +:::moniker-end + +:::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" + +:::code language="razor" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild2.razor"::: + +:::moniker-end + +:::moniker range="< aspnetcore-5.0" + +:::code language="razor" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/Shared/index/AttributeOrderChild2.razor"::: + +:::moniker-end + +`Pages/AttributeOrderParent2.razor`: + +:::moniker range=">= aspnetcore-7.0" + +:::code language="razor" source="~/../blazor-samples/7.0/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent2.razor"::: + +:::moniker-end + +:::moniker range=">= aspnetcore-6.0 < aspnetcore-7.0" + +:::code language="razor" source="~/../blazor-samples/6.0/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent2.razor"::: + +:::moniker-end + +:::moniker range=">= aspnetcore-5.0 < aspnetcore-6.0" + +:::code language="razor" source="~/../blazor-samples/5.0/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent2.razor"::: + +:::moniker-end + +:::moniker range="< aspnetcore-5.0" + +:::code language="razor" source="~/../blazor-samples/3.1/BlazorSample_WebAssembly/Pages/index/AttributeOrderParent2.razor"::: + +:::moniker-end + +The `
` in the parent component's rendered webpage contains `extra="10"` when passed through the additional attribute: + +```html +
+``` diff --git a/aspnetcore/blazor/performance.md b/aspnetcore/blazor/performance.md index 66e2a26f63..619cc290b0 100644 --- a/aspnetcore/blazor/performance.md +++ b/aspnetcore/blazor/performance.md @@ -420,7 +420,7 @@ This approach allows passing arbitrary additional attributes to the element. How Use where component rendering performance isn't critical, such as components that aren't repeated frequently. For components that render at scale, such as each item in a large list or in the cells of a grid, try to avoid attribute splatting. -For more information, see . +For more information, see . #### Implement `SetParametersAsync` manually diff --git a/aspnetcore/toc.yml b/aspnetcore/toc.yml index 1123c44538..11675fc84c 100644 --- a/aspnetcore/toc.yml +++ b/aspnetcore/toc.yml @@ -439,6 +439,8 @@ items: uid: blazor/components/key - name: Overwriting parameters uid: blazor/components/overwriting-parameters + - name: Attribute splatting and arbitrary parameters + uid: blazor/components/attribute-splatting - name: Layouts uid: blazor/components/layouts - name: Sections