Литмир - Электронная Библиотека

Создание своих вариантов ActionResult — это исключительно мощное средство для расширения базовой функциональности MVC Framework. Реализуя свои экземпляры ActionResult, вы сможете генерировать ответ на клиентский запрос в любой форме с любой структурой данных.

Asp.net mvc framework - img_32

Model Binding

После того как механизм MVC Framework получил запрос от клиента с набором некоторых параметров, произвел определение необходимого контроллера и действия, возникает задача сопоставления параметров запроса параметрам выбранного действия контроллера. Задача решается просто, когда параметров немного. В этом случае сопоставление параметров происходит по их наименованию: параметрам метода действия с определенным именем присваиваются значения параметров запроса с теми же именами. В качестве примера рассмотрим действие Update контроллера AdminController. Во фрагменте приведено определение метода с параметрами:

public ActionResult Update(Guid? userid, string email,

      string comment, bool isApproved, bool isLockedOut)

Для этого действия подразумевается, что при его вызове будут переданы параметры с именами: userid, email, comment, isApproved, isLockedOut. Такие параметры передаются с запросом при отправлении формы с нашего представления Select. В следующем фрагменте рассмотрим основной HTML-код формы этого представления, который отображается в браузере пользователя:

<form action="/Admin/Update" method="post">

...

  <input id="userId" name="userId" type="hidden"

      value="a4530eee-8634-4258-ac00-0ea63f7cc783" />

...

  <input id="email" name="email" type="text" value="[email protected]" />

...

  <textarea cols="20" id="comment" name="comment" rows="2">

    <b&gt,Добрый день!</b>

  </textarea>

...

  <input checked="checked" id="isApproved"

      name="isApproved" type="checkbox" value="true" />

...

  <input id="isLockedOut" name="isLockedOut"

      type="checkbox" value="true" />

...

  <input type="submit" value="Coxpaнить" />

</form>

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

Но что делать, когда форма содержит десятки вводимых полей? Неужели создавать десятки параметров у метода действия контроллера? Нет, MVC Framework содержит механизм, который позволяет избежать такого некрасивого шага, как многочисленные параметры метода. Такой механизм называется Model Binding (привязка модели). Чтобы продемонстрировать работу этого механизма, выполним ряд изменений в коде. Для начала определим комплексный тип, который будет содержать все необходимые данные, передаваемые в действие Update:

public class UserData

{

  public Guid? UserId { get; set; }

  public string Email { get; set; }

  public string Comment { get; set; }

  public bool IsApproved { get; set; }

  public bool IsLockedOut { get; set; }

}

Обратите внимание, что для определения параметров мы используем свойства. Механизм Model Binding требует, чтобы использовались свойства, но не простые поля. Соответственно данному типу изменим определение метода Update:

public ActionResult Update(UserData userData)

{

  if (!userData.UserId.HasValue)

    throw new HttpException(404, "Пользователь не найден");

  MembershipProvider mp = Membership.Provider;

  MembershipUser user = mp.GetUser(userData.UserId, false);

  user.Email = userData.Email;

  user.Comment = userData.Comment;

  user.IsApproved = userData.IsApproved;

  if (user.IsLockedOut && !userData.IsLockedOut)

    user.UnlockUser();

  mp.UpdateUser(user);

  return RedirectToAction("Index");

}

Теперь, чтобы механизм MVC Framework смог произвести сопоставление параметров с помощью встроенного механизма Model Binding, нам необходимо модифицировать код представления Select так, как показано в следующем фрагменте:

<% using (Html.BeginForm("Update", "Admin")) { %>

<%= Html.Hidden("userData.UserId", (Guid)user.ProviderUserKey)%>

<%= Html.AntiForgeryToken() %>

<fieldset>

  <legend>Данные</legend>

  <p>

    <label for="email">Email</label>

    <%=Html.TextBox("userData.Email", user.Email)%>

  </p>

  <p>

    <label for="comment">Комментарий</label>

    <%=Html.TextArea("userData.Comment", user.Comment)%>

  </p>

  <p>

    <label for="isApproved">

      <%=Html.CheckBox("userData.IsApproved", user.IsApproved)%>

        подтвержден

    </label>

  </p>

  <p>

    <label for="isLockedOut">

      <%=Html.CheckBox("userData.IsLockedOut", user.IsLockedOut)%>

        заблокирован

    </label>

  </p>

</fieldset>

<input type="submit" value="Coxpaнить" />

<% } %>

Обратите внимание на то, что для всех полей формы мы использовали наименование вида userData.Свойство. Например, поле email стало полем с именем userData.Email. Такое именование позволяет классу DefaultModelBinder, механизму Model Binding по умолчанию, сопоставить множественные параметры формы комплексному типу UserData.

Примечание

Строго говоря, в данном случае вам необязательно указывать для элементов формы префикс userData. Так как в сопоставлении участвует только один параметр комплексного типа, то механизм DefaultModelBinder автоматически определит значения его свойств, предположив, что все элементы формы относятся к единственному параметру userData. Однако мы рекомендуем указывать подобный префикс в любом случае для повышения читаемости кода и возможности более простого расширения кода в будущем.

*********************************************

Важной частью MVC Framework является возможность определять собственные механизмы Model Binding. Эта возможность предоставляет разработчику определять то, как параметры запроса или значения формы поступают к действию контроллера для обработки. Для демонстрации работы этого механизма добавим к нашей модели UserData еще одно свойство CurrentMembershipUser, которое будет автоматически инициализироваться при сопоставлении параметров:

public class UserData {

public MembershipUser CurrentMembershipUser { get; set; }

}

Теперь реализуем наш собственный механизм Model Binding, создав класс UserDataBinder, реализующий интерфейс IModelBinder. Этот интерфейс содержит всего один метод BindModel, с помощью которого и выполняется вся работа по сопоставлению параметров:

29
{"b":"971383","o":1}