Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

ASP.Net MVC ModelState / Html.TextBox postback issue

I have an issue cropping up in a form I'm trying to post. In the scenario where the form doesn't validate, I'm taking the standard route of calling ModelState.AddModelError() then returning a View result.

The thing is, the HTML.* helpers are supposed to pick up the posted value when rendering and I'm noticing that my text fields ONLY do so if I include them in the parameter list of the postback action, which shouldn't be required seeing as some forms have way too many fields to want to list them all as parameters.

My action code is roughly:

[AcceptVerbs(HttpVerbs.Post)]
public ActionResult EditDataDefinition(long? id, string name)
{
    var dataDefinition = ...

    // do some validation stuff
    if (!ModelState.IsValid)
    {
        // manually set checkbox fields via ViewData seeing as this STILL doesn't work in MC 1.0 :P
        // ...
        return View(dataDefinition);
    }

}

Now, dataDefinition (which is a LINQ to SQL entity) has a field MinVolume, is handled in the view by this line:

Minimum: <%= Html.TextBox("MinVolume", null, new { size = 5 })%>

Yet when the view is rendered after a failed ModelState validation, the value typed into it on the original page we posted is not preserved UNLESS I include it as a parameter in the postback method. Literally, I can "solve the problem" by doing this:

[AcceptVerbs(HttpVerbs.Post)]
public ActionResult EditDataDefinition(long? id, string name, string minVolume)

For some reason that will force the field value to be preserved. This seems stupid to me because my form has way more values than just that and I shouldn't have to add a parameter for just that field.

Any ideas?

like image 424
Nathan Ridley Avatar asked May 07 '09 14:05

Nathan Ridley


3 Answers

Oh man I've just improved my application design. The problem occurs because you have custom validation (I have too). You have to add after

ModelState.AddModelError()

this

ModelState.SetModelValue("MinVolume", ValueProvider["MinVolume"]);

In view it has to be

Mimum:<%=Html.Textbox("MinVolume")%>

Still not sure why it works but it worked for me.

like image 119
Eugeniu Torica Avatar answered Oct 17 '22 10:10

Eugeniu Torica


Could it be that your code:

<%= Html.TextBox("MinVolume", null, new { size = 5 })%>

..has the null for the default value param? Maybe if you change the null to Model.MinVolume it will persist the value. Like this:

<%= Html.TextBox("MinVolume", Model.MinVolume, new { size = 5 })%>

I'm not sure if your action returns the value MinVolume in the model tho. If it does, the above should work. Else, you may need to refactor the action slightly.

like image 23
Matt Kocaj Avatar answered Oct 17 '22 10:10

Matt Kocaj


What is the key you are using when you set the value in the ModelState on error? The code that sets the value parameter for a TextBox looks like:

Relevant portion of the downloaded framework code.

string attemptedValue = (string)htmlHelper.GetModelStateValue(name, typeof(string));
tagBuilder.MergeAttribute("value", attemptedValue ?? ((useViewData) ? htmlHelper.EvalString(name) : valueParameter), isExplicitValue);

As you can see if the attempt value exists, it will use it -- but only if the same key is available.

I know that this works because I have an action that takes no parameters and gets the values directly from the ValueProvider and it uses AddModelError to indicate validation errors. I'm sure that the values in my TextBoxes are retained.

EDIT: In order for the values to be retained, they need to be associated with the model in some way. One way to do this is to add them to the parameter list. Another way is to use UpdateModel (with the parameter names in the whitelist or no whitelist). A third way is to add the parameter explicitly to the model as in @Jenea's answer. Since the helper only pulls from the model state, they must be in there for the values to be retained. It does not look at the request's Form property.

like image 29
tvanfosson Avatar answered Oct 17 '22 08:10

tvanfosson