Подтвердить что ты не робот

MVC 4 @HTML.HiddenFor не обновляется после обратной передачи

У вас возникли проблемы с состоянием представления в серии просмотров страниц. На начальном представлении страницы в Razor я использую Html.HiddenFor следующим образом:

    @Html.HiddenFor(x => Model.err)
    @Html.HiddenFor(x => Model.errField)
    @Html.HiddenFor(x => Model.errMessage)
    @Html.HiddenFor(x => Model.IsMove)

который работает нормально. Мои скрытые теги ввода содержат правильные значения. Однако, когда я отправляю форму [HTTPPost] и обновляю модель в своем действии с контроллером с помощью.

       model.err = transHelper.err;
       model.errField = transHelper.errField;
       model.errMessage = transHelper.errMessage;
       return View(model);

Скрытые поля, похоже, не обновляются, они содержат исходные значения из исходного представления. Однако, когда я использую эти поля в другом контексте в том же виде бритвы, как это...

     @*      
        this seems to not update correctly...

    @Html.HiddenFor(x => Model.err)
    @Html.HiddenFor(x => Model.errField)
    @Html.HiddenFor(x => Model.errMessage)
    @Html.HiddenFor(x => Model.IsMove)

        *@
        <input type="hidden" id="err" value="@Model.err" />
        <input type="hidden" id="errField" value="@Model.errField" />
        <input type="hidden" id="errMessage" value="@Model.errMessage" />
        <input type="hidden" id="IsMove" value="@Model.IsMove" />

    </div>

Затем поля ввода обновляются правильно. Я даже создал помощник представления, чтобы помочь отлаживать, и во всех случаях у модели, кажется, есть правильные данные в HtmlHelper<TModel> - я даже вернул модель как return Json(model);, и данные были в порядке.

В этот момент я работаю с работой, но кто-нибудь знает, почему @Html.HiddenFor является грязным.

Обновление: вот мои действия с контроллером

  [HttpPost]
   public ActionResult Index(HomePageModel model)
   {


       // process transaction
       Transactionr transr = new Transactionr();
       transr.Process(model);

       model.err = transr.err;
       model.errField = transr.errField;
       model.errMessage = transr.errMessage;

       return View(model);
   }

Вот мой взгляд:

        @model App.Models.HomePageModel
    @{
        ViewBag.Title = "Product Categorizer";
    }
    <form id="formData" method="post" action="/Home/Index">
        @Html.AntiForgeryToken()
        <fieldset>
            <div>

            @Html.HiddenFor(model => model.err)
            @Html.HiddenFor(model => model.errField)
            @Html.HiddenFor(model => model.errMessage)
            @Html.HiddenFor(model => model.IsMove)

            <input type="hidden" id="myerr" value="@Model.err" />
            <input type="hidden" id="myerrField" value="@Model.errField" />

            </div>

           <div class="section group">
                <div class="col span_2_of_2">
                     <div class="message" id ="message">
                         @if (Model.err < 0)
                         {
                             <span style="color: purple;">@Model.errMessage (@Model.err) - (@Model.errField)</span>
                         }
                         else if (Model.err > 0)
                         {
                             <span style="color:red;">@Model.errMessage (@Model.err) (@Model.errField)</span>
                         } else {
                            <span>@Model.errMessage (@Model.err) (@Model.errField)</span>
                         }
                         </div>
                     </div>
            </div>

            <div class="section group" id="workspace">
                  @Html.Partial("_WorkspacePartial", Model)
            </div>
              <div class="section group" id="details">
                  @Html.Partial("_DetailPartial", Model)
              </div>


        </fieldset>
        </form>

Вот моя модель:

 public class HomePageModel
 {
    public int FromStore { get; set; }

    //  the "To" part of the copy/move transaction
    public int ToStore { get; set; }

    // a list of the copy/move transaction
    public List<int> Details { get; set; }


    // true is move false is copy
    public bool IsMove { get; set; }

    // current message
    public int err { get; set; }
    public int errField { get; set; }
    public string errMessage { get; set; }
4b9b3361

Ответ 1

Поведение HtmlHelpers по умолчанию (@Html.HiddenFor и т.д.) должно вести себя точно так, как вы описали.

то есть. любые изменения, которые вы делаете в ViewModel в сообщении, выполняются, любые изменения, которые вы возвращаете из Почты, принимаются в представлении, но при повторном рендеринге с помощью HTMLHELPERS предыдущие значения Post имеют приоритет над измененными значениями ViewModel.

Хотите "исправить" это поведение быстрым и грязным способом, очистите ModelState.Clear() перед возвратом из HttpPost ActionMethod!

Ответ 2

Как упоминалось joedotnot, это предполагаемое поведение. Другим "быстрым решением" для этого является кодирование html для скрытого поля и обновление только значения из модели, например:

<input type="hidden" id="ErrMessage" name="ErrMessage" value="@Model.ErrMessage">

Используйте те же id и name как свойство модели, и обновленное значение будет отображаться после обратной передачи.

Ответ 3

Я столкнулся с подобной проблемой в последнее время и закончил тем, что написал новый простой хелперный метод + 2 перегрузки. Я разделяю его здесь, если кто-то все еще ищет какое-то обходное решение, потому что эта "функция" иногда раздражает.

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;

        if (modelState.ContainsKey(fullName))
        {
            ModelMetadata metadata = ModelMetadata.FromLambdaExpression(expression, htmlHelper.ViewData);
            ValueProviderResult currentValue = modelState[fullName].Value;
            modelState[fullName].Value = new ValueProviderResult(metadata.Model, Convert.ToString(metadata.Model), currentValue.Culture);
        }
    }
}

Затем вы просто используете его как обычно изнутри:

@Html.HiddenFor2(m => m.Id)

Стоит упомянуть, что он также работает с коллекциями.

Ответ 4

Я думаю, вы должны использовать их вот так:

@Html.HiddenFor(x => x.Err)
@Html.HiddenFor(x => x.ErrField)
@Html.HiddenFor(x => x.ErrMessage)
@Html.HiddenFor(x => x.IsMove)

Не видя свою модель, я предполагаю, что она выглядит примерно так:

public class ErroViewModel
{
  public string Err { get; set; }
  public string ErrField { get; set; }
  public string ErrMessage { get; set; }
  public bool IsMove { get; set; }
}

Если это не должно быть похоже на общедоступные свойства, как указано выше.

Обновление

У вас есть следующее:

public ActionResult Index(HomePageModel model)
{
   var model = new HomePageModel();
   return View(model);
}

Я бы тоже изменил вашу форму:

 <form id="formData" method="post" action="/Home/Index">

Для этого:

@using (Html.BeginForm("Index", "Home", FormMethod.Post))
{
  // rest of form
}

Ответ 5

Вы можете попробовать

<input type="hidden" id="SomeFieldID" name="SomeFieldID" value="@Model.SomeFieldID" />

Ответ 6

У меня была аналогичная проблема и она была решена так.

@Html.TextBoxFor(m => m.Email, new { onclick = "this.select()", Value = Model.Email, Placeholder = "E-Mail" })

Ответ 7

Вам нужен ключ в вашей модели просмотра. Если у вас нет какого-либо свойства с именем Id в вашей модели просмотра, установите один из них (который должен быть uniqe identifiyer) в качестве ключа с [Key].

[Key]
Myproperty {get;set;}