4

我想在我的模型中将一组文本框、小时、分钟和秒绑定到 TimeSpan。我不想将小时、分钟和秒添加到我的模型中,并且我不想将它们作为传递给操作方法的参数。

我发现了一些关于使用编辑器模板的讨论,但找不到一个很好的例子。

4

1 回答 1

7

有两种方法可以做到这一点。第一种方法是实现IModelBinder,这需要您将TimeSpan作为单独的参数传递给您的操作,然后将模型的TimeSpan属性设置为它:

[HttpPost]
public ActionResult Index(YourViewModel model, 
    [ModelBinder(typeof(TimeSpanModelBinder))] TimeSpan ts)
{
    model.LengthOfTime = ts;

    // do stuff
}

第二个是派生自DefaultModelBinder,然后您可以简单地直接绑定到您的模型:

[HttpPost]
public ActionResult Index([ModelBinder(typeof(TimeSpanModelBinder))] YourViewModel model)
{
    // do stuff
}

实现的优点IModelBinder是您不需要定义自定义属性来标记您的属性。您也不需要在运行时动态查找模型上的属性。

派生的优点DefaultModelBinder是,只要您使用自定义属性,它就适用于您拥有的任何模型,因此如果您需要在多个地方使用它,则可以在您的操作中保持一致性。

第一种方法:实现 IModelBinder

对于这种方法,实现IModelBinder只是意味着为该方法创建一个实现BindModel

public class TimeSpanModelBinder : IModelBinder
{
    public object BindModel(ControllerContext controllerContext,
        ModelBindingContext bindingContext)
    {
        if (bindingContext.ModelType != typeof(TimeSpan))
            return null;

        int hours = 0, minutes = 0, seconds = 0;

        hours = ParseTimeComponent(HoursKey, bindingContext);
        minutes = ParseTimeComponent(MinutesKey, bindingContext);
        seconds = ParseTimeComponent(SecondsKey, bindingContext);

        return new TimeSpan(hours, minutes, seconds);
    }

    public int ParseTimeComponent(string component,
        ModelBindingContext bindingContext)
    {
        int result = 0;
        var val = bindingContext.ValueProvider.GetValue(component);

        if (!int.TryParse(val.AttemptedValue, out result))
            bindingContext.ModelState.AddModelError(component,
                String.Format("The field '{0}' is required.", component));

        // This is important
        bindingContext.ModelState.SetModelValue(component, val);

        return result;
    }

    private readonly string HoursKey = "Hours";
    private readonly string MinutesKey = "Minutes";
    private readonly string SecondsKey = "Seconds";
}

注意对 的调用bindingContext.ModelState.SetModelValue。此方法可确保您的模型在需要重新显示在表单上时使用正确的数据重新填充。这很重要,因为如果用户提交的数据验证失败,它会保留再次填写所有表单字段的默认功能。

第二种方法:从 DefaultModelBinder 派生

您需要做的第一件事是创建一个自定义属性,您将使用它来标记您希望绑定到的模型的属性:

[AttributeUsage(AttributeTargets.Property, AllowMultiple = false)]
public class TimeSpanComponentAttribute : Attribute
{
    public override bool Match(object obj)
    {
        return obj.GetType() == typeof(TimeSpan);
    }
}

然后,您将在您的视图模型中使用它,如下所示:

public class YourViewModel
{
    [Required]
    public string SomeRequiredProperty { get; set; }
    [TimeSpanComponent]
    public TimeSpan LengthOfTime { get; set; }
}

然后,这允许我们在自定义模型绑定器中查找用该属性标记的属性:

public class TimeSpanModelBinder : DefaultModelBinder
{
    public override object BindModel(ControllerContext controllerContext,
        ModelBindingContext bindingContext)
    {
        var model = this.CreateModel(controllerContext,
            bindingContext, bindingContext.ModelType);
        bindingContext.ModelMetadata.Model = model;

        var target = model.GetType()
            .GetProperties(BindingFlags.Public | BindingFlags.Instance)
            .Where(p => Attribute.IsDefined(p, typeof(TimeSpanComponentAttribute)))
            .Single();

        if (target == null)
            throw new MemberAccessException(PropertyNotFound);

        int hours = 0, minutes = 0, seconds = 0;

        hours = ParseTimeComponent(HoursKey, bindingContext);
        minutes = ParseTimeComponent(MinutesKey, bindingContext);
        seconds = ParseTimeComponent(SecondsKey, bindingContext);

        target.SetValue(model, new TimeSpan(hours, minutes, seconds));

        return base.BindModel(controllerContext, bindingContext);
    }

    public int ParseTimeComponent(string component,
        ModelBindingContext bindingContext)
    {
        int result = 0;
        var val = bindingContext.ValueProvider.GetValue(component);

        if (!int.TryParse(val.AttemptedValue, out result))
            bindingContext.ModelState.AddModelError(component,
                String.Format("The field '{0}' is required.", component));

        // Again, this is important
        bindingContext.ModelState.SetModelValue(component, val);

        return result;
    }

    private readonly string HoursKey = "Hours";
    private readonly string MinutesKey = "Minutes";
    private readonly string SecondsKey = "Seconds";
    private readonly string PropertyNotFound = "Could not bind to TimeSpan property.  Did you forget to decorate " +
                                               "a property with TimeSpanComponentAttribute?";
}

请注意BindModel我们如何根据自定义属性定位正确的属性。此外,在我们完成对属性的绑定之后,对基本版本的调用BindModel允许默认模型绑定器处理其他所有事情。

笔记

这两种方法都假设您的文本框的名称分别是Hours,MinutesSeconds。如果不是,只需更改 3 个私有字符串的值。

这次编辑我改变了很多,所以如果我错过了什么,请告诉我。

感谢您提出这个问题——我从中学到了很多。

于 2013-09-28T17:42:13.877 回答