我需要基于派生类型创建一个动态输入表单,但当传递到控制器的POST方法时,我无法正确绑定复杂属性。其他属性绑定良好。以下是我所拥有的一个人为的例子:
型号
public abstract class ModelBase {}
public class ModelDerivedA : ModelBase
{
public string SomeProperty { get; set; }
public SomeType MySomeType{ get; set; }
public ModelDerivedA()
{
MySomeType = new SomeType();
}
}
public class SomeType
{
public string SomeTypeStringA { get; set; }
public string SomeTypeStringB { get; set; }
}
自定义模型活页夹
绑定器基于这个答案:多态模型绑定
public class BaseViewModelBinder : DefaultModelBinder
{
protected override object CreateModel(ControllerContext controllerContext, ModelBindingContext bindingContext, Type modelType)
{
var typeValue = bindingContext.ValueProvider.GetValue("ModelType");
var type = Type.GetType(
(string)typeValue.ConvertTo(typeof(string)),
true
);
if (!typeof(ModelBase).IsAssignableFrom(type))
{
throw new InvalidOperationException("The model does not inherit from mode base");
}
var model = Activator.CreateInstance(type);
bindingContext.ModelMetadata = ModelMetadataProviders.Current.GetMetadataForType(() => model, type);
return model;
}
}
控制器
[HttpPost]
public ActionResult GetDynamicForm([ModelBinder(typeof(BaseViewModelBinder))] ModelBase model)
{
// model HAS values for SomeProperty
// model has NO values for MySomeType
}
查看摘录
@Html.Hidden("ModelType", Model.GetType())
@Html.Test(Model);
JavaScript
表单是使用$.ajax
和data: $(this).serialize()
发布的,如果我调试它,它会显示正确填充的表单数据。
模型中填充了除SomeType
之外的所有特性。我需要更改什么才能填充它们?
感谢
值没有被填充,因为您正在创建如下类型的新实例:
var model = Activator.CreateInstance(type);
bindingContext.ModelMetadata = ModelMetadataProviders.Current.GetMetadataForType(() => model, type);
return model;
而返回不正确的相同模型。
做下面这样的事情。
ValueProviderResult valueResult;
bindingContext.ModelState.SetModelValue("ModelType", valueResult);
return valueResult;
这里有关于modelBinder的非常好的讨论。
http://odetocode.com/blogs/scott/archive/2009/05/05/iterating-on-an-asp-net-mvc-model-binder.aspx
我通过以下方式解决了眼前的问题:
- 获取
FormvalueProvider
的实例(以访问已发布的内容) -
递归地遍历我的模型,并将每个属性值设置为
FormValueProvider
中的匹配值private FormValueProvider vp; protected override object CreateModel(ControllerContext controllerContext, ModelBindingContext bindingContext, Type modelType) { var typeValue = bindingContext.ValueProvider.GetValue("ModelType"); var type = Type.GetType( (string)typeValue.ConvertTo(typeof(string)), true ); if (!typeof(ModelBase).IsAssignableFrom(type)) { throw new InvalidOperationException("Bad Type"); } var model = Activator.CreateInstance(type); vp = new FormValueProvider(controllerContext); bindingContext.ValueProvider = vp; SetModelPropertValues(model); return model; }
和递归,基于这个答案在这里打印属性的嵌套对象
private void SetModelPropertValues(object obj)
{
Type objType = obj.GetType();
PropertyInfo[] properties = objType.GetProperties();
foreach (PropertyInfo property in properties)
{
object propValue = property.GetValue(obj, null);
var elems = propValue as IList;
if (elems != null)
{
foreach (var item in elems)
{
this.SetModelPropertValues(item);
}
}
else
{
if (property.PropertyType.Assembly == objType.Assembly)
{
this.SetModelPropertValues(propValue);
}
else
{
property.SetValue(obj, this.vp.GetValue(property.Name).AttemptedValue, null);
}
}
}
}
任何使用此功能的人都可能需要使其更加健壮以满足他们的需求。
作为解决这类问题的一般方法,我很乐意听到它的任何缺点。
不过,我希望这篇文章能在某些情况下有所帮助。
尝试将默认构造函数添加到ModelDerivedA
以初始化MySomeType
public class ModelDerivedA : ModelBase
{
public ModelDerivedA()
{
MySomeType = new SomeType();
}
}