Модель Binder & Скрытые поля - PullRequest
3 голосов
/ 20 мая 2011

У меня есть упрощенный тестовый сценарий, полезный для постановки этого вопроса: у продукта может быть много компонентов, компонент может принадлежать многим продуктам.EF создал классы, я уменьшил их следующим образом:

public partial class Product
{
    public int Id { get; set; }
    public string Name { get; set; }
    public virtual ICollection<Component> Components { get; set; }
}
public partial class Component
{
    public int Id { get; set; }
    public string Name { get; set; }
    public virtual ICollection<Product> Products { get; set; }
}

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

public ActionResult Create(int ProductId)
{
    Product p = db.Products.Find(ProductId);
    Component c = new Component();
    c.Products.Add(p);
    return PartialView(c);
} 

[HttpPost]
public ActionResult Create(Component model)
{
    db.Components.Add(model);
    db.SaveChanges();
}

и представление, возвращаемое GETметод выглядит следующим образом:

@model Test.Models.Product

<fieldset>
    <legend>Product</legend>
    <div class="display-label">Name</div>
    <div class="display-field">@Model.Name</div>
</fieldset>

@Html.Action("Create", "Component", new {ProductId = Model.Id}) 
<p>
    @Html.ActionLink("Edit", "Edit", new { id=Model.Id }) |
    @Html.ActionLink("Back to List", "Index")
</p>

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

@model Test.Models.Component
@using Test.Models

<script type="text/javascript">
    function Success() {
        alert('ok');
    }
    function Failure() {
        alert('err');
    }
</script>
@using (Ajax.BeginForm("Create", "Component", new AjaxOptions
{
    HttpMethod = "Post",
    OnSuccess = "Success",
    OnFailure = "Failure"
}))
{
    <fieldset>
        <legend>Components</legend>

        <div class="editor-label">
            @Html.LabelFor(model => model.Name)
        </div>
        <div class="editor-field">
            @Html.EditorFor(model => model.Name)
            @Html.ValidationMessageFor(model => model.Name)
        </div>
        @Html.HiddenFor(x => x.Products.First().Id)
        @Html.HiddenFor(x => x.Products)
        @foreach (Product p in Model.Products)
        {
            @Html.Hidden("Products[0].Id", p.Id)
        }
        @foreach (Product p in Model.Products)
        {
            @Html.Hidden("[0].Id", p.Id)
        }
    </fieldset>
    <input type="submit" value="go" />
}

ки.так вот с чем я борюсь: мне нужен параметр model в [HttpPost] для правильного заполнения, т.е. он должен содержать Product, так как я не могу создать новый компонент с нулевым продуктом.Чтобы получить продукт, мне нужно посмотреть его через идентификатор продукта.Я ожидаю, что смогу сделать:

model.Products.Add(db.Products.Find(model.Products.First().Id));

или что-то подобное, что зависит от model получения идентификатора.Это означает, что представление должно размещать там идентификатор, предположительно в скрытом поле, и, как видно из кода моего представления, я предпринял несколько попыток его заполнения, но все они потерпели неудачу.

ОбычноЯ предпочитаю методы * For, поскольку они становятся ответственными за создание правильной номенклатуры.Если бы .Products были в единственном числе (.Product), я мог бы ссылаться на него как x => x.Product.Id, и все было бы хорошо, но, поскольку это множественное число, я не могу сделать x => x.Products.Id, поэтому я попытался x => x.Products.First().Id, который компилирует и производит правильное значениено получает имя Id (что неверно, поскольку связыватель модели думает, что это Component.Id, а не Component.Products[0].Id.

Моя вторая попытка состояла в том, чтобы позволить HiddenFor повторяться (как я сделал бы с EditorFor):

@Html.HiddenFor(x => x.Products)

но это ничего не дает - я читал, что этот помощник не повторяется. Я пробовал x => x.Products.First(), но он даже не компилируется. Наконец, я решил отказаться от * For исам кодируйте имя:

@foreach (Product p in Model.Products)
{
    @Html.Hidden("Products[0].Id", p.Id)

и хотя это выглядит правильно, постбэк не видит мое значение (Products.Count == 0). Я видел в некоторых публикациях, что формат должен выглядеть как [0].Idно это тоже не сработает. grr ...

Я понял, что могу написать это так:

@Html.Hidden("ProductId", p.Id)

, а затем заново объявить действие моего контроллера следующим образом:

[HttpPost] ActionResult Create(Component model, int ProductId)

но это кажется глупым. Трудно поверить, что это так сложно. Кто-нибудь может помочь?

  • e

ps У меня есть проект, который я могу сделать доступным для скачивания, если кому-то захочется

1 Ответ

13 голосов
/ 20 мая 2011

Вместо записи этих foreach циклов попробуйте использовать шаблоны редактора:

<fieldset>
    <legend>Components</legend>

    <div class="editor-label">
        @Html.LabelFor(model => model.Name)
    </div>

    <div class="editor-field">
        @Html.EditorFor(model => model.Name)
        @Html.ValidationMessageFor(model => model.Name)
    </div>

    @Html.EditorFor(x => x.Products)
</fieldset>

и внутри соответствующего шаблона редактора (~/Views/Shared/EditorTemplates/Product.cshtml)

@model Product
@Html.HiddenFor(x => x.Id)
...