AspNetCore.Docs/aspnetcore/mvc/views/view-components.md

344 lines
18 KiB
Markdown
Raw Normal View History

---
title: View components in ASP.NET Core
2016-10-29 01:35:15 +08:00
author: rick-anderson
description: Learn how view components are used in ASP.NET Core and how to add them to apps.
2018-01-29 23:21:31 +08:00
ms.author: riande
ms.custom: mvc
ms.date: 12/18/2019
2020-10-28 05:27:47 +08:00
no-loc: [appsettings.json, "ASP.NET Core Identity", cookie, Cookie, Blazor, "Blazor Server", "Blazor WebAssembly", "Identity", "Let's Encrypt", Razor, SignalR]
uid: mvc/views/view-components
2016-10-29 01:35:15 +08:00
---
# View components in ASP.NET Core
2016-10-29 01:35:15 +08:00
By [Rick Anderson](https://twitter.com/RickAndMSFT)
[View or download sample code](https://github.com/dotnet/AspNetCore.Docs/tree/master/aspnetcore/mvc/views/view-components/sample) ([how to download](xref:index#how-to-download-a-sample))
2016-10-29 01:35:15 +08:00
2018-05-10 02:40:55 +08:00
## View components
2016-10-29 01:35:15 +08:00
2019-01-23 02:02:55 +08:00
View components are similar to partial views, but they're much more powerful. View components don't use model binding, and only depend on the data provided when calling into it. This article was written using controllers and views, but view components also work with Razor Pages.
2018-05-10 02:40:55 +08:00
A view component:
2016-10-29 01:35:15 +08:00
2018-01-29 05:49:15 +08:00
* Renders a chunk rather than a whole response.
* Includes the same separation-of-concerns and testability benefits found between a controller and view.
* Can have parameters and business logic.
* Is typically invoked from a layout page.
View components are intended anywhere you have reusable rendering logic that's too complex for a partial view, such as:
2016-10-29 01:35:15 +08:00
* Dynamic navigation menus
* Tag cloud (where it queries the database)
* Login panel
* Shopping cart
* Recently published articles
* Sidebar content on a typical blog
* A login panel that would be rendered on every page and show either the links to log out or log in, depending on the log in state of the user
A view component consists of two parts: the class (typically derived from [ViewComponent](/dotnet/api/microsoft.aspnetcore.mvc.viewcomponent)) and the result it returns (typically a view). Like controllers, a view component can be a POCO, but most developers will want to take advantage of the methods and properties available by deriving from `ViewComponent`.
2016-10-29 01:35:15 +08:00
When considering if view components meet an app's specifications, consider using Razor components instead. Razor components also combine markup with C# code to produce reusable UI units. Razor components are designed for developer productivity when providing client-side UI logic and composition. For more information, see <xref:blazor/components/index>. For information on how to incorporate Razor components into an MVC or Razor Pages app, see <xref:blazor/components/prerendering-and-integration?pivots=server>.
2016-10-29 01:35:15 +08:00
## Creating a view component
This section contains the high-level requirements to create a view component. Later in the article, we'll examine each step in detail and create a view component.
2016-10-29 01:35:15 +08:00
### The view component class
A view component class can be created by any of the following:
* Deriving from *ViewComponent*
* Decorating a class with the `[ViewComponent]` attribute, or deriving from a class with the `[ViewComponent]` attribute
* Creating a class where the name ends with the suffix *ViewComponent*
Like controllers, view components must be public, non-nested, and non-abstract classes. The view component name is the class name with the "ViewComponent" suffix removed. It can also be explicitly specified using the `ViewComponentAttribute.Name` property.
2016-10-29 01:35:15 +08:00
A view component class:
* Fully supports constructor [dependency injection](../../fundamentals/dependency-injection.md)
* Doesn't take part in the controller lifecycle, which means you can't use [filters](../controllers/filters.md) in a view component
2016-10-29 01:35:15 +08:00
### View component methods
A view component defines its logic in an `InvokeAsync` method that returns a `Task<IViewComponentResult>` or in a synchronous `Invoke` method that returns an `IViewComponentResult`. Parameters come directly from invocation of the view component, not from model binding. A view component never directly handles a request. Typically, a view component initializes a model and passes it to a view by calling the `View` method. In summary, view component methods:
2016-10-29 01:35:15 +08:00
* Define an `InvokeAsync` method that returns a `Task<IViewComponentResult>` or a synchronous `Invoke` method that returns an `IViewComponentResult`.
* Typically initializes a model and passes it to a view by calling the `ViewComponent` `View` method.
* Parameters come from the calling method, not HTTP. There's no model binding.
* Are not reachable directly as an HTTP endpoint. They're invoked from your code (usually in a view). A view component never handles a request.
* Are overloaded on the signature rather than any details from the current HTTP request.
2016-10-29 01:35:15 +08:00
### View search path
The runtime searches for the view in the following paths:
* /Views/{Controller Name}/Components/{View Component Name}/{View Name}
* /Views/Shared/Components/{View Component Name}/{View Name}
2019-01-31 05:20:55 +08:00
* /Pages/Shared/Components/{View Component Name}/{View Name}
The search path applies to projects using controllers + views and Razor Pages.
2016-10-29 01:35:15 +08:00
The default view name for a view component is *Default*, which means your view file will typically be named *Default.cshtml*. You can specify a different view name when creating the view component result or when calling the `View` method.
We recommend you name the view file *Default.cshtml* and use the *Views/Shared/Components/{View Component Name}/{View Name}* path. The `PriorityList` view component used in this sample uses *Views/Shared/Components/PriorityList/Default.cshtml* for the view component view.
2016-10-29 01:35:15 +08:00
### Customize the view search path
To customize the view search path, modify Razor's <xref:Microsoft.AspNetCore.Mvc.Razor.RazorViewEngineOptions.ViewLocationFormats> collection. For example, to search for views within the path "/Components/{View Component Name}/{View Name}", add a new item to the collection:
2020-07-10 06:22:19 +08:00
[!code-csharp[](view-components/samples_snapshot/2.x/Startup.cs?name=snippet_ViewLocationFormats&highlight=4)]
In the preceding code, the placeholder "{0}" represents the path "Components/{View Component Name}/{View Name}".
2016-10-29 01:35:15 +08:00
## Invoking a view component
2017-04-05 05:16:54 +08:00
To use the view component, call the following inside a view:
2017-11-01 00:10:09 +08:00
```cshtml
@await Component.InvokeAsync("Name of view component", {Anonymous Type Containing Parameters})
2017-04-05 05:16:54 +08:00
```
The parameters will be passed to the `InvokeAsync` method. The `PriorityList` view component developed in the article is invoked from the *Views/ToDo/Index.cshtml* view file. In the following, the `InvokeAsync` method is called with two parameters:
2016-10-29 01:35:15 +08:00
[!code-cshtml[](view-components/sample/ViewCompFinal/Views/ToDo/IndexFinal.cshtml?range=35)]
::: moniker range=">= aspnetcore-1.1"
## Invoking a view component as a Tag Helper
For ASP.NET Core 1.1 and higher, you can invoke a view component as a [Tag Helper](xref:mvc/views/tag-helpers/intro):
[!code-cshtml[](view-components/sample/ViewCompFinal/Views/ToDo/IndexTagHelper.cshtml?range=37-38)]
2019-01-28 21:18:41 +08:00
Pascal-cased class and method parameters for Tag Helpers are translated into their [kebab case](https://stackoverflow.com/questions/11273282/whats-the-name-for-dash-separated-case/12273101). The Tag Helper to invoke a view component uses the `<vc></vc>` element. The view component is specified as follows:
2017-11-01 00:10:09 +08:00
```cshtml
<vc:[view-component-name]
parameter1="parameter1 value"
parameter2="parameter2 value">
</vc:[view-component-name]>
```
To use a view component as a Tag Helper, register the assembly containing the view component using the `@addTagHelper` directive. If your view component is in an assembly called `MyWebApp`, add the following directive to the *_ViewImports.cshtml* file:
2017-11-01 00:10:09 +08:00
```cshtml
@addTagHelper *, MyWebApp
```
You can register a view component as a Tag Helper to any file that references the view component. See [Managing Tag Helper Scope](xref:mvc/views/tag-helpers/intro#managing-tag-helper-scope) for more information on how to register Tag Helpers.
The `InvokeAsync` method used in this tutorial:
[!code-cshtml[](view-components/sample/ViewCompFinal/Views/ToDo/IndexFinal.cshtml?range=35)]
In Tag Helper markup:
[!code-cshtml[](view-components/sample/ViewCompFinal/Views/ToDo/IndexTagHelper.cshtml?range=37-38)]
2019-01-28 21:18:41 +08:00
In the sample above, the `PriorityList` view component becomes `priority-list`. The parameters to the view component are passed as attributes in kebab case.
2016-10-29 01:35:15 +08:00
::: moniker-end
2016-10-29 01:35:15 +08:00
### Invoking a view component directly from a controller
View components are typically invoked from a view, but you can invoke them directly from a controller method. While view components don't define endpoints like controllers, you can easily implement a controller action that returns the content of a `ViewComponentResult`.
2016-10-29 01:35:15 +08:00
In this example, the view component is called directly from the controller:
[!code-csharp[](view-components/sample/ViewCompFinal/Controllers/ToDoController.cs?name=snippet_IndexVC)]
2016-10-29 01:35:15 +08:00
## Walkthrough: Creating a simple view component
[Download](https://github.com/dotnet/AspNetCore.Docs/tree/master/aspnetcore/mvc/views/view-components/sample), build and test the starter code. It's a simple project with a `ToDo` controller that displays a list of *ToDo* items.
2016-10-29 01:35:15 +08:00
![List of ToDos](view-components/_static/2dos.png)
2016-10-29 01:35:15 +08:00
### Add a ViewComponent class
Create a *ViewComponents* folder and add the following `PriorityListViewComponent` class:
2016-10-29 01:35:15 +08:00
[!code-csharp[](view-components/sample/ViewCompFinal/ViewComponents/PriorityListViewComponent1.cs?name=snippet1)]
2016-10-29 01:35:15 +08:00
Notes on the code:
* View component classes can be contained in **any** folder in the project.
2016-12-02 04:38:32 +08:00
* Because the class name PriorityList**ViewComponent** ends with the suffix **ViewComponent**, the runtime will use the string "PriorityList" when referencing the class component from a view. I'll explain that in more detail later.
* The `[ViewComponent]` attribute can change the name used to reference a view component. For example, we could've named the class `XYZ` and applied the `ViewComponent` attribute:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2016-10-29 01:35:15 +08:00
[ViewComponent(Name = "PriorityList")]
public class XYZ : ViewComponent
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
* The `[ViewComponent]` attribute above tells the view component selector to use the name `PriorityList` when looking for the views associated with the component, and to use the string "PriorityList" when referencing the class component from a view. I'll explain that in more detail later.
* The component uses [dependency injection](../../fundamentals/dependency-injection.md) to make the data context available.
* `InvokeAsync` exposes a method which can be called from a view, and it can take an arbitrary number of arguments.
* The `InvokeAsync` method returns the set of `ToDo` items that satisfy the `isDone` and `maxPriority` parameters.
2016-10-29 01:35:15 +08:00
### Create the view component Razor view
2016-12-02 04:38:32 +08:00
* Create the *Views/Shared/Components* folder. This folder **must** be named *Components*.
2016-10-29 01:35:15 +08:00
2016-12-02 04:38:32 +08:00
* Create the *Views/Shared/Components/PriorityList* folder. This folder name must match the name of the view component class, or the name of the class minus the suffix (if we followed convention and used the *ViewComponent* suffix in the class name). If you used the `ViewComponent` attribute, the class name would need to match the attribute designation.
2016-10-29 01:35:15 +08:00
2016-12-02 04:38:32 +08:00
* Create a *Views/Shared/Components/PriorityList/Default.cshtml* Razor view:
[!code-cshtml[](view-components/sample/ViewCompFinal/Views/Shared/Components/PriorityList/Default1.cshtml)]
2016-10-29 01:35:15 +08:00
The Razor view takes a list of `TodoItem` and displays them. If the view component `InvokeAsync` method doesn't pass the name of the view (as in our sample), *Default* is used for the view name by convention. Later in the tutorial, I'll show you how to pass the name of the view. To override the default styling for a specific controller, add a view to the controller-specific view folder (for example *Views/ToDo/Components/PriorityList/Default.cshtml)*.
2016-10-29 01:35:15 +08:00
If the view component is controller-specific, you can add it to the controller-specific folder (*Views/ToDo/Components/PriorityList/Default.cshtml*).
* Add a `div` containing a call to the priority list component to the bottom of the *Views/ToDo/index.cshtml* file:
[!code-cshtml[](view-components/sample/ViewCompFinal/Views/ToDo/IndexFirst.cshtml?range=34-38)]
2016-10-29 01:35:15 +08:00
2016-12-02 04:38:32 +08:00
The markup `@await Component.InvokeAsync` shows the syntax for calling view components. The first argument is the name of the component we want to invoke or call. Subsequent parameters are passed to the component. `InvokeAsync` can take an arbitrary number of arguments.
2016-10-29 01:35:15 +08:00
2016-12-02 04:38:32 +08:00
Test the app. The following image shows the ToDo list and the priority items:
2016-10-29 01:35:15 +08:00
2016-12-02 04:38:32 +08:00
![todo list and priority items](view-components/_static/pi.png)
2016-10-29 01:35:15 +08:00
You can also call the view component directly from the controller:
[!code-csharp[](view-components/sample/ViewCompFinal/Controllers/ToDoController.cs?name=snippet_IndexVC)]
2016-12-02 04:38:32 +08:00
![priority items from IndexVC action](view-components/_static/indexvc.png)
2016-10-29 01:35:15 +08:00
### Specifying a view name
A complex view component might need to specify a non-default view under some conditions. The following code shows how to specify the "PVC" view from the `InvokeAsync` method. Update the `InvokeAsync` method in the `PriorityListViewComponent` class.
[!code-csharp[](../../mvc/views/view-components/sample/ViewCompFinal/ViewComponents/PriorityListViewComponentFinal.cs?highlight=4,5,6,7,8,9&range=28-39)]
2016-10-29 01:35:15 +08:00
Copy the *Views/Shared/Components/PriorityList/Default.cshtml* file to a view named *Views/Shared/Components/PriorityList/PVC.cshtml*. Add a heading to indicate the PVC view is being used.
[!code-cshtml[](../../mvc/views/view-components/sample/ViewCompFinal/Views/Shared/Components/PriorityList/PVC.cshtml?highlight=3)]
2016-10-29 01:35:15 +08:00
Update *Views/ToDo/Index.cshtml*:
2016-10-29 01:35:15 +08:00
<!-- Views/ToDo/Index.cshtml is never imported, so change to test tutorial -->
2016-12-02 04:38:32 +08:00
[!code-cshtml[](view-components/sample/ViewCompFinal/Views/ToDo/IndexFinal.cshtml?range=35)]
2016-10-29 01:35:15 +08:00
Run the app and verify PVC view.
![Priority View Component](view-components/_static/pvc.png)
2016-10-29 01:35:15 +08:00
If the PVC view isn't rendered, verify you are calling the view component with a priority of 4 or higher.
2016-10-29 01:35:15 +08:00
### Examine the view path
* Change the priority parameter to three or less so the priority view isn't returned.
* Temporarily rename the *Views/ToDo/Components/PriorityList/Default.cshtml* to *1Default.cshtml*.
2016-12-02 04:38:32 +08:00
* Test the app, you'll get the following error:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```
2017-04-05 05:16:54 +08:00
An unhandled exception occurred while processing the request.
InvalidOperationException: The view 'Components/PriorityList/Default' wasn't found. The following locations were searched:
2017-04-05 05:16:54 +08:00
/Views/ToDo/Components/PriorityList/Default.cshtml
/Views/Shared/Components/PriorityList/Default.cshtml
EnsureSuccessful
```
* Copy *Views/ToDo/Components/PriorityList/1Default.cshtml* to *Views/Shared/Components/PriorityList/Default.cshtml*.
* Add some markup to the *Shared* ToDo view component view to indicate the view is from the *Shared* folder.
2016-12-02 04:38:32 +08:00
* Test the **Shared** component view.
2016-10-29 01:35:15 +08:00
2016-12-02 04:38:32 +08:00
![ToDo output with Shared component view](view-components/_static/shared.png)
2016-10-29 01:35:15 +08:00
### Avoiding hard-coded strings
2016-10-29 01:35:15 +08:00
If you want compile time safety, you can replace the hard-coded view component name with the class name. Create the view component without the "ViewComponent" suffix:
2016-10-29 01:35:15 +08:00
[!code-csharp[](../../mvc/views/view-components/sample/ViewCompFinal/ViewComponents/PriorityList.cs?highlight=10&range=5-35)]
2016-10-29 01:35:15 +08:00
Add a `using` statement to your Razor view file, and use the `nameof` operator:
2016-10-29 01:35:15 +08:00
[!code-cshtml[](view-components/sample/ViewCompFinal/Views/ToDo/IndexNameof.cshtml?range=1-6,35-)]
2016-10-29 01:35:15 +08:00
## Perform synchronous work
The framework handles invoking a synchronous `Invoke` method if you don't need to perform asynchronous work. The following method creates a synchronous `Invoke` view component:
```csharp
public class PriorityList : ViewComponent
{
public IViewComponentResult Invoke(int maxPriority, bool isDone)
{
var items = new List<string> { $"maxPriority: {maxPriority}", $"isDone: {isDone}" };
return View(items);
}
}
```
The view component's Razor file lists the strings passed to the `Invoke` method (*Views/Home/Components/PriorityList/Default.cshtml*):
```cshtml
@model List<string>
<h3>Priority Items</h3>
<ul>
@foreach (var item in Model)
{
<li>@item</li>
}
</ul>
```
::: moniker range=">= aspnetcore-1.1"
The view component is invoked in a Razor file (for example, *Views/Home/Index.cshtml*) using one of the following approaches:
* <xref:Microsoft.AspNetCore.Mvc.IViewComponentHelper>
* [Tag Helper](xref:mvc/views/tag-helpers/intro)
To use the <xref:Microsoft.AspNetCore.Mvc.IViewComponentHelper> approach, call `Component.InvokeAsync`:
::: moniker-end
::: moniker range="< aspnetcore-1.1"
The view component is invoked in a Razor file (for example, *Views/Home/Index.cshtml*) with <xref:Microsoft.AspNetCore.Mvc.IViewComponentHelper>.
Call `Component.InvokeAsync`:
::: moniker-end
```cshtml
@await Component.InvokeAsync(nameof(PriorityList), new { maxPriority = 4, isDone = true })
```
::: moniker range=">= aspnetcore-1.1"
To use the Tag Helper, register the assembly containing the View Component using the `@addTagHelper` directive (the view component is in an assembly called `MyWebApp`):
```cshtml
@addTagHelper *, MyWebApp
```
Use the view component Tag Helper in the Razor markup file:
```cshtml
<vc:priority-list max-priority="999" is-done="false">
</vc:priority-list>
```
::: moniker-end
The method signature of `PriorityList.Invoke` is synchronous, but Razor finds and calls the method with `Component.InvokeAsync` in the markup file.
## All view component parameters are required
Each parameter in a view component is a required attribute. See [this GitHub issue](https://github.com/dotnet/AspNetCore/issues/5011). If any parameter is omitted:
* The `InvokeAsync` method signature won't match, therefore the method won't execute.
* The ViewComponent won't render any markup.
* No errors will be thrown.
## Additional resources
2016-10-29 01:35:15 +08:00
* [Dependency injection into views](xref:mvc/views/dependency-injection)