Asp.net mvc html.hiddenfor с неправильной стоимостью
-
11-10-2019 - |
Вопрос
Я использую MVC 3 в своем проекте, и я вижу очень странное поведение.
Я пытаюсь создать скрытое поле для определенного значения в моей модели, проблема в том, что по какой -то причине значение, установленное в поле, не соответствует значению в модели.
например
У меня есть этот код, как и тест:
<%:Html.Hidden("Step2", Model.Step) %>
<%:Html.HiddenFor(m => m.Step) %>
Я думаю, что оба скрытых поля будут иметь одинаковое значение. Что я делаю, так это установите значение на 1, когда впервые отображаю представление, а затем после подачи я увеличиваю значение поля модели на 1.
Итак, в первый раз, когда я рендеринг страницы оба элемента управления имеют значение 1, но во второй раз, когда отображаемые значения - это:
<input id="Step2" name="Step2" type="hidden" value="2" />
<input id="Step" name="Step" type="hidden" value="1" />
Как вы можете видеть, первое значение верно, но второе значение кажется таким же, как первый раз, когда я отображаю представление.
Что мне не хватает? *Для HTML -помощников каким -то образом кэшируют значения? Если да, то как я могу отключить это кэширование?
Спасибо за вашу помощь.
Решение
Это нормально, и именно так работают HTML -помощники. Сначала они используют значение запроса POST, а после этого значение в модели. Это означает, что даже если вы измените значение модели в действии вашего контроллера, если в запросе Post есть такая же переменная, ваша модификация будет игнорирована, и будет использоваться опубликованное значение.
Одним из возможных обходных путей является удаление этого значения из состояния модели в действии контроллера, которое пытается изменить значение:
// remove the Step variable from the model state
// if you want the changes in the model to be
// taken into account
ModelState.Remove("Step");
model.Step = 2;
Другая возможность состоит в том, чтобы написать пользовательский помощник HTML, который всегда будет использовать значение модели и игнорировать значения почты.
И еще одна возможность:
<input type="hidden" name="Step" value="<%: Model.Step %>" />
Другие советы
Я столкнулся с одной и той же проблемой при написании волшебника, который показывает разные части большей модели на каждом шаге.
Данные и/или ошибки из «Шага 1» будут смешиваться с «Шагом 2» и т. Д., Пока я наконец не понял, что Stastate будет «обвинять».
Это было мое простое решение:
if (oldPageIndex != newPageIndex)
{
ModelState.Clear(); // <-- solution
}
return View(model[newPageIndex]);
Этот код не будет работать
// remove the Step variable from the model state
// if you want the changes in the model to be
// taken into account
ModelState.Remove("Step");
model.Step = 2;
... потому что скрытый для всегда (!) читается из модели, а не самой модели. И если он не найдет ключ «шаг», он даст по умолчанию этот тип переменной, который будет 0 в этом случае
Вот решение. Я написал это для себя, но не против поделиться этим, потому что я вижу, что многие люди борются с этим непослушным, скрытым для помощника.
public static class CustomExtensions
{
public static MvcHtmlString HiddenFor2<TModel, TProperty>(this HtmlHelper<TModel> htmlHelper, Expression<Func<TModel, TProperty>> expression)
{
ReplacePropertyState(htmlHelper, expression);
return htmlHelper.HiddenFor(expression);
}
public static MvcHtmlString HiddenFor2<TModel, TProperty>(this HtmlHelper<TModel> htmlHelper, Expression<Func<TModel, TProperty>> expression, object htmlAttributes)
{
ReplacePropertyState(htmlHelper, expression);
return htmlHelper.HiddenFor(expression, htmlAttributes);
}
public static MvcHtmlString HiddenFor2<TModel, TProperty>(this HtmlHelper<TModel> htmlHelper, Expression<Func<TModel, TProperty>> expression, IDictionary<string, object> htmlAttributes)
{
ReplacePropertyState(htmlHelper, expression);
return htmlHelper.HiddenFor(expression, htmlAttributes);
}
private static void ReplacePropertyState<TModel, TProperty>(HtmlHelper<TModel> htmlHelper, Expression<Func<TModel, TProperty>> expression)
{
string text = ExpressionHelper.GetExpressionText(expression);
string fullName = htmlHelper.ViewContext.ViewData.TemplateInfo.GetFullHtmlFieldName(text);
ModelStateDictionary modelState = htmlHelper.ViewContext.ViewData.ModelState;
ModelMetadata metadata = ModelMetadata.FromLambdaExpression(expression, htmlHelper.ViewData);
if (modelState.ContainsKey(fullName))
{
ValueProviderResult currentValue = modelState[fullName].Value;
modelState[fullName].Value = new ValueProviderResult(metadata.Model, Convert.ToString(metadata.Model), currentValue.Culture);
}
else
{
modelState[fullName] = new ModelState
{
Value = new ValueProviderResult(metadata.Model, Convert.ToString(metadata.Model), CultureInfo.CurrentUICulture)
};
}
}
}
Тогда вы просто используете его, как обычно, изнутри, просмотр:
@Html.HiddenFor2(m => m.Id)
Стоит отметить, что это работает и с коллекциями.
Я слишком борюсь с той же ситуацией, где я использую одно и то же состояние модели между вызовами, и когда я изменяю свойство модели на бэкэнде. Хотя это не имеет значения для меня, если я использую Textboxfor или Hiddenfor.
Я просто обошел ситуацию, используя сценарии страниц для хранения значения модели в качестве переменной JS, потому что мне нужно вначале Hiddenfield для этой цели.
Не уверен, поможет ли это, но просто подумайте ..