ASP.NET MVC - проблема с EditorTemplate для ICollection <T>, сопоставленной с Enum - PullRequest
5 голосов
/ 11 января 2011

У меня есть веб-сайт ASP.NET MVC 3 (Razor) и (упрощенная) модель под названием Review :

public class Review
{
   public int ReviewId { get; set; }
   public bool RecommendationOne
   {
       // hook property - gets/set values in the ICollection
   }
   public bool RecommendationTwo { // etc }
   public ICollection<Recommendation> Recommendations { get; set; }
}

Рекомендация выглядит следующим образом:

public class Recommendation
{
   public byte RecommendationTypeId
}

У меня также есть enum , называемый PracticeType , который я использую для сопоставления вышеуказанной рекомендации. (основано на Рекомендации Типа).

Итак, подведем итог - один обзор имеет множество Рекомендаций , и каждая из этих Рекомендаций соответствует определенному типу перечисления, я раскрываю свойства ловушки для упрощения привязки / кода модели.

Итак, на представление:

@Html.EditorFor(model => model.Recommendations, "Recommendations")

Довольно просто.

Теперь для шаблона редактора я хочу отобразить флажок для каждого возможного РекомендацияТип (перечисление), и если модель имеет эту рекомендацию (например, в режиме редактирования), я устанавливаю флажок.

Вот что у меня есть:

@model IEnumerable<xxxx.DomainModel.Core.Posts.Recommendation>
@using xxxx.DomainModel.Core.Posts;

@{
    Layout = null;
}

<table>
    @foreach (var rec in Enum.GetValues(typeof(RecommendationType)).Cast<RecommendationType>())
    {
        <tr>
            <td>
                @* If review contains this recommendation, check the box *@
                @if (Model != null && Model.Any(x => x.RecommendationTypeId == (byte)rec))
                {
                    @* How do i create a (checked) checkbox here? *@
                }
                else
                {
                    @* How do i created a checkbox here? *@
                }

                @rec.ToDescription()
            </td>
        </tr>
    }
</table>

Как следует из комментариев - я не знаю, как использовать @Html.CheckBoxFor. Обычно для этого требуется выражение, основанное на модели, но я уверен, как связать со свойством hook, основываясь на текущем зацикленном значении перечисления. Например, он должен динамически делать @Html.CheckBoxFor(x => x.RecommendationOne), @Html.CheckBoxFor(x => x.RecommendationTwo) и т. Д.

Текущее решение, которое у меня есть (работает), включает в себя создание <input> вручную (включая скрытые поля).

Но так как я только знакомлюсь с шаблонами редакторов, надеюсь, что кто-то со стажем может направить меня в "строго типизированном" направлении.

Или есть ли лучший способ (HTML Helper), я могу это сделать?

1 Ответ

10 голосов
/ 11 января 2011

Я бы начал с представления правильной модели представления для сценария:

public enum RecommendationType { One, Two, Three }

public class ReviewViewModel
{
    public IEnumerable<RecommendationViewModel> Recommendations { get; set; }
}

public class RecommendationViewModel
{
    public RecommendationType RecommendationType { get; set; }
    public bool IsChecked { get; set; }
}

Тогда контроллер:

public class HomeController : Controller
{
    public ActionResult Index()
    {
        // TODO: query the repository to fetch your model
        // and use AutoMapper to map between it and the 
        // corresponding view model so that you have a true/false
        // for each enum value
        var model = new ReviewViewModel
        {
            Recommendations = new[]
            {
                new RecommendationViewModel { 
                    RecommendationType = RecommendationType.One, 
                    IsChecked = false 
                },
                new RecommendationViewModel { 
                    RecommendationType = RecommendationType.Two, 
                    IsChecked = true 
                },
                new RecommendationViewModel { 
                    RecommendationType = RecommendationType.Three, 
                    IsChecked = true 
                },
            }
        };
        return View(model);
    }

    [HttpPost]
    public ActionResult Index(ReviewViewModel model)
    {
        // Here you will get for each enum value the corresponding
        // checked value
        // TODO: Use AutoMapper to map back to your model and persist
        // using a repository
        return RedirectToAction("Success");
    }
}

и соответствующий вид (~/Views/Home/Index.cshtml):

@model YourAppName.Models.ReviewViewModel

@{
    ViewBag.Title = "Index";
}

@using (Html.BeginForm())
{
    @Html.EditorFor(model => model.Recommendations)
    <input type="submit" value="Go" />
}

и, наконец, шаблон редактора (~/Views/Home/EditorTemplates/RecommendationViewModel.cshtml)

@model YourAppName.Models.RecommendationViewModel
<div>
    @Html.HiddenFor(x => x.RecommendationType)
    @Model.RecommendationType 
    @Html.CheckBoxFor(x => x.IsChecked)
</div>

Теперь код представления очищен, как и должно быть. Нет ifs, нет циклов, нет LINQ, нет отражения, это ответственность уровня контроллера / преобразователя. Поэтому каждый раз, когда вы пишете какую-то продвинутую логику C # в своем представлении, я бы порекомендовал вам переосмыслить модели представлений и адаптировать их по мере необходимости. Вот для чего предназначены модели представлений: быть как можно ближе к логике представления.

...