在MVC3中绑定嵌套集合

时间:2012-11-21 05:13:37

标签: c# asp.net asp.net-mvc-3 razor model-binding

我有一个ViewModel,它包含一个多态对象列表。

public class InputParameters
{
    public InputParameters()
    {
        InputPrompts = new List<IInputPrompt>();
    }

    public string Name { get; set; }
    public string Path { get; set; }

    public IList<IInputPrompt> InputPrompts { get; set; }
}

反过来看这个:

public interface IInputPrompt
{
    string Name { get; set; }
    bool IsHidden { get; set; }
    bool IsRequired { get; set; }

    dynamic Value { get; set; }
}
public class TextPrompt : IInputPrompt
{
    public string Name { get; set; }
    public bool IsHidden { get; set; }
    public bool IsRequired { get; set; }
    public dynamic Value { get; set; }
}
public class MultiSelectPrompt : IInputPrompt
{
    public string Name { get; set; }
    public bool IsHidden { get; set; }
    public bool IsRequired { get; set; }
    public dynamic Value { get; set; }

    public MultiSelectList Values
    {
        get
        {
            return new MultiSelectList(((IDictionary<int, string>)Value).ToList(), "Key", "Value");
        }
    }
}

每个派生类型都有一个编辑器视图模板,视图看起来像这样:

@model OptionListModelBinding.Models.InputParameters
@{
    ViewBag.Title = "Index";
}
<h2>Index</h2>
@using (Html.BeginForm())
{
    <fieldset>
        <legend><strong>@Model.Name</strong></legend>
        <div><em>@Model.Path</em></div>
        <div>@Html.EditorFor(p => p.InputPrompts)</div>
        <div><input type="submit" /></div>
    </fieldset>

}
// editor template
@model OptionListModelBinding.Models.InputParameters
@{
    ViewBag.Title = "Index";
}
<h2>Index</h2>
@using (Html.BeginForm())
{
    <fieldset>
        <legend><strong>@Model.Name</strong></legend>
        <div><em>@Model.Path</em></div>
        <div>@Html.EditorFor(p => p.InputPrompts)</div>
        <div><input type="submit" /></div>
    </fieldset>

} 
// editor template
@model OptionListModelBinding.Models.MultiSelectPrompt
@{
    ViewBag.Title = "MultiSelectPrompt";
}

<h2>MultiSelectPrompt</h2>
<div><strong>@Model.Name</strong></div>
<div><em>@Html.ListBox(Model.Name, Model.Values)</em></div>

这一切都很好地呈现:Screen Capture

问题是:

如何将其绑定回模型?我有一个自定义模型绑定器:(原谅此代码的黑客性质)。

    public class InputParameterModelBinder : IModelBinder
{
    public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        if (bindingContext == null)
        {
            throw new ArgumentNullException("bindingContext");
        }
        // iterate over the form fields
        foreach (string item in controllerContext.HttpContext.Request.Form.AllKeys)
        {
            // this is where the problem is ... I cannot use the generic here.
            var dog = FromPostedData<string>(bindingContext, item);

        }

        InputParameters userInput = new InputParameters();

        return userInput;
    }

    // Dino Esposito code
    private T FromPostedData<T>(ModelBindingContext context, string id)
    {
        if (String.IsNullOrEmpty(id)) return default(T);
        var key = String.Format("{0}.{1}", context.ModelName, id);
        var result = context.ValueProvider.GetValue(key);
        if (result == null && context.FallbackToEmptyPrefix)
        {
            result = context.ValueProvider.GetValue(id);
            if (result == null) return default(T);
        }

        context.ModelState.SetModelValue(id, result);

        T valueToReturn = default(T);
        try
        {
            valueToReturn = (T)result.ConvertTo(typeof(T));
        }
        catch { }
        return valueToReturn;
    }
}

修改 我是否提到列表中的项目是在运行时确定的?

修改 这是报告生成工具的前端。后端服务提供可用报告列表以及运行每个报告所需的参数。这些都不是在编译时知道的,报告定义甚至可以随着需要重新编译的Web门户而改变。

我可以有一个可变数量和类型的输入参数。

2 个答案:

答案 0 :(得分:0)

我认为您不需要自定义活页夹。您只需将视图模型传递回post方法:

[HttpPost]
public ActionResult ProcessForm(InputParameters viewModel)
{
    ...
}

默认的MVC绑定器会将表单值与InputParameters参数的属性进行匹配。您可能需要在该参数上使用[Bind]属性来指示绑定器必须查找的变量名称前缀,如果您的视图使用任何类型的特殊表单变量名称。

答案 1 :(得分:0)

你不能指望MVC实例化IInputPrompt,因为它没有逻辑来确定你真正想要的类。您可以选择创建ModelBinder以模型绑定到此特定接口,或将接口更改为具体类。