Маршрутизация. Контроллеры. Представления. Хелперы

Содержание

Слайд 2

Маршрутизация

За сопоставление запросов с конкретными адресами внутри приложения в ASP.NET Core отвечает

Маршрутизация За сопоставление запросов с конкретными адресами внутри приложения в ASP.NET Core
система маршрутизации.

Маршрутизация в MVC это процесс соответствия входящего запроса и обработчика запроса. Обработчиками запросов в MVC обычно выступают действия контроллера.

Запрос поступающий к серверу ASP.NET Core - это запрос к файлу или к маршруту.
Чтобы включить запрос к статическим файлам нужно добавить соответствующее промежуточное ПО в конвейер запросов. По умолчанию любой шаблон приложения в Visual Studio прописывает этот метод расширения в методе Configure.

app.UseStaticFiles();

Слайд 3

Если это middleware не включено в конвейер обработки запроса или запрашиваемого файла

Если это middleware не включено в конвейер обработки запроса или запрашиваемого файла
не существует, то при запросе система сразу будет искать обработчик маршрута. Типичный обработчик маршрута – это контроллер.

Для направления запросов на соответствующие действия контроллера в MVC существует таблица маршрутизации. Добавление маршрута в таблицу маршрутизации называется регистрацией маршрута. 

Есть два API, которые можно использовать для регистрации маршрутов:
Fluent API;
атрибуты маршрутизации.

MVC поддерживает 2 типа маршрутизации: 
на основе соглашений (Convention-Based Routing);
на основе атрибутов (Attribute-Based Routing).
Маршрутизация к действиям должна быть добавлена в конвейер обработки запросов.

Слайд 4

Чтобы задействовать систему маршрутизации на основе конечных точек в приложении MVC, надо:

Чтобы задействовать систему маршрутизации на основе конечных точек в приложении MVC, надо:

добавить в классе Startup в методе  ConfigureServices()  сервисы MVC
с помощью соответствующего middleware в методе  Configure() определить как минимум один маршрут

В MVC можно использовать два подхода к определению системы маршрутизации:
Определение и использование конечных точек с помощью компонентов EndpointRoutingMiddleware и EndpointMiddleware (применяется по умолчанию при создании нового проекта MVC).
Использование RouterMiddleware и метода UseMvc. 

Слайд 5

app.UseRouting();
app.UseAuthorization();
app.UseEndpoints(endpoints =>
{
endpoints.MapControllerRoute(
name: "default",
pattern: "{controller=Home}/{action=Index}/{id?}");
});

Добавляет

app.UseRouting(); app.UseAuthorization(); app.UseEndpoints(endpoints => { endpoints.MapControllerRoute( name: "default", pattern: "{controller=Home}/{action=Index}/{id?}"); }); Добавляет
в конвейер обработки запроса компонент EndpointRoutingMiddleware.

Компонент EndpointRoutingMiddleware позволяет определить маршрут, который соответствует запрошенному адресу, установить для его обработки конечную точку в виде объекта Microsoft.AspNetCore.Http.Endpoint, а также определить данные маршрута.

Встраивает в конвейер обработки запроса компонент EndpointMiddleware.

Компонент EndpointMiddleware определяет набор конечных точек, которые будут сопоставляться с определенными маршрутами и будут обрабатывать соответствующие маршрутам входящие запросы.

Слайд 6

Метод app.UseEndpoints в качестве параметра принимает делегат Action с параметром  IEndpointRouteBuilder, который добавляет конечные

Метод app.UseEndpoints в качестве параметра принимает делегат Action с параметром IEndpointRouteBuilder, который
точки.

app.UseEndpoints(endpoints =>
{
endpoints.MapControllerRoute(
name: "default",
pattern: "{controller=Home}/{action=Index}/{id?}");
});

Шаблоны маршрутов

Шаблоны маршрутизации используют литералы и маркеры (для параметров маршрутизации). Литералы соответствуют точно тексту в URL, тогда как маркеры заменяются при сопоставлении маршрута.

public/{controller=Home}/{action=Index}/{id?}

Слайд 7

Чтобы соответствовать шаблону, URL должен содержать маркеры контроллера и действия, так как

Чтобы соответствовать шаблону, URL должен содержать маркеры контроллера и действия, так как
это ключевая информация, которую MVC использует для поиска контроллера/действия. Другие маркеры в URL-адресе сопоставляются с параметрами методов действий с помощью привязки модели.

При использовании шаблона маршрута:  
public/{controller=Home}/{action=Index}/{id?}
следующие URL сработают:

public/Home/Index/2 Имеется полное соответствие. 

public/Home/Index Тоже полное соответствие, id объявлен необязательным.

Слайд 8

public/Home Неполное соответствие. У третьего сегмента шаблона есть значение по умолчанию – Index.

public У

public/Home Неполное соответствие. У третьего сегмента шаблона есть значение по умолчанию –
второго и третьего сегмента есть значения по умолчанию

Методы IEndpointRouteBuilder для добавления маршрутов:

MapControllerRoute(
string name, string pattern, [object defaults = null],
[object constraints = null], [object dataTokens = null]
)  определяет произвольный маршрут .
name: название машрута
pattern: шаблон машрута
defaults: значения параметров маршрутов по умолчанию
constraints: ограничения маршрута
dataTokens: определения токенов маршрута

Слайд 9

MapDefaultControllerRoute() определяет стандартный маршрут, эквивалентен вызову

endpoints.MapControllerRoute(
name: "default",
pattern: "{controller=Home}/{action=Index}/{id?}");

MapAreaControllerRoute(string name, string areaName,

MapDefaultControllerRoute() определяет стандартный маршрут, эквивалентен вызову endpoints.MapControllerRoute( name: "default", pattern: "{controller=Home}/{action=Index}/{id?}"); MapAreaControllerRoute(string

string pattern, [object defaults = null],
[object constraints = null], [object dataTokens = null]
) определяет маршрут, который также учитывает область приложения. 

MapControllers() сопоставляет действия контроллера с запросами, используя маршрутизацию на основе атрибутов. 

MapFallbackToController(string action, string controller) определяет действие контроллера, которое будет обрабатывать запрос, если все остальные определенные маршруты не соответствуют запросу. 

Слайд 10

 MapGet() добавляет конечную точку для определенного маршрута по запросу типа GET и ее

MapGet() добавляет конечную точку для определенного маршрута по запросу типа GET и
обработчик.

app.UseEndpoints(endpoints =>
{
endpoints.MapGet("/Hello", async context =>
{
await context.Response.WriteAsync("

Hello, Vasja!

");
});
endpoints.MapControllerRoute(
name: "default",
pattern: "{controller=Home}/{action=Index}/{id?}");
});

Слайд 11

Маршрутизация с помощью RouterMiddleware

Это старая форма маршрутизации. Была в Core 2.

Необходимо

Маршрутизация с помощью RouterMiddleware Это старая форма маршрутизации. Была в Core 2.
при подключении сервисов MVC в методе ConfigureServices явным образом отключить использование конечных точек.
Все методы для подключения сервисов MVC в качестве параметра могут принимать делегат Action. С помощью свойства EnableEndpointRouting объекта MvcOptions можно отключить использование конечных точек:

services.AddControllersWithViews(mvcOtions=>
    {
        mvcOtions.EnableEndpointRouting = false;
    });

Слайд 12

app.UseMvc(routes =>
{
    routes.MapRoute("api", "api/get", new { controller = "Home", action = "About" });
routes.MapRoute(
        name:

app.UseMvc(routes => { routes.MapRoute("api", "api/get", new { controller = "Home", action =
"default",
        template: "{controller=Home}/{action=Index}/{id?}");
});

Для определения маршрутов в методе Configure применяется метод UseMvc(): 

При составлении таблиц маршрутизации важно, чтобы самые специфичные были зарегистрированы первыми, поскольку, как только будет найдено первое соответствие входящему URL поиск в таблице завершится.

Слайд 13

Маршрутизация на основе атрибутов

Маршруты, определенные с помощью атрибутов, имеют приоритет по сравнению

Маршрутизация на основе атрибутов Маршруты, определенные с помощью атрибутов, имеют приоритет по
с маршрутами, определенными в классе Startup.

Для определения маршрута необходимо использовать атрибут [Route]. В качестве параметра атрибут Route принимает шаблон URL, с которым будет сопоставляться запрошенный адрес. Этот атрибут применяется к действиям контроллера.

[Route("Trepatnja")]
[HttpGet]
public ActionResult Comments()
{
var comments = CommentsDb.Comments;
return View(comments);
}

Слайд 14

Если в проекте планируется использовать только маршрутизацию на основе атрибутов, то в

Если в проекте планируется использовать только маршрутизацию на основе атрибутов, то в
классе Startup можно не определять никаких маршрутов. Например, при использовании EndpointMiddleware и конечных точек можно применить метод MapControllers.

Слайд 15

Контроллеры

Контроллер представляет обычный класс, который наследуется от базового класса Microsoft.AspNetCore.Mvc.Controller.
В свою очередь

Контроллеры Контроллер представляет обычный класс, который наследуется от базового класса Microsoft.AspNetCore.Mvc.Controller. В
класс Controller реализует абстрактный базовый класс ControllerBase.
Таким образом, чтобы создать свой класс контроллера, достаточно создать класс, наследующий от Controller и имеющий в имени суффикс Controller.

Методы действий (action methods) представляют такие методы контроллера, которые обрабатывают запросы по определенному URL. 

Слайд 16

Так как запросы бывают разных типов, например, GET и POST, ASP.NET MVC

Так как запросы бывают разных типов, например, GET и POST, ASP.NET MVC
позволяет определить тип обрабатываемого запроса для действия, применив к нему соответствующий атрибут: [HttpGet], [HttpPost], [HttpDelete] или [HttpPut].
http://datanets.ru/tipy-zaprosov-http-protokola-get-post-put.html
https://habr.com/post/50147/
Не все методы контроллера являются методами действий. Методы действий всегда имеют модификатор public. Закрытых приватных методов действий не бывает.
Но контроллер может также включать и обычные методы, которые могут использоваться в вспомогательных целях.

Для контроллеров предназначена папка Controllers. По умолчанию при создании проекта в нее добавляется контроллер HomeController.

Слайд 17

Чтобы указать, что класс не является контроллером (т.е. не может обрабатывать запрос),

Чтобы указать, что класс не является контроллером (т.е. не может обрабатывать запрос),
нам надо использовать над ним атрибут [NonController]

Если нужно, чтобы какой-то public метод контроллера не рассматривался как действие, то можно применить к нему атрибут [NonAction].

Атрибут [ActionName] позволяет для метода задать другое имя действия. 

[HttpPost]
[ActionName("InsertComments")]
public ActionResult AddComments()
{
Comment comment = new Comment();
comment.Nik = Request.Form["Nik"];
comment.Text = Request.Form["Text"];
comment.Time = DateTime.Now;
CommentsDb.Comments.Add(comment);
return RedirectToAction("Comments");
}

Слайд 18

Результаты действий
https://metanit.com/sharp/aspnet5/5.3.php

Метод контроллера формирует некоторый ответ в виде результата действия, как правило,

Результаты действий https://metanit.com/sharp/aspnet5/5.3.php Метод контроллера формирует некоторый ответ в виде результата действия,
это объект класса, реализующего интерфейс  IActionResult.

 Интерфейс IActionResult находится в пространстве имен Microsoft.AspNetCore.Mvc

Этот интерфейс реализуется абстрактным базовым классом ActionResult. В нем есть синхронный метод ExecuteResult, который выполняется в асинхронном.

public interface IActionResult
{
    Task ExecuteResultAsync(ActionContext context);
}

Слайд 19

Для создания своего класса результата действий нужно либо унаследовать его от ActionResult,

Для создания своего класса результата действий нужно либо унаследовать его от ActionResult,
либо реализовать интерфейс IActionResult.

public class MyActionResult:ActionResult
{
private string text;
private string action;
public MyActionResult(string text, string action)
{
this.text = text;
this.action = action;
}

public override async Task ExecuteResultAsync(ActionContext context)
{
string fullCode = "";
fullCode += "Суперпроект";
fullCode += "";
fullCode += " ";
fullCode += "

" + text + "

";
fullCode += "Куда-то";
fullCode += "";
await context.HttpContext.Response.WriteAsync(fullCode);
}

Слайд 20

Метод действия в контроллере:

Стандартные результаты действий:
ContentResult: пишет указанный контент напрямую в

Метод действия в контроллере: Стандартные результаты действий: ContentResult: пишет указанный контент напрямую
ответ в виде строки

[Route("/")]
public IActionResult GetResult()
{
return new MyActionResult("Привет, Вася!", "Index");
}

EmptyResult: отправляет пустой ответ в виде статусного кода 200

FileResult: является базовым классом для всех объектов, которые пишут набор байтов в выходной поток. Предназначен для отправки файлов
И т.д. https://metanit.com/sharp/aspnet5/5.3.php

Слайд 21

Например,

ViewResult: производит рендеринг представления и отправляет результаты рендеринга в виде html-страницы клиенту

Чтобы

Например, ViewResult: производит рендеринг представления и отправляет результаты рендеринга в виде html-страницы
возвратить объект ViewResult используется метод View

return View();

По умолчанию контроллер производит поиск представления в проекте по следующему пути:
 /Views/Имя_контроллера/Имя_представления.cshtml

public IActionResult Index()
{
return NotFound("Ничего не найдено");
}

Слайд 22

В качестве представления будет использоваться то, имя которого совпадает с именем действия.

Можно

В качестве представления будет использоваться то, имя которого совпадает с именем действия.
также задать имя представления явным образом:
return View("Index");

Можно полностью переопределить путь, по которому система будет искать представление:
return View("~/Views/Some/Index.cshtml");

Передать данные из контроллера в представление можно, используя объекты ViewData и ViewBag.

ViewBag.User = "Вася";
ViewData["User"] = "Вася";

В представлении
@Html.TextBox("User",ViewData["User"])

Так нельзя: @Html.TextBox("User",ViewBag.User)

Слайд 23

Так можно: @Html.TextBox("User",(string)ViewBag.User)

RedirectResult: перенаправляет пользователя по другому адресу URL

Для временной переадресации применяется

Так можно: @Html.TextBox("User",(string)ViewBag.User) RedirectResult: перенаправляет пользователя по другому адресу URL Для временной
метод Redirect:
return Redirect("/Home/Index");

RedirectToRouteResult: перенаправляет пользователя по определенному адресу URL, указанному через параметры маршрута

Метод RedirectToAction позволяет перейти к определенному действию определенного контроллера, а также позволяет задать передаваемые параметры:

Слайд 24

Например,
return RedirectToAction("Index");
return RedirectToAction("Square", "Home", new { a=10,h=12});

HttpNotFoundResult: возвращает клиенту ответ

Например, return RedirectToAction("Index"); return RedirectToAction("Square", "Home", new { a=10,h=12}); HttpNotFoundResult: возвращает клиенту
в виде статусного кода HTTP 404, указывая, что запрошенный ресурс не найден.

public ActionResult Buy(int? id)
{
if (id == null)
return NotFound();
ViewBag.ProductId = id;
ViewBag.Title = "Оформление покупки "+Request.Params["item"];
return View();
}

Слайд 25

Передача данных в контроллер

При отправке GET-запроса значения передаются через строку запроса.
Стандартный

Передача данных в контроллер При отправке GET-запроса значения передаются через строку запроса.
get-запрос:
 название_ресурса?параметр1=значение1&параметр2=значение2
Например, определим следующий метод:

public IActionResult GetDollars(double? rubles,double? rate)
{
if(rubles!=null && rate!=null)
{
double dollars = rubles.Value / rate.Value;
return Content(
$" {rubles} руб. = {dollars} $ по курсу {rate}");
}
return StatusCode(401);
}

Запрос этого действия:
https://localhost:44305/Home/GetDollars?rubles=1000&rate=2.5

Слайд 26

Передача в действие контроллера объекта класса

public IActionResult ShowStudent(Student student)
{
return Content($"Студент

Передача в действие контроллера объекта класса public IActionResult ShowStudent(Student student) { return
{student.Lastname} {student.Name} получил по ИГИ {student.MarkIGI}");
}

Слайд 27

Передача данных в запросе POST

Чтобы система могла связать параметры метода и данные

Передача данных в запросе POST Чтобы система могла связать параметры метода и
формы, необходимо, чтобы атрибуты name у полей формы соответствовали названиям параметров.
















Слайд 28

Данные в действии можно получать, не используя параметры, через свойство Request контроллера.

Данные в действии можно получать, не используя параметры, через свойство Request контроллера.

public ActionResult AddComments()
{
Comment comment = new Comment();
comment.Nik = Request.Form["Nik"];
comment.Text = Request.Form["Text"];
comment.Time = DateTime.Now;
CommentsDb.Comments.Add(comment);
return RedirectToAction("Comments");
}

comment.Nik = Request.Form.FirstOrDefault(p=>p.Key=="Nik").Value;

Слайд 29

Представления

Представление не является html-страницей. При компиляции приложения на основе требуемого представления сначала

Представления Представление не является html-страницей. При компиляции приложения на основе требуемого представления
генерируется класс на языке C#, наследующий от класса Microsoft.AspNetCore.Mvc.Razor.RazorPage
, где T - это класс модели, которая будет использоваться, или ключевое слово dynamic. Затем этот класс компилируется. 
https://docs.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.mvc.razor.razorpage-1?view=aspnetcore-3.1

Все добавляемые представления, как правило, группируются по контроллерам в соответствующие папки в каталоге Views.

Для рендеринга представления в выходной поток, используется метод View().

Контроллер готовит данные и выбирает представление, а затем объект ViewResult обращается к движку представления Razor для рендеринга представления в выходной результат.
https://habr.com/post/98241/

Слайд 30

Cпособы передачи данных из контроллера в представление:
ViewData
ViewBag
Модель представления

ViewData представляет словарь из пар ключ-значение.

ViewData["Title"]=

Cпособы передачи данных из контроллера в представление: ViewData ViewBag Модель представления ViewData
"Шимпанзе - не обезьяны";

В качестве значения может выступать любой объект.

ViewBag позволяет определить различные свойства и присвоить им любое значение.

ViewBag.Title = "Шимпанзе - не обезьяны";

Слайд 31

Строго типизированные представления

Позволяют передавать данные не через объект ViewBag и т.п., а

Строго типизированные представления Позволяют передавать данные не через объект ViewBag и т.п.,
напрямую в представление через параметр метода View. 

Чтобы связать представление с передаваемым параметром, надо добавить в представление директиву  @model  с указанием типа передаваемых данных

Объект Model представляет тип модели и будет содержать переданные из контроллера данные

Слайд 32

@using MvcSimple.Models;
@model IEnumerable









@using MvcSimple.Models; @model IEnumerable Товар Категория Цена Выбор @foreach(var b in Model)
@foreach(var b in Model)
{




@*адрес, по которому будет размещаться форма оформления покупки*@


}

Товар

Категория

Цена

Выбор

@b.Name

@b.ProductCategory

@b.Price

Купить



Слайд 33

public ActionResult Index()
{
// получаем из бд данные
ViewBag.Title = "Магазин";

public ActionResult Index() { // получаем из бд данные ViewBag.Title = "Магазин";
var products = db.Products;
// возвращаем представление
return View(products);
}

Можно автоматически создавать строго типизированное представление, указав в диалоговом окне при создании представления соответствующие параметры.

Слайд 34

Мастер-страница может иметь несколько секций, куда представления могут поместить свое содержимое.

Добавим к

Мастер-страница может иметь несколько секций, куда представления могут поместить свое содержимое. Добавим
мастер-странице секцию footer


@Html.ActionLink("Главная", "Index", "Home")
@RenderBody()


@RenderSection("Footer", false)


В представлении

@section Footer{
Разаработчик Вася
}

Слайд 35

Код страницы _ViewStart.cshtml выполняется до кода любого из представлений, расположенных в том

Код страницы _ViewStart.cshtml выполняется до кода любого из представлений, расположенных в том
же каталоге. Этот файл последовательно применяется к каждому представлению, находящемуся в одном каталоге.

HTML-хелперы

Позволяют генерировать html-код.

  Для создания хелпера в проекте нужно создать папку под названием App_Code. Это специальное зарезервированное имя для папки, которая содержит html-хелперы, поэтому нужно использовать данное название.

Хелперы представляют собой методы расширения, поэтому в папку App_Code для создания хелперов нужно добавить статический класс.

Слайд 36

public static class ListHelper
{
public static HtmlString CreateList(this IHtmlHelper html, string[]

public static class ListHelper { public static HtmlString CreateList(this IHtmlHelper html, string[]
items)
{
string result = "
    ";
    foreach (string item in items)
    {
    result = $"{result}
  • {item}
  • ";
    }
    result = $"{result}
";
return new HtmlString(result);
}
}

Пример использования:

@{
ViewData["Title"] = "Home Page";
string[] students = { "Вася", "Петя", "Маша", "Даша" };
}
@using WebApplication1.App_Code
@Html.CreateList(students)

Слайд 37

Встроенные html-хелперы

Встроенный хелпер BeginForm позволяет создать форму

Принимает в качестве параметров имя метода действия и

Встроенные html-хелперы Встроенный хелпер BeginForm позволяет создать форму Принимает в качестве параметров
имя контроллера, а также тип запроса

@using(Html.BeginForm("Buy","Home",FormMethod.Post))
{
...
}

Хелпер Html.TextBox генерирует тег input со значением атрибута type равным text. Перегружен 7 раз.

@Html.TextBox("Address")

Html.Hidden генерирует тег input type="hidden"

@Html.Hidden("ProductId",(int) ViewBag.ProductId)

Один из параметров хелперов позволяет задать атрибуты создаваемого элемента формы, в том числе и классы css.

Слайд 38

@Html.TextBox("Text", "Наберите текст комментария",
new { style = "width:350px;height:100px",
maxlength

@Html.TextBox("Text", "Наберите текст комментария", new { style = "width:350px;height:100px", maxlength = 250,
= 250, minlengh = 5, required = true })

Этот тег генерирует тег:


Html.Password
Html.RadioButton
Html.CheckBox
Html.Label
Html.DropDownList
Html.ListBox

Слайд 39

Строго типизированные хелперы

Принимают в качестве параметра лямбда-выражение, в котором указывается то свойство

Строго типизированные хелперы Принимают в качестве параметра лямбда-выражение, в котором указывается то
модели, к которому должен быть привязан данный хелпер.
Могут использоваться только в строго типизированных представлениях

Тип модели, которая передается в хелпер, должен быть тем же самым, что указан в директиве @model.

Для каждого базового встроенного хелпера имеется свой строго типизированный хелпер, только в имени добавлен суффикс For.

Слайд 40

@using MvcSimple.Models
@model Purchase


@using(Html.BeginForm("Buy","Home",FormMethod.Post))
{
@Html.HiddenFor(m=>m.ProductId,(int) ViewBag.ProductId)










Введите@using MvcSimple.Models @model Purchase @using(Html.BeginForm("Buy","Home",FormMethod.Post)) { @Html.HiddenFor(m=>m.ProductId,(int) ViewBag.ProductId) Введите имя @Html.TextBoxFor(m=>m.Client) Введите

имя

@Html.TextBoxFor(m=>m.Client)

Введите адрес :

@Html.TextBoxFor(m => m.Address)

}

Слайд 41

Шаблонные хелперы

Не генерируют определенный элемент html.
Генерируют тот элемент html, который наиболее

Шаблонные хелперы Не генерируют определенный элемент html. Генерируют тот элемент html, который
подходит данному свойству, исходя из его типа и метаданных.

Display (DisplayFor)
Создает элемент разметки для отображения значения указанного свойства модели  
Html.Display("Name") (Html.DisplayFor(m => m.Name))
Editor
Создает элемент разметки для редактирования указанного свойства модели: Html.Editor("Name")

Слайд 42

DisplayForModel
Создает поля для чтения для всех свойств модели: 
Html.DisplayForModel()
EditorForModel
Создает поля для редактирования для

DisplayForModel Создает поля для чтения для всех свойств модели: Html.DisplayForModel() EditorForModel Создает
всех свойств модели:
Html.EditorForModel()

Для хелпера можно задавать в качестве параметра имя шаблона для отображения. Для этого в папке с представлениями нужно создать подпапку DisplayTemplates. В эту папку нужно добавить частичное представление с моделью, тип которой совпадает с типом параметра в шаблонном хелпере.

Слайд 43

Например, создадим шаблон для отображения списка строк

@* Шаблон для списка строк *@
@model

Например, создадим шаблон для отображения списка строк @* Шаблон для списка строк
List

    @foreach (var item in Model)
    {
  • @item

  • }

Тогда имя этого шаблона можно указывать в хелпере:

@Html.DisplayFor(m => m.Readers, "LisstTemplate")

Также шаблон можно указать прямо в определении модели, используя атрибут UIHint

[UIHint("LisstTemplate")]
public List Readers { get; set; }

Слайд 44

Хелпер ActionLink создает гиперссылку на действие контроллера. Если создается ссылка на действие, определенное в

Хелпер ActionLink создает гиперссылку на действие контроллера. Если создается ссылка на действие,
том же контроллере, можно указать только имя действия.

Если нужно создать ссылку на действие из другого контроллера, то в хелпере ActionLink в качестве третьего аргумента указывается имя контроллера.

Параметры для действия передаются через параметр routeValues (можно в виде объекта анонимного класса).

Также можно установить значения атрибутов элемента HTML через параметр htmlAttributes.

@Html.ActionLink("Перейти к комментариям", "Comments",
"Home", new { header = Model.Title },
new { @class = "btn btn-primary btn-large" })

Слайд 45

Tag-хелперы

Tag-хелперы – это специальные элементы в представлении, выглядят как обычные html-элементы или

Tag-хелперы Tag-хелперы – это специальные элементы в представлении, выглядят как обычные html-элементы
атрибуты, но при работе приложения они обрабатываются движком Razor на стороне сервера и преобразуются в стандартные html-элементы.
Например,

asp-route-header="@Model.Title"
class="btn btn-primary btn-large">
Перейти к комментариям

Слайд 46

ScriptTagHelper применяется для подключения внешних файлов скриптов. 

Атрибуты:
asp-append-version: если true, то к пути

ScriptTagHelper применяется для подключения внешних файлов скриптов. Атрибуты: asp-append-version: если true, то
к файлу скрипта добавляется номер версии
asp-fallback-src: указывает вспомогательный путь к скрипту, если загрузка скрипта, указанного в атрибуте src пройдет неудачно
asp-fallback-test: определяет выражение, которое тестирует загрузку основного скрипта из атрибута src
asp-src-include: определяет шаблон подключаемых файлов, через запятую можно задать несколько шаблонов
asp-src-exclude: определяет через запятую набор шаблонов для тех файлов, которые следует исключить из загрузки
asp-fallback-src-include: определяет набор шаблонов файлов, которые подключаются, если загрузка основного скрипта из атрибута src прошла неудачно
asp-fallback-src-exclude: определяет набор шаблонов файлов, которые следует исключить из загрузки, если загрузка основного скрипта из атрибута src прошла неудачно

Слайд 47

Символы подстановки в шаблонах :
? любой одиночный символ кроме слеша.
 js/script?.js будет соответствовать

Символы подстановки в шаблонах : ? любой одиночный символ кроме слеша. js/script?.js
 js/script1.js или js/scriptX.js, но не js/script35.js
* любое количество символов кроме слеша.
 js/*.js будет соответствовать  js/script.js или js/scriptX25.js, но не  js/bootstrap/script.js
** любое количество символов, в том числе и слеш.
 js/**/script.js будет соответствовать  js/script.js или  js/bootstrap/script.js, но не  js/script35.js


src="~/lib/jquery-ajax-unobtrusive/jquery.unobtrusive-ajax.js">

Слайд 61

@model List
@foreach (var comment in Model)
{



@comment.Nik


@comment.Time


@model List @foreach (var comment in Model) { @comment.Nik @comment.Time @comment.Text }


@comment.Text




}

Если есть частичное представление GetComments

его можно встроить в представление с помощью вызова метода Html.PartialAsync() 

@await Html.PartialAsync("GetComments",Model.Comments)

Слайд 62

За рендеринг частичных представлений отвечает объект PartialViewResult, который возвращается методом PartialView. 

asp-action="InsertComments"
method="post"

За рендеринг частичных представлений отвечает объект PartialViewResult, который возвращается методом PartialView. asp-action="InsertComments"
data-ajax="true"
data-ajax-update="#com"
data-ajax-mode="replace"
data-ajax-method="post">
…..

[HttpPost]
[ActionName("InsertComments")]
public ActionResult AddComments(string title)
{
Comment comment = new Comment();
comment.Nik = Request.Form.FirstOrDefault(p=>p.Key=="Nik").Value;
comment.Text = Request.Form["Text"];
comment.Time = DateTime.Now;
CommentsDb.Comments.Add(comment);
return RedirectToAction("GetComments");
}

Слайд 63

public ActionResult GetComments()
{
return PartialView(CommentsDb.Comments);
}

Валидация на стороне клиента

Для валидации модели

public ActionResult GetComments() { return PartialView(CommentsDb.Comments); } Валидация на стороне клиента Для
на стороне клиента нужно добавить скрипты


Это уже есть в файле _ValidationScriptsPartial.cshtml Поэтому вместо прямого подключения скриптов можно подключать данное частичное представление. Перед подключением скриптов валидации должно идти подключение jquery.

Слайд 64

При объявлении модели нужно использовать атрибуты валидации.
В представлении нужно использовать tag-хелперы

При объявлении модели нужно использовать атрибуты валидации. В представлении нужно использовать tag-хелперы
валидации.
При каждом свойстве нужно использовать хелпер валидации 
(ValidationMessageTagHelper)

public class LoginModel
{
[Required]
[Display(Name = "Логин")]
public string UserName { get; set; }
[Required]
[DataType(DataType.Password)]
[Display(Name = "Пароль")]
public string Password { get; set; }
[Display(Name = "Запомнить?")]
public bool RememberMe { get; set; }
public string ReturnUrl { get; set; }
}

Слайд 65





Для отображения сводки ошибок валидации применяется ValidationSummaryTagHelper. Он применяется к элементу div в виде атрибута asp-validation-summary







Для отображения сводки ошибок валидации применяется ValidationSummaryTagHelper. Он применяется к элементу div в виде атрибута asp-validation-summary

Слайд 66

В качестве значения атрибут asp-validation-summary принимает одно из значений перечисления ValidationSummary:
None: ошибки валидации не отображаются
ModelOnly:

В качестве значения атрибут asp-validation-summary принимает одно из значений перечисления ValidationSummary: None:
отображаются только ошибка валидации уровня модели, ошибки валидации для отдельных свойств не отображаются
All: отображаются все ошибки валидации

Слайд 67

Атрибуты:

RegularExpression
[RegularExpression (@"[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\.[A-Za-z]{2,4}",
ErrorMessage = "Некорректный адрес")]
public string Email { get; set; }

Required

StringLength

Хелпер

Атрибуты: RegularExpression [RegularExpression (@"[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\.[A-Za-z]{2,4}", ErrorMessage = "Некорректный адрес")] public string Email {
валидации Html.ValidationMessageFor для свойств модели

@Html.EditorFor(m => m.Client)
@Html.ValidationMessageFor(m => m.Client)

Для отображения сводки сообщений об ошибках при валидации модели можно использовать хелпер Html.ValidationSummary

Слайд 68

Range
[Range(1700,2000,ErrorMessage="Недопустимый год")]
public int Year { get; set; }
[Range(typeof(decimal), "0.00", "49.99")]
public decimal Price

Range [Range(1700,2000,ErrorMessage="Недопустимый год")] public int Year { get; set; } [Range(typeof(decimal), "0.00",
{ get; set; }

Compare
[DataType(DataType.Password)]
public string Password { get; set; }
[Compare("Password",ErrorMessage="Пароли не совпадают")]
[DataType(DataType.Password)]
public  string PasswordConfirm { get; set; }

Слайд 69

Для проверки на корректность электронной почты определены специальные атрибуты:
[CreditCard]
[EmailAddress]
[Phone]
[Url]

Remote
Для валидации свойства выполняет

Для проверки на корректность электронной почты определены специальные атрибуты: [CreditCard] [EmailAddress] [Phone]
запрос на сервер к определенному методу контроллера. И если требуемый метод контроллера вернет значение false, то валидация не пройдена. Например:

[Remote(action: "CheckEmail", controller: "Home",
ErrorMessage ="Email уже используется")]
public string Email { get; set; }

Слайд 70

[AcceptVerbs("Get", "Post")]
    public IActionResult CheckEmail(string email)
    {
        if (email == "admin@mail.ru" || email == "aaa@gmail.com")
            return

[AcceptVerbs("Get", "Post")] public IActionResult CheckEmail(string email) { if (email == "admin@mail.ru" ||
Json(false);
        return Json(true);
    }

Слайд 71

Все атрибуты валидации образованы от базового класса ValidationAttribute, который находится в пространстве имен System.ComponentModel.DataAnnotations

Можно

Все атрибуты валидации образованы от базового класса ValidationAttribute, который находится в пространстве
создавать свои атрибуты для свойств и для модели.

public class PurschaseAllowedAttribute:ValidationAttribute
{
public override bool IsValid(object value)
{
Purchase p = value as Purchase;
if (p.Client == "Вася" && p.Address.Contains("Гомель"))
{
return false;
}
return true;
}
}

Слайд 72

[PurschaseAllowed(ErrorMessage = "Вася из Гомеля покупать не может")]
public class Purchase
{

[PurschaseAllowed(ErrorMessage = "Вася из Гомеля покупать не может")] public class Purchase {
// ID покупки
public int Id { get; set; }
// имя и фамилия покупателя
[StringLength(50, MinimumLength = 3, ErrorMessage = "Мало букв")]
[Required(ErrorMessage = "Введите свое имя")]
public string Client { get; set; }
// адрес покупателя
public string Address { get; set; }
// ID книги
public int ProductId { get; set; }
// дата покупки
public DateTime Date { get; set; }
}

Слайд 73

Валидация на стороне сервера

Если в браузере отключен javascript , то сообщение с

Валидация на стороне сервера Если в браузере отключен javascript , то сообщение
ошибками не отображается, а форма отправляется на сервер.

Узнать, проходит модель валидацию или нет, можно с помощью свойства ModelState.IsValid

[HttpPost]
public ActionResult Buy(Purchase purchase)
{
purchase.Date = DateTime.Now;
if (ModelState.IsValid)
{
db.Purchases.Add(purchase);
db.SaveChanges();
return new SubmitResult(" Спасибо за покупку ",
"Info?ProductId=" + purchase.ProductId);
}
return new SubmitResult(" Ошибка ", "Index" );
}

Слайд 74

Объект ModelState сохраняет все значения, которые пользователь ввел для свойств модели, а

Объект ModelState сохраняет все значения, которые пользователь ввел для свойств модели, а
также все ошибки, связанные с каждым свойством и с моделью в целом. 

Метод ModelState.AddModelError добавляет для свойства, указанного в качестве первого параметра, ошибку, сообщение для которой указано в качестве второго параметра.

if (purchase.Client == "Вася" && purchase.Address.Length < 3)
ModelState.AddModelError("", "Вася с коротким адресом!");//ошибки
//на уровне модели


Слайд 76

Внедрение зависимостей в контроллер

Это можно делать следующими способами:
Через конструктор
Через параметр метода, к

Внедрение зависимостей в контроллер Это можно делать следующими способами: Через конструктор Через
которому применяется атрибут FromServices
Через свойство HttpContext.RequestServices

Добавление новых классов в контейнер происходит в методе ConfigureServices() класса Startup. Для этого у IServiceCollection существует метод 
Add(ServiceDescriptor serviceDescriptor).

Класс ServiceDescriptor описывает параметры зависимости и содержит следующие свойства:
ServiceLifetime Lifetime – Время жизни экземпляра класса:

Слайд 77

Transient – Новый экземпляр будет создан при каждом обращении к контейнеру.
Scoped – Новый экземпляр

Transient – Новый экземпляр будет создан при каждом обращении к контейнеру. Scoped
будет создан для каждого запроса к серверу. При этом в рамках самого запроса всегда будет передаваться один и тоже экземпляр.
Singleton – Всегда будет использоваться один и тоже экземпляр данного типа

Type ServiceType – Тип регистрируемого сервиса, как правило интерфейса.

Один из вариантов, указывающих на способ создания реализации ServiceType:
Type ImplementationType – Тип реализации. Будет создан в контейнере при обращении.
object ImplementationInstance – Созданный заранее объект типа ServiceType (только для Singleton);
public Func ImplementationFactory – Фабричный метод для создания экземпляра типа ServiceType.

Имя файла: Маршрутизация.-Контроллеры.-Представления.-Хелперы.pptx
Количество просмотров: 27
Количество скачиваний: 0