使用ASP.NET MVC中的自定义模型绑定器验证整数字段

时间:2010-03-03 14:13:35

标签: c# asp.net-mvc validation

我遇到了一个我认为很有趣的问题。

获取一个输入表单,其中包含一个整数值的输入字段(在我的情况下为cost)。我不希望将“$ 100”或“Twenty Dollars”发送到数据库,因为它可能会抱怨。我想向模型添加错误并将其发送回用户。似乎很简单;)。

我正在使用自定义模型绑定器,因为我在域模型中使用了一些继承。我有多种类型的事件,每种类型实现IEvent - 因此,我需要一个自定义模型绑定器。

我遇到的问题是,当我尝试绑定成本字段,并且从字符串到int的转换失败时,我真的不确定如何最好地处理它。 < /强>

我的自定义模型活页夹     public class EventModelBinder:IModelBinder     {         #region IModelBinder会员

    public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        // Trivial code ... binding other fields...

        _event.ProjectedCost = GetA<int>(bindingContext, "ProjectedCost").GetValueOrDefault(-1);

        return _event;
    }

    #endregion

    // From Scott Hanselman's blog :)
    private Nullable<T> GetA<T>(ModelBindingContext bindingContext, string key) where T : struct
    {
        if (String.IsNullOrEmpty(key)) return null;
        ValueProviderResult valueResult;
        Nullable<T> ret;
        //Try it with the prefix...
        bindingContext.ValueProvider.TryGetValue(bindingContext.ModelName + "." + key, out valueResult);
        //Didn't work? Try without the prefix if needed...
        if (valueResult == null && bindingContext.FallbackToEmptyPrefix == true)
        {
            bindingContext.ValueProvider.TryGetValue(key, out valueResult);
        }
        if (valueResult == null)
        {
            return null;
        }

        try
        {
            ret = (Nullable<T>)valueResult.ConvertTo(typeof(T));
        }
        catch
        {
            return null;
        }

        return ret;
    }

这非常有效 - 如果Enums为null或未选中,我可以对它们进行验证并让用户知道它是必需的。对于成本字段,如果转换失败,我得-1。但是,当我去验证服务器上的数据并返回UI时,Model.Event.ProjectedCost字段为空

我的EventService

// ...
protected bool ValidateEvent(IEvent eventToValidate)
    {            
        if (eventToValidate.ProjectedCost < 0)
            _validationDictionary.AddError("ProjectedCost", "Incorrect format");

        return _validationDictionary.IsValid;
    }

    public bool SaveEvent(IEvent _event)
    {
        if (!ValidateEvent(_event))
            return false;
        try
        {
            _repository.SaveEvent(_event);
            return true;
        }
        catch
        {
            return false;
        }

我的Edit.aspx视图

<h2>Edit</h2>

<%= Html.ValidationSummary("Edit was unsuccessful. Please correct the errors and try again.") %>

<% using (Html.BeginForm()) {%>

    <fieldset>
        <legend>Fields</legend>
        <p>
            <label for="ProjectedCost">ProjectedCost:</label>
            <%= Html.TextBox("ProjectedCost", Model.Event.ProjectedCost) %>
            <%= Html.ValidationMessage("ProjectedCost", "*") %>
        </p>

        <% Html.RenderPartial(String.Format("~/Views/Shared/{0}Form.ascx", Model.Event.Type), Model.Event, ViewData); %>

        <p>
            <input type="submit" value="Save" />
        </p>
    </fieldset>

<% } %>

<div>
    <%=Html.ActionLink("Back to List", "Index") %>
</div>

在我看来,Model.Event.ProjectedCost如果没有验证则为null,给我这个(第54行是罪魁祸首):

Exception Details: System.NullReferenceException: Object reference not set to an instance of an object.

Source Error:

Line 52:             <p>
Line 53:                 <label for="ProjectedCost">ProjectedCost:</label>
Line 54:                 <%= Html.TextBox("ProjectedCost", (Model.Event.ProjectedCost == null ? 0 : Model.Event.ProjectedCost)) %>
Line 55:                 <%= Html.ValidationMessage("ProjectedCost", "*") %>
Line 56:             </p>

我喜欢做的是将用户输入的值发送回用户,但我的自定义模型绑定器和/或验证逻辑似乎将某些内容设置为null?

我意识到这可能不是最容易阅读的问题,所以如果我能以任何方式澄清,请告诉我!

2 个答案:

答案 0 :(得分:1)

您可以使用javascript将文本框的输入限制为整数。

答案 1 :(得分:0)

似乎如果你先调用没有SetModelValue的AddModelError,你有时会得到一个新的NullReferenceExceptions:

http://www.crankingoutcode.com/?aspxerrorpath=/2009/02/01/IssuesWithAddModelErrorSetModelValueWithMVCRC1.aspx