ASP.NET MVC проблема культуры времени и даты при передаче значения обратно в контроллер - PullRequest
18 голосов
/ 25 октября 2011

Как я могу сказать своему контроллеру / модели, какую культуру следует ожидать для анализа даты и времени?

Я использовал некоторые из этого поста для реализации jquery datepicker в моем приложении mvc.

Когда я отправляю дату, она «теряется в переводе», я не использую форматирование даты в США, поэтому, когда оно отправляется на мой контроллер, оно просто становится нулевым.

У меня есть форма, в которой пользователь выбирает дату:

@using (Html.BeginForm("List", "Meter", FormMethod.Get))
{
    @Html.LabelFor(m => m.StartDate, "From:")
    <div>@Html.EditorFor(m => m.StartDate)</div>

    @Html.LabelFor(m => m.EndDate, "To:")
    <div>@Html.EditorFor(m => m.EndDate)</div>
}

Я сделал для этого шаблон редактирования, чтобы реализовать Jickery DatePicker:

@model DateTime
@Html.TextBox("", Model.ToString("dd-MM-yyyy"), new { @class = "date" }) 

Затем я создаю виджеты DatePicker, как это.

$(document).ready(function () {
    $('.date').datepicker({ dateFormat: "dd-mm-yy" });
});

Все это прекрасно работает.

Вот где начинаются проблемы, это мой контроллер:

[HttpGet]
public ActionResult List(DateTime? startDate = null, DateTime? endDate = null)
{
    //This is where startDate and endDate becomes null if the dates dont have the expected formatting.
}

Вот почему я хотел бы как-то сказать своему контролеру, какую культуру ему следует ожидать? Моя модель не так? Могу ли я как-то сказать, какую культуру использовать, например, с атрибутами аннотации данных?

public class MeterViewModel {
    [Required]
    public DateTime StartDate { get; set; }
    [Required]
    public DateTime EndDate { get; set; }
}

Редактировать: эта ссылка объясняет мою проблему и очень хорошее решение, а также. Благодаря Гдорон

Ответы [ 6 ]

20 голосов
/ 07 ноября 2011

вы можете изменить связыватель модели по умолчанию для использования пользовательской культуры с помощью IModelBinder

   public class DateTimeBinder : IModelBinder
   {
       public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
       {
           var value = bindingContext.ValueProvider.GetValue(bindingContext.ModelName);
           var date = value.ConvertTo(typeof(DateTime), CultureInfo.CurrentCulture);

           return date;    
       }
   }

А в Global.Asax напишите:

ModelBinders.Binders.Add(typeof(DateTime), new DateTimeBinder());
ModelBinders.Binders.Add(typeof(DateTime?), new DateTimeBinder());

Подробнее на thisотличный блог , в котором рассказывается, почему команда разработчиков Mvc Framework применила стандартную культуру для всех пользователей.

12 голосов
/ 25 октября 2011

Вы можете создать расширение Binder для обработки даты в культурном формате.

Это пример, который я написал для решения той же проблемы с типом Decimal, надеюсь, вы поняли идею

 public class DecimalModelBinder : IModelBinder
 {
   public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
   {
     ValueProviderResult valueResult = bindingContext.ValueProvider.GetValue(bindingContext.ModelName);
     ModelState modelState = new ModelState { Value = valueResult };
     object actualValue = null;
     try
     {
       actualValue = Convert.ToDecimal(valueResult.AttemptedValue, CultureInfo.CurrentCulture);
     }
     catch (FormatException e)
     {
       modelState.Errors.Add(e);
     }

     bindingContext.ModelState.Add(bindingContext.ModelName, modelState);
     return actualValue;
  }
}

Обновление

Чтобы использовать его, просто объявите связыватель в Global.asax следующим образом:

protected void Application_Start()
{
  AreaRegistration.RegisterAllAreas();
  RegisterGlobalFilters(GlobalFilters.Filters);
  RegisterRoutes(RouteTable.Routes);

  //HERE you tell the framework how to handle decimal values
  ModelBinders.Binders.Add(typeof(decimal), new DecimalModelBinder());

  DependencyResolver.SetResolver(new ETAutofacDependencyResolver());
}

Затем, когда связыватель модели выполнит некоторую работу, он будетзнать автоматически, что делать.Например, это действие с моделью, содержащей некоторые свойства типа decimal.Я просто ничего не делаю

[HttpPost]
public ActionResult Edit(int id, MyViewModel viewModel)
{
  if (ModelState.IsValid)
  {
    try
    {
      var model = new MyDomainModelEntity();
      model.DecimalValue = viewModel.DecimalValue;
      repository.Save(model);
      return RedirectToAction("Index");
    }
    catch (RulesException ex)
    {
      ex.CopyTo(ModelState);
    }
    catch
    {
      ModelState.AddModelError("", "My generic error message");
    }
  }
  return View(model);
}
10 голосов
/ 28 октября 2011

Эта проблема возникает потому, что вы используете метод GET в вашей форме.Поставщик значений QueryString в MVC всегда использует формат даты Инвариант / США.См .: MVC DateTime привязка с неправильным форматом даты

Есть три решения:

  1. Измените свой метод на POST.
  2. Как кто-то ещеговорит, измените формат даты на ISO 8601 "гггг-мм-дд" перед отправкой.
  3. Используйте пользовательское связующее, чтобы всегда обрабатывать даты строки запроса как ГБ.Если вы делаете это, вы должны убедиться, что все даты в этой форме:

    public class UKDateTimeModelBinder : IModelBinder
    {
    private static readonly ILog logger = LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
    
    /// <summary>
    /// Fixes date parsing issue when using GET method. Modified from the answer given here:
    /// /626319/mvc-datetime-privyazka-s-nepravilnym-formatom-daty
    /// </summary>
    /// <param name="controllerContext">The controller context.</param>
    /// <param name="bindingContext">The binding context.</param>
    /// <returns>
    /// The converted bound value or null if the raw value is null or empty or cannot be parsed.
    /// </returns>
    public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        var vpr = bindingContext.ValueProvider.GetValue(bindingContext.ModelName);
    
        if (vpr == null)
        {
            return null;
    
        }
    
        var date = vpr.AttemptedValue;
    
        if (String.IsNullOrEmpty(date))
        {
            return null;
        }
    
        logger.DebugFormat("Parsing bound date '{0}' as UK format.", date);
    
        // Set the ModelState to the first attempted value before we have converted the date. This is to ensure that the ModelState has
        // a value. When we have converted it, we will override it with a full universal date.
        bindingContext.ModelState.SetModelValue(bindingContext.ModelName, bindingContext.ValueProvider.GetValue(bindingContext.ModelName));
    
        try
        {
            var realDate = DateTime.Parse(date, System.Globalization.CultureInfo.GetCultureInfoByIetfLanguageTag("en-GB"));
    
            // Now set the ModelState value to a full value so that it can always be parsed using InvarianCulture, which is the
            // default for QueryStringValueProvider.
            bindingContext.ModelState.SetModelValue(bindingContext.ModelName, new ValueProviderResult(date, realDate.ToString("yyyy-MM-dd hh:mm:ss"), System.Globalization.CultureInfo.GetCultureInfoByIetfLanguageTag("en-GB")));
    
            return realDate;
        }
        catch (Exception)
        {
            logger.ErrorFormat("Error parsing bound date '{0}' as UK format.", date);
    
            bindingContext.ModelState.AddModelError(bindingContext.ModelName, String.Format("\"{0}\" is invalid.", bindingContext.ModelName));
            return null;
        }
    }
    }
    
3 голосов
/ 25 октября 2011

При отправке даты вы всегда должны пытаться отправить ее в формате «гггг-мм-дд».Это позволит ему стать независимым от культуры.

У меня обычно есть скрытое поле, которое поддерживает дату в этом формате.Это сравнительно просто с помощью выбора даты в jQuery UI.

1 голос
/ 26 октября 2013

Почему бы просто не проверить культуру данных и не преобразовать ее как таковую?Этот простой подход позволил мне использовать строго типизированные даты в моделях, показывать ссылки на действия и редактировать поля в нужной локали, и вообще не нужно было суетиться, связывая их обратно в строго типизированный DateTime:

public class DateTimeBinder : IModelBinder
{
    public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        var value = bindingContext.ValueProvider.GetValue(bindingContext.ModelName);
        return value.ConvertTo(typeof(DateTime), value.Culture);
    }
}
0 голосов
/ 23 октября 2012

, который сделал мне трюк

    <system.web>     
       <globalization enableClientBasedCulture="true" uiCulture="Auto" culture="Auto" />
    </system.web>
...