45

MVC 3.0 中的默认模型绑定器是否能够处理非顺序索引(对于简单和复杂模型类型)?我遇到过建议应该这样做的帖子,但是在我的测试中似乎没有。

给定回发值:

items[0].Id = 10
items[0].Name = "Some Item"
items[1].Id = 3
items[1].Name = "Some Item"
items[4].Id = 6
items[4].Name = "Some Item"

还有一个控制器方法:

public ActionResult(IList<MyItem> items) { ... }

加载的唯一值是项目 0 和 1;第 4 项被简单地忽略。

我见过许多生成自定义索引的解决方案(模型绑定到列表),但是它们似乎都针对以前版本的 MVC,而且大多数都是有点“笨拙”的 IMO。

我错过了什么吗?

4

6 回答 6

78

我有这个工作,你必须记住添加一个常见的索引隐藏输入,如参考文章中所述:

隐藏的输入name = Items.Index是关键部分

<input type="hidden" name="Items.Index" value="0" />
<input type="text" name="Items[0].Name" value="someValue1" />

<input type="hidden" name="Items.Index" value="1" />
<input type="text" name="Items[1].Name" value="someValue2" />

<input type="hidden" name="Items.Index" value="3" />
<input type="text" name="Items[3].Name" value="someValue3" />

<input type="hidden" name="Items.Index" value="4" />
<input type="text" name="Items[4].Name" value="someValue4" />

希望这可以帮助

于 2011-12-22T01:01:24.450 回答
5

这种源自 Steve Sanderson 方法的辅助方法要简单得多,可用于锚定集合中的任何项目,而且它似乎可以与 MVC 模型绑定一起使用。

public static IHtmlString AnchorIndex(this HtmlHelper html)
{
    var htmlFieldPrefix = html.ViewData.TemplateInfo.HtmlFieldPrefix;
    var m = Regex.Match(htmlFieldPrefix, @"([\w]+)\[([\w]*)\]");
    if (m.Success && m.Groups.Count == 3)
        return
            MvcHtmlString.Create(
                string.Format(
                    "<input type=\"hidden\" name=\"{0}.index\" autocomplete=\"off\" value=\"{1}\" />",
                    m.Groups[1].Value, m.Groups[2].Value));
    return null;
}

例如,只需在 EditorTemplate 中调用它,或者在您可以生成输入的任何其他位置调用它,如下所示,以生成索引锚定隐藏变量(如果适用)。

@model SomeViewModel
@Html.AnchorIndex()
@Html.TextBoxFor(m => m.Name)
... etc.

我认为它比史蒂夫桑德森的方法有一些优势。

  1. 它与 EditorFor 和其他用于处理枚举的内置机制一起使用。因此,如果ItemsIEnumerable<T>视图模型上的属性,则以下内容按预期工作:

    <ul id="editorRows" class="list-unstyled"> @Html.EditorFor(m => m.Items) @* Each item will correctly anchor allowing for dynamic add/deletion via Javascript *@ </ul>

  2. 它更简单,不需要更多的魔术字符串。

  3. 对于数据类型,您可以使用单个 EditorTemplate/DisplayTemplate,如果未在列表中的项目上使用,它将只是无操作。

唯一的缺点是,如果被绑定的根模型是可枚举的(即 Action 方法本身的参数,而不仅仅是参数对象图中某个更深处的属性),则绑定将在第一个非顺序索引处失败。不幸的是,.IndexDefaultModelBinder 的功能仅适用于非根对象。在这种情况下,您唯一的选择仍然是使用上述方法。

于 2014-01-24T04:11:02.187 回答
4

您引用的文章是一篇旧文章(MVC2),但据我所知,这仍然是使用默认模型绑定器对绑定集合进行建模的事实上的方法。

如果你想要非顺序索引,就像 Bassam 说的那样,你需要指定一个索引器。索引器不需要是数字的。

为此,我们使用Steve Sanderson 的 BeginCollectionItem Html Helper。它会自动将索引器生成为 Guid。我认为当您的集合项 HTML 是非顺序的时,这比使用数字索引器更好。

于 2011-12-22T03:57:01.290 回答
2

这周我一直在努力解决这个问题,而巴萨姆的回答是让我走上正轨的关键。我有一个可以有数量字段的库存项目的动态列表。我需要知道他们选择了多少项目,除了项目列表可以从 1 到n变化。

最后我的解决方案相当简单。我创建了一个名为 ItemVM 的 ViewModel,它有两个属性。项目 ID 和数量。在发布操作中,我接受了这些列表。启用索引后,所有项目都会通过..即使数量为空。您必须在服务器端验证和处理它,但是通过迭代处理这个动态列表是微不足道的。

在我看来,我正在使用这样的东西:

@foreach (Item item in Items)
{
<input type="hidden" name="OrderItems.Index" value="@item.ItemID" />
<input type="hidden" name="OrderItems[@item.ItemID].ItemID" value="@item.ItemID" />
<input type="number" name="OrderItems[@item.ItemID].Quantity" />
}

这给了我一个基于 0 的索引的列表,但控制器中的迭代会从新的强类型模型中提取所有必要的数据。

public ActionResult Marketing(List<ItemVM> OrderItems)
...
        foreach (ItemVM itemVM in OrderItems)
            {
                OrderItem item = new OrderItem();
                item.ItemID = Convert.ToInt16(itemVM.ItemID);
                item.Quantity = Convert.ToInt16(itemVM.Quantity);
                if (item.Quantity > 0)
                {
                    order.Items.Add(item);
                }
            }

然后,您将得到一个数量大于 0 的项目集合和项目 ID。

这项技术在 MVC 5 中使用 Visual Studio 2015 中的 EF 6。也许这会帮助像我一样搜索此解决方案的人。

于 2015-10-10T15:04:16.820 回答
1

或者使用这个 javascript 函数来修复索引:(显然替换 EntityName 和 FieldName)

function fixIndexing() {
        var tableRows = $('#tblMyEntities tbody tr');

        for (x = 0; x < tableRows.length; x++) {
            tableRows.eq(x).attr('data-index', x);

            tableRows.eq(x).children('td:nth-child(1)').children('input:first').attr('name', 'EntityName[' + x + "].FieldName1");

            tableRows.eq(x).children('td:nth-child(2)').children('input:first').attr('name', 'EntityName[' + x + "].FieldName2");

            tableRows.eq(x).children('td:nth-child(3)').children('input:first').attr('name', 'EntityName[' + x + "].FieldName3");
        }

        return true; //- Submit Form -
    }
于 2015-01-14T08:55:57.977 回答
1

我最终制作了一个更通用的 HTML Helper:-

using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
using System.Text;
using System.Text.RegularExpressions;
using System.Web;
using System.Web.Mvc;

namespace Wallboards.Web.Helpers
{
    /// <summary>
    /// Hidden Index Html Helper
    /// </summary>
    public static class HiddenIndexHtmlHelper
    {
        /// <summary>
        /// Hiddens the index for.
        /// </summary>
        /// <typeparam name="TModel">The type of the model.</typeparam>
        /// <typeparam name="TProperty">The type of the property.</typeparam>
        /// <param name="htmlHelper">The HTML helper.</param>
        /// <param name="expression">The expression.</param>
        /// <param name="index">The Index</param>
        /// <returns>Returns Hidden Index For</returns>
        public static MvcHtmlString HiddenIndexFor<TModel, TProperty>(this HtmlHelper<TModel> htmlHelper, Expression<Func<TModel, TProperty>> expression, int index)
        {
            var metadata = ModelMetadata.FromLambdaExpression(expression, htmlHelper.ViewData);
            var propName = metadata.PropertyName;

            StringBuilder sb = new StringBuilder();
            sb.AppendFormat("<input type=\"hidden\" name=\"{0}.Index\" autocomplete=\"off\" value=\"{1}\" />", propName, index);

            return MvcHtmlString.Create(sb.ToString());
        }
    }
}

然后将其包含在 Razor 视图中列表元素的每次迭代中:-

@Html.HiddenIndexFor(m => m.ExistingWallboardMessages, i)
于 2017-03-10T09:55:26.277 回答