AspNetCore.Docs/aspnetcore/mvc/views/layout.md

154 lines
8.1 KiB
Markdown
Raw Normal View History

2016-10-29 01:35:15 +08:00
---
title: Layout in ASP.NET Core
2016-11-12 00:28:14 +08:00
author: ardalis
description: Learn how to use common layouts, share directives, and run common code before rendering views in an ASP.NET Core app.
2018-01-29 23:21:31 +08:00
ms.author: riande
2019-07-30 18:47:06 +08:00
ms.date: 07/30/2019
2016-10-29 01:35:15 +08:00
uid: mvc/views/layout
---
# Layout in ASP.NET Core
2016-10-29 01:35:15 +08:00
By [Steve Smith](https://ardalis.com/) and [Dave Brock](https://twitter.com/daveabrock)
2016-10-29 01:35:15 +08:00
Pages and views frequently share visual and programmatic elements. This article demonstrates how to:
* Use common layouts.
* Share directives.
* Run common code before rendering pages or views.
This document discusses layouts for the two different approaches to ASP.NET Core MVC: Razor Pages and controllers with views. For this topic, the differences are minimal:
* Razor Pages are in the *Pages* folder.
* Controllers with views uses a *Views* folder for views.
2016-10-29 01:35:15 +08:00
## What is a Layout
Most web apps have a common layout that provides the user with a consistent experience as they navigate from page to page. The layout typically includes common user interface elements such as the app header, navigation or menu elements, and footer.
![Page Layout example](layout/_static/page-layout.png)
2016-10-29 01:35:15 +08:00
Common HTML structures such as scripts and stylesheets are also frequently used by many pages within an app. All of these shared elements may be defined in a *layout* file, which can then be referenced by any view used within the app. Layouts reduce duplicate code in views.
2016-10-29 01:35:15 +08:00
By convention, the default layout for an ASP.NET Core app is named `_Layout.cshtml`. The layout files for new ASP.NET Core projects created with the templates are:
* Razor Pages: `Pages/Shared/_Layout.cshtml`
2016-10-29 01:35:15 +08:00
2019-07-30 18:47:06 +08:00
![Pages folder in Solution Explorer](layout/_static/rp-web-project-views.png)
2016-10-29 01:35:15 +08:00
* Controller with views: `Views/Shared/_Layout.cshtml`
2016-10-29 01:35:15 +08:00
2019-07-30 18:47:06 +08:00
![Views folder in Solution Explorer](layout/_static/mvc-web-project-views.png)
2016-10-29 01:35:15 +08:00
The layout defines a top level template for views in the app. Apps don't require a layout. Apps can define more than one layout, with different views specifying different layouts.
The following code shows the layout file for a template created project with a controller and views:
[!code-cshtml[](~/common/samples/WebApplication1/Views/Shared/_Layout.cshtml?highlight=44,72)]
2016-10-29 01:35:15 +08:00
## Specifying a Layout
Razor views have a `Layout` property. Individual views specify a layout by setting this property:
[!code-cshtml[](../../common/samples/WebApplication1/Views/_ViewStart.cshtml?highlight=2)]
2016-10-29 01:35:15 +08:00
The layout specified can use a full path (for example, `/Pages/Shared/_Layout.cshtml` or `/Views/Shared/_Layout.cshtml`) or a partial name (example: `_Layout`). When a partial name is provided, the Razor view engine searches for the layout file using its standard discovery process. The folder where the handler method (or controller) exists is searched first, followed by the *Shared* folder. This discovery process is identical to the process used to discover [partial views](xref:mvc/views/partial#partial-view-discovery).
2016-10-29 01:35:15 +08:00
By default, every layout must call `RenderBody`. Wherever the call to `RenderBody` is placed, the contents of the view will be rendered.
2017-10-14 04:50:30 +08:00
<a name="layout-sections-label"></a>
2019-08-31 08:09:30 +08:00
<!-- https://stackoverflow.com/questions/23327578 -->
2016-10-29 01:35:15 +08:00
### Sections
A layout can optionally reference one or more *sections*, by calling `RenderSection`. Sections provide a way to organize where certain page elements should be placed. Each call to `RenderSection` can specify whether that section is required or optional:
```html
<script type="text/javascript" src="~/scripts/global.js"></script>
@RenderSection("Scripts", required: false)
```
2016-10-29 01:35:15 +08:00
If a required section isn't found, an exception is thrown. Individual views specify the content to be rendered within a section using the `@section` Razor syntax. If a page or view defines a section, it must be rendered (or an error will occur).
An example `@section` definition in Razor Pages view:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```html
2016-10-29 01:35:15 +08:00
@section Scripts {
<script type="text/javascript" src="~/scripts/main.js"></script>
}
```
In the preceding code, `scripts/main.js` is added to the `scripts` section on a page or view. Other pages or views in the same app might not require this script and wouldn't define a scripts section.
The following markup uses the [Partial Tag Helper](xref:mvc/views/tag-helpers/builtin-th/partial-tag-helper) to render `_ValidationScriptsPartial.cshtml`:
```html
@section Scripts {
<partial name="_ValidationScriptsPartial" />
}
```
2016-10-29 01:35:15 +08:00
The preceding markup was generated by [scaffolding Identity](xref:security/authentication/scaffold-identity).
2016-10-29 01:35:15 +08:00
Sections defined in a page or view are available only in its immediate layout page. They cannot be referenced from partials, view components, or other parts of the view system.
2016-10-29 01:35:15 +08:00
### Ignoring sections
By default, the body and all sections in a content page must all be rendered by the layout page. The Razor view engine enforces this by tracking whether the body and each section have been rendered.
To instruct the view engine to ignore the body or sections, call the `IgnoreBody` and `IgnoreSection` methods.
2016-10-29 01:35:15 +08:00
The body and every section in a Razor page must be either rendered or ignored.
2017-10-14 04:50:30 +08:00
<a name="viewimports"></a>
2016-10-29 01:35:15 +08:00
## Importing Shared Directives
Views and pages can use Razor directives to import namespaces and use [dependency injection](dependency-injection.md). Directives shared by many views may be specified in a common `_ViewImports.cshtml` file. The `_ViewImports` file supports the following directives:
2016-10-29 01:35:15 +08:00
* `@addTagHelper`
* `@removeTagHelper`
* `@tagHelperPrefix`
* `@using`
* `@model`
* `@inherits`
* `@inject`
2021-11-15 09:48:14 +08:00
* `@namespace`
2016-10-29 01:35:15 +08:00
The file doesn't support other Razor features, such as functions and section definitions.
2016-10-29 01:35:15 +08:00
A sample `_ViewImports.cshtml` file:
[!code-cshtml[](../../common/samples/WebApplication1/Views/_ViewImports.cshtml)]
2016-10-29 01:35:15 +08:00
The `_ViewImports.cshtml` file for an ASP.NET Core MVC app is typically placed in the *Pages* (or *Views*) folder. A `_ViewImports.cshtml` file can be placed within any folder, in which case it will only be applied to pages or views within that folder and its subfolders. `_ViewImports` files are processed starting at the root level and then for each folder leading up to the location of the page or view itself. `_ViewImports` settings specified at the root level may be overridden at the folder level.
2016-10-29 01:35:15 +08:00
For example, suppose:
2016-10-29 01:35:15 +08:00
* The root level `_ViewImports.cshtml` file includes `@model MyModel1` and `@addTagHelper *, MyTagHelper1`.
* A subfolder `_ViewImports.cshtml` file includes `@model MyModel2` and `@addTagHelper *, MyTagHelper2`.
2016-10-29 01:35:15 +08:00
Pages and views in the subfolder will have access to both Tag Helpers and the `MyModel2` model.
2016-10-29 01:35:15 +08:00
If multiple `_ViewImports.cshtml` files are found in the file hierarchy, the combined behavior of the directives are:
2016-10-29 01:35:15 +08:00
* `@addTagHelper`, `@removeTagHelper`: all run, in order
* `@tagHelperPrefix`: the closest one to the view overrides any others
2016-10-29 01:35:15 +08:00
* `@model`: the closest one to the view overrides any others
* `@inherits`: the closest one to the view overrides any others
* `@using`: all are included; duplicates are ignored
* `@inject`: for each property, the closest one to the view overrides any others with the same property name
2017-10-14 04:50:30 +08:00
<a name="viewstart"></a>
2016-10-29 01:35:15 +08:00
## Running Code Before Each View
Code that needs to run before each view or page should be placed in the `_ViewStart.cshtml` file. By convention, the `_ViewStart.cshtml` file is located in the *Pages* (or *Views*) folder. The statements listed in `_ViewStart.cshtml` are run before every full view (not layouts, and not partial views). Like [ViewImports.cshtml](xref:mvc/views/layout#viewimports), `_ViewStart.cshtml` is hierarchical. If a `_ViewStart.cshtml` file is defined in the view or pages folder, it will be run after the one defined in the root of the *Pages* (or *Views*) folder (if any).
2016-10-29 01:35:15 +08:00
A sample `_ViewStart.cshtml` file:
2016-10-29 01:35:15 +08:00
[!code-cshtml[](../../common/samples/WebApplication1/Views/_ViewStart.cshtml)]
2016-10-29 01:35:15 +08:00
The file above specifies that all views will use the `_Layout.cshtml` layout.
2016-10-29 01:35:15 +08:00
`_ViewStart.cshtml` and `_ViewImports.cshtml` are **not** typically placed in the */Pages/Shared* (or */Views/Shared*) folder. The app-level versions of these files should be placed directly in the */Pages* (or */Views*) folder.