MVC DateTime привязка с неправильным форматом даты - PullRequest
126 голосов
/ 09 февраля 2009

Asp.net-MVC теперь позволяет неявно связывать объекты DateTime. У меня есть действие по линии

public ActionResult DoSomething(DateTime startDate) 
{ 
... 
}

Это успешно преобразует строку из вызова ajax в DateTime. Однако мы используем формат даты dd / MM / yyyy; MVC конвертируется в MM / дд / гггг. Например, отправка вызова к действию со строкой '09 / 02/2009 'приводит к DateTime '02 / 09/2009 00:00:00' или 2 сентября в наших локальных настройках.

Я не хочу закатывать свою собственную модель для подшивки ради формата даты. Но, похоже, нет необходимости менять действие, чтобы принимать строку, а затем использовать DateTime.Parse, если MVC способен сделать это для меня.

Есть ли способ изменить формат даты, используемый в связывателе модели по умолчанию для DateTime? Разве связыватель модели по умолчанию не должен использовать ваши настройки локализации в любом случае?

Ответы [ 10 ]

161 голосов
/ 09 февраля 2009

Я только что нашел ответ на этот вопрос с более исчерпывающим поиском:

Мелвин Харбор подробно объясняет, почему MVC работает с датами так, как работает, и как вы можете переопределить это при необходимости:

http://weblogs.asp.net/melvynharbour/archive/2008/11/21/mvc-modelbinder-and-localization.aspx

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

  1. RouteData (не показано выше)
  2. Строка запроса URI
  3. Форма запроса

Однако только последний из них будет осведомлен о культуре. Для этого есть очень веская причина с точки зрения локализации. Представьте, что я написал веб-приложение, отображающее информацию о рейсе авиакомпании, которое я публикую в Интернете. Я просматриваю рейсы на определенную дату, нажимая на ссылку на этот день (возможно, что-то вроде http://www.melsflighttimes.com/Flights/2008-11-21),, а затем хочу отправить эту ссылку по электронной почте моему коллеге в США. Единственный способ, которым мы можем гарантировать, что мы будем оба будут смотреть на одну и ту же страницу данных, если используется InvariantCulture. Напротив, если я использую форму для бронирования своего рейса, все происходит в узком цикле. Данные могут соответствовать CurrentCulture, когда они записываются в форма, и поэтому необходимо уважать ее, возвращаясь из формы.

35 голосов
/ 24 мая 2010

Я бы глобально установил ваши культуры. ModelBinder подними это!

  <system.web>
    <globalization uiCulture="en-AU" culture="en-AU" />

Или вы просто измените это для этой страницы.
Но глобально в web.config я думаю, что лучше

29 голосов
/ 15 июля 2013

У меня возникла та же проблема с привязкой формата даты к свойствам модели DateTime. Посмотрев на множество разных примеров (не только касающихся DateTime), я собрал следующее:

using System;
using System.Globalization;
using System.Web.Mvc;

namespace YourNamespaceHere
{
    public class CustomDateBinder : IModelBinder
    {
        public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
        {
            if (controllerContext == null)
                throw new ArgumentNullException("controllerContext", "controllerContext is null.");
            if (bindingContext == null)
                throw new ArgumentNullException("bindingContext", "bindingContext is null.");

            var value = bindingContext.ValueProvider.GetValue(bindingContext.ModelName);

            if (value == null)
                throw new ArgumentNullException(bindingContext.ModelName);

            CultureInfo cultureInf = (CultureInfo)CultureInfo.CurrentCulture.Clone();
            cultureInf.DateTimeFormat.ShortDatePattern = "dd/MM/yyyy";

            bindingContext.ModelState.SetModelValue(bindingContext.ModelName, value);

            try
            {
                var date = value.ConvertTo(typeof(DateTime), cultureInf);

                return date;
            }
            catch (Exception ex)
            {
                bindingContext.ModelState.AddModelError(bindingContext.ModelName, ex);
                return null;
            }
        }
    }

    public class NullableCustomDateBinder : IModelBinder
    {
        public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
        {
            if (controllerContext == null)
                throw new ArgumentNullException("controllerContext", "controllerContext is null.");
            if (bindingContext == null)
                throw new ArgumentNullException("bindingContext", "bindingContext is null.");

            var value = bindingContext.ValueProvider.GetValue(bindingContext.ModelName);

            if (value == null) return null;

            CultureInfo cultureInf = (CultureInfo)CultureInfo.CurrentCulture.Clone();
            cultureInf.DateTimeFormat.ShortDatePattern = "dd/MM/yyyy";

            bindingContext.ModelState.SetModelValue(bindingContext.ModelName, value);

            try
            {
                var date = value.ConvertTo(typeof(DateTime), cultureInf);

                return date;
            }
            catch (Exception ex)
            {
                bindingContext.ModelState.AddModelError(bindingContext.ModelName, ex);
                return null;
            }
        }
    }
}

Чтобы сохранить способ, которым маршруты и т. Д. Регистрируются в файле Global ASAX, я также добавил новый класс sytatic в папку App_Start моего проекта MVC4 с именем CustomModelBinderConfig:

using System;
using System.Web.Mvc;

namespace YourNamespaceHere
{
    public static class CustomModelBindersConfig
    {
        public static void RegisterCustomModelBinders()
        {
            ModelBinders.Binders.Add(typeof(DateTime), new CustomModelBinders.CustomDateBinder());
            ModelBinders.Binders.Add(typeof(DateTime?), new CustomModelBinders.NullableCustomDateBinder());
        }
    }
}

Затем я просто вызываю статический RegisterCustomModelBinders из моего Global ASASX Application_Start следующим образом:

protected void Application_Start()
{
    /* bla blah bla the usual stuff and then */

    CustomModelBindersConfig.RegisterCustomModelBinders();
}

Важным примечанием здесь является то, что если вы записываете значение DateTime в скрытое поле следующим образом:

@Html.HiddenFor(model => model.SomeDate) // a DateTime property
@Html.Hiddenfor(model => model) // a model that is of type DateTime

Я сделал это, и фактическое значение на странице было в формате «ММ / дд / гггг чч: мм: сс тт» вместо «дд / мм / гггг чч: мм: сс тт», как я и хотел. Это привело к тому, что моя проверка модели либо не прошла, либо вернула неправильную дату (очевидно, меняя местами значения дня и месяца).

После многих попыток поцарапать голову и потерпеть неудачу, было решено установить информацию о культуре для каждого запроса, выполнив это в Global.ASAX:

protected void Application_BeginRequest()
{
    CultureInfo cInf = new CultureInfo("en-ZA", false);  
    // NOTE: change the culture name en-ZA to whatever culture suits your needs

    cInf.DateTimeFormat.DateSeparator = "/";
    cInf.DateTimeFormat.ShortDatePattern = "dd/MM/yyyy";
    cInf.DateTimeFormat.LongDatePattern = "dd/MM/yyyy hh:mm:ss tt";

    System.Threading.Thread.CurrentThread.CurrentCulture = cInf;
    System.Threading.Thread.CurrentThread.CurrentUICulture = cInf;
}

Это не сработает, если вы вставите его в Application_Start или даже Session_Start, поскольку это назначит его текущему потоку для сеанса. Как вы хорошо знаете, веб-приложения не имеют состояния, поэтому поток, который обслуживал ваш запрос ранее, не является тем же потоком, который обслуживает ваш текущий запрос, следовательно, информация о вашей культуре перешла в большой GC в цифровом небе.

Спасибо! Иван Златев - http://ivanz.com/2010/11/03/custom-model-binding-using-imodelbinder-in-asp-net-mvc-two-gotchas/

Гарик - https://stackoverflow.com/a/2468447/578208

Дмитрий - https://stackoverflow.com/a/11903896/578208

13 голосов
/ 10 августа 2012

В MVC 3 будет немного по-другому.

Предположим, у нас есть контроллер и представление с методом Get

public ActionResult DoSomething(DateTime dateTime)
{
    return View();
}

Мы должны добавить ModelBinder

public class DateTimeBinder : IModelBinder
{
    #region IModelBinder Members
    public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        DateTime dateTime;
        if (DateTime.TryParse(controllerContext.HttpContext.Request.QueryString["dateTime"], CultureInfo.GetCultureInfo("en-GB"), DateTimeStyles.None, out dateTime))
            return dateTime;
        //else
        return new DateTime();//or another appropriate default ;
    }
    #endregion
}

и команда в Application_Start () из Global.asax

ModelBinders.Binders.Add(typeof(DateTime), new DateTimeBinder());
8 голосов
/ 15 февраля 2009

Стоит также отметить, что даже без создания вашей собственной подшивки модели может выполняться несколько различных форматов.

Например, в США все следующие строки эквивалентны и автоматически связываются с тем же значением DateTime:

/ компании / Пресс / май% 2001% 202008

/ Компания / Пресс / 2008-05-01

/ компании / Пресс / 05.01.2008

Я бы настоятельно рекомендовал использовать yyyy-mm-dd, потому что он намного более переносим. Вы действительно не хотите иметь дело с обработкой нескольких локализованных форматов. Если кто-то забронирует рейс 1 мая вместо 5 января, у вас будут большие проблемы!

NB. Я не совсем точно понимаю, если yyyy-mm-dd универсально анализируется во всех культурах, поэтому, возможно, кто-то, кто знает, может добавить комментарий.

5 голосов
/ 29 января 2015

Попробуйте использовать toISOString (). Возвращает строку в формате ISO8601.

Метод GET

Javascript

$.get('/example/doGet?date=' + new Date().toISOString(), function (result) {
    console.log(result);
});

с #

[HttpGet]
public JsonResult DoGet(DateTime date)
{
    return Json(date.ToString(), JsonRequestBehavior.AllowGet);
}

Метод POST

Javascript

$.post('/example/do', { date: date.toISOString() }, function (result) {
    console.log(result);
});

с #

[HttpPost]
public JsonResult Do(DateTime date)
{
     return Json(date.ToString());
}
5 голосов
/ 10 января 2013

Я установил приведенную ниже конфигурацию на моем MVC4, и она работает как шарм

<globalization uiCulture="auto" culture="auto" />
1 голос
/ 31 мая 2013
public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
{
    var str = controllerContext.HttpContext.Request.QueryString[bindingContext.ModelName];
    if (string.IsNullOrEmpty(str)) return null;
    var date = DateTime.ParseExact(str, "dd.MM.yyyy", null);
    return date;
}
1 голос
/ 30 января 2013
  public class DateTimeFilter : ActionFilterAttribute
{
    public override void OnActionExecuting(ActionExecutingContext filterContext)
    {
        if (filterContext.HttpContext.Request.RequestType == "GET")
        {

            foreach (var parameter in filterContext.ActionParameters)
            {
                var properties = parameter.Value.GetType().GetProperties();

                foreach (var property in properties)
                {
                    Type type = Nullable.GetUnderlyingType(property.PropertyType) ?? property.PropertyType;

                    if (property.PropertyType == typeof(System.DateTime) || property.PropertyType == typeof(DateTime?))
                    {
                        DateTime dateTime;

                        if (DateTime.TryParse(filterContext.HttpContext.Request.QueryString[property.Name], CultureInfo.CurrentUICulture, DateTimeStyles.None, out dateTime))
                            property.SetValue(parameter.Value, dateTime,null);
                    }
                }

            }
        }
    }
}
0 голосов
/ 28 августа 2016

Я установил CurrentCulture и CurrentUICulture мой пользовательский базовый контроллер

    protected override void Initialize(RequestContext requestContext)
    {
        base.Initialize(requestContext);

        Thread.CurrentThread.CurrentCulture = CultureInfo.GetCultureInfo("en-GB");
        Thread.CurrentThread.CurrentUICulture = CultureInfo.GetCultureInfo("en-GB");
    }
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...