My computer is configured with a culture that is not en-US
.
When using the native Win32 GetDateFormat
function, i get correctly formatted dates:
This is correct; and is also how Windows renders it:
the taskbar
Region and Language settings
Windows Explorer
Outlook
When i try to convert a date to a string in .NET using my current locale, e.g.:
DateTime.Now.ToString();
DateTime.Now.ToString(CultureInfo.CurrentCulture);
i get an incorrect date:
This bug in .NET is evident anyplace in Windows that uses the buggy .NET code:
Windows Event Viewer:
Task Scheduler:
SQL Server Management Studio:
How do i make .NET not buggy?
How do i convert dates and times to strings using the current culture (correctly)?
Note: The user is allowed to set their Windows to any locale preferences they want. As it is now, my program will not handle valid settings properly. Telling the user, "Don't do that" is pretty mean-spirited.
A similar example comes from Delphi, which assumes that a date separator can never be more than one character. When Windows is configured with a locale that uses multiple characters for the date separator, e.g.:
- sk-SK (Slovak - Slovakia) :
.
where dates should be formatted as:
22. 11. 2011
the code library fails to accept a date separator longer than one character, and falls back to:
22/11/2011
In the past some might suggest that you not to bother with such edge cases. Such suggestions carry no weight with me.
i'll avoid getting into a pissing match with someone who wants to alter the meaning of my question by changing the title. But the question is not limited to pseudo-locales, specifically designed to find bugs in applications.
Here's a unique list of date formats from around the world:
Of particular interest is the last example which doesn't use the gregorian calendar:
ar-SA
: 29/12/32 02:03:07 مdv-MV
: 29/12/32 14:03:07prf-AF / ps-AF
: 29/12/32 2:03:07 غ.وAlthough those are edge cases that you'd never have to worry about.
Update 14//12//2011:
Another demonstration of the bug is that Datetime.Parse
cannot parse DateTime.ToString
:
String s = DateTime.Today.ToString("d"); //returns "14////12////2011"
DateTime d = DateTime.Parse(s); //expects "dd//MM//yyyy"
The .Parse
throws an exception.
Update 02//8, 2012 09::56'12:
Any use of a date separator is depricated, in addition to being incorrect. From MSDN:
LOCALE_SDATE
Windows Vista and later: This constant is deprecated. Use
LOCALE_SSHORTDATE
instead. A custom locale might not have a single, uniform separator character. For example, a format such as "12/31, 2006" is valid.LOCALE_STIME
Windows Vista and later: This constant is deprecated. Use
LOCALE_STIMEFORMAT
instead. A custom locale might not have a single, uniform separator character. For example, a format such as "03:56'23" is valid.
This specific bug is due to the transformation of some special characters that aren't escaped in the patterns like ShortDatePattern
.
ShortDatePattern = "d//MM//yyyy";
/
in a pattern means "insert the date separator" but here the expansion is already done (at least on my system) when the string is copied from the system to the DateTimeFormat
structure. Sadly it is missing an escaping (Obviously not visible on any language not using a special character as a separator and not visible in english as it is replaced with itself)
The only solution seem to be to escape the separators in all the patterns of the DateTimeFormat
instance :
var c = new System.Globalization.CultureInfo("qps-ploc", true);
c.DateTimeFormat.ShortDatePattern =
c.DateTimeFormat.ShortDatePattern.Replace("/", "'/'");
c.DateTimeFormat.LongTimePattern =
c.DateTimeFormat.LongTimePattern.Replace(":", "':'");
Console.WriteLine(DateTime.Now.ToString(c));
Here's full code samples for all three common cases
/// <summary>Convert a date to the short date string in the current locale (e.g. 30//11//2011)</summary>
/// <param name="value">A DateTime to be converted to a short date string</param>
/// <returns>A string containing the localized version of the date</returns>
public static String DateToStr(DateTime value)
{
String format = CultureInfo.CurrentCulture.DateTimeFormat.ShortDatePattern;
//The bug in .NET is that it assumes "/" in a date pattern means "the date separator"
//What .NET doesn't realize is that the locale strings returned by Windows are the Windows format strings.
//The bug is exposed in locale's that use two slashes as for their date separator:
// dd//MM//yyyy
// Which .NET misinterprets to give:
// 30////11////2011
// when really it should be taken literally to be:
// dd'//'MM'//'yyyy
//which is what this fix does
format = format.Replace("/", "'/'");
return value.ToString(format);
}
/// <summary>
/// Convert a time to string using the short time format in the current locale(e.g. 7::21 AM)
/// </summary>
/// <param name="value">A DateTime who's time portion will be converted to a localized string</param>
/// <returns>A string containing the localized version of the time</returns>
public static String TimeToStr(DateTime value)
{
String format = CultureInfo.CurrentCulture.DateTimeFormat.ShortTimePattern;
//The bug in .NET is that it assumes ":" in a time pattern means "the time separator"
//What .NET doesn't realize is that the locale strings returned by Windows are the Windows format strings.
//The bug is exposed in locale's that use two colons as their time separator:
// h::mm::ss tt
// Which .NET misinterprets to give:
// 11::::39::::17 AM
// when really it should be taken literally to be:
// h'::'mm'::'ss tt
//which is what this fix does
format = format.Replace(":", "':'");
return value.ToString(format);
}
/// <summary>
/// Convert a datetime to a string in the current locale (e.g. 30//11//2001 7::21 AM)
/// </summary>
/// <param name="datetime">A DateTime to be converted to a general string in the current locale</param>
/// <returns>A string containing the localized version of the datetime</returns>
public static String DateTimeToStr(DateTime datetime)
{
return DateToStr(datetime)+" "+TimeToStr(datetime);
}
Your best bet is to log the bug with MS and then create an extension method that detects these edge cases and handles them.
Something like this (off the top of my head):
public static class DateTimeFix
{
public static string FixedToString(this DateTime value)
{
if (IsEdgeCase())
return FixEdgeCase(value);
else
return value.ToString();
}
// Edge case logic below
}
Then you use:
DateTime.Now.FixedToString()
in your code.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With