ASP.NET MVC 2 Beta - 默认模型绑定器

时间:2009-11-19 03:45:34

标签: c# .net asp.net asp.net-mvc asp.net-mvc-2

从ASP.NET MVC 1.0切换到ASP.NET MVC 2 Beta后,我遇到了一些不同的行为。我检查了一些重大变化,但目前还不清楚问题所在。

问题与默认模型绑定器和实现IDataErrorInfo的模型有关。

属性(IDataErrorInfo.Item):

public string this[string columnName]
不再为每个属性调用

。我错过了什么?

2 个答案:

答案 0 :(得分:1)

MVC 1.0中的DefaultModelBinder:

protected virtual void OnPropertyValidated(ControllerContext controllerContext, ModelBindingContext bindingContext, PropertyDescriptor propertyDescriptor, object value)
{
    IDataErrorInfo model = bindingContext.Model as IDataErrorInfo;
    if (model != null)
    {
        string str = model[propertyDescriptor.Name];
        if (!string.IsNullOrEmpty(str))
        {
            string key = CreateSubPropertyName(bindingContext.ModelName, propertyDescriptor.Name);
            bindingContext.ModelState.AddModelError(key, str);
        }
    }
}

MVC 2.0 beta中的DefaultModelBinder:

protected virtual void OnPropertyValidated(ControllerContext controllerContext, ModelBindingContext bindingContext, PropertyDescriptor propertyDescriptor, object value)
{
    ModelMetadata metadata = bindingContext.PropertyMetadata[propertyDescriptor.Name];
    metadata.Model = value;
    string prefix = CreateSubPropertyName(bindingContext.ModelName, metadata.PropertyName);
    foreach (ModelValidator validator in metadata.GetValidators(controllerContext))
    {
        foreach (ModelValidationResult result in validator.Validate(bindingContext.Model))
        {
            bindingContext.ModelState.AddModelError(CreateSubPropertyName(prefix, result.MemberName), result.Message);
        }
    }
    if ((bindingContext.ModelState.IsValidField(prefix) && (value == null)) && !TypeHelpers.TypeAllowsNullValue(propertyDescriptor.PropertyType))
    {
        bindingContext.ModelState.AddModelError(prefix, GetValueRequiredResource(controllerContext));
    }
}

它没有使用IDataErrorInfo这个[string columnName]属性......好像是一个bug,因为DefaultModelBinder仍然使用Error属性。至少是不一致。

修改

我使用了反射器并注意到似乎没有使用DataErrorInfoPropertyModelValidator,所以我创建了自己的类:

public class DataErrorInfoModelPropertyValidatorProvider : ModelValidatorProvider
{
    // Methods
    public override IEnumerable<ModelValidator> GetValidators(ModelMetadata metadata, ControllerContext context)
    {
        if (metadata == null)
        {
            throw new ArgumentNullException("metadata");
        }
        if (context == null)
        {
            throw new ArgumentNullException("context");
        }

        var validators = new List<ModelValidator>();
        validators.Add(new DataErrorInfoPropertyModelValidator(metadata, context));
        return validators;
    }

    internal sealed class DataErrorInfoPropertyModelValidator : ModelValidator
    {
        // Methods
        public DataErrorInfoPropertyModelValidator(ModelMetadata metadata, ControllerContext controllerContext)
            : base(metadata, controllerContext)
        {
        }

        public override IEnumerable<ModelValidationResult> Validate(object container)
        {
            if (container != null)
            {
                IDataErrorInfo info = container as IDataErrorInfo;
                if (info != null)
                {
                    string str = info[Metadata.PropertyName];
                    if (!string.IsNullOrEmpty(str))
                    {
                        ModelValidationResult[] resultArray = new ModelValidationResult[1];
                        ModelValidationResult result = new ModelValidationResult();
                        result.Message = str;
                        resultArray[0] = result;
                        return resultArray;
                    }
                }
            }
            return Enumerable.Empty<ModelValidationResult>();
        }
    }
}

然后我用了:

ModelValidatorProviders.Providers.Add(new DataErrorInfoModelPropertyValidatorProvider());

它的工作原理:)这只是暂时的解决方案。必须在最终的MVC 2中予以纠正。

修改

我还在if (base.Metadata.Model != null)的{​​{1}}方法中将Validate()更改为if(container!= null)。

答案 1 :(得分:1)

经过一些进一步的调试工作后,我相信我理解为什么在我的特定情况下IDataErrorInfo.Item没有被调用。 ASP.NET MVC 2 Beta中使用以下代码来验证IDataErrorInfo属性:

internal sealed class DataErrorInfoPropertyModelValidator : ModelValidator
{
    public DataErrorInfoPropertyModelValidator(ModelMetadata metadata, ControllerContext controllerContext)
        : base(metadata, controllerContext)
    {
    }

    public override IEnumerable<ModelValidationResult> Validate(object container)
    {
        if (Metadata.Model != null)
        {
            var castContainer = container as IDataErrorInfo;
            if (castContainer != null)
            {
                string errorMessage = castContainer[Metadata.PropertyName];
                if (!String.IsNullOrEmpty(errorMessage))
                {
                    return new[] { new ModelValidationResult { Message = errorMessage } };
                }
            }
        }
        return Enumerable.Empty<ModelValidationResult>();
    }
}

我的模型包含一个System.Nullable<int>的属性,当模型绑定器绑定HTML帖子中的空字符串时,Metadata.Model等于null,因此验证不会运行。

这与ASP.NET MVC 1.0有根本的不同,在这种情况下,此方案会一直激活验证程序,直至调用IDataErrorInfo.Item

我是不是按照预期的方式使用了某种东西?