1 /*
2 ********************************************************************************
3 *   Copyright (C) 2005-2011, International Business Machines
4 *   Corporation and others.  All Rights Reserved.
5 ********************************************************************************
6 *
7 * File WINDTTST.CPP
8 *
9 ********************************************************************************
10 */
11 
12 #include "unicode/utypes.h"
13 
14 #if U_PLATFORM_HAS_WIN32_API
15 
16 #if !UCONFIG_NO_FORMATTING
17 
18 #include "unicode/format.h"
19 #include "unicode/numfmt.h"
20 #include "unicode/locid.h"
21 #include "unicode/ustring.h"
22 #include "unicode/testlog.h"
23 #include "unicode/utmscale.h"
24 
25 #include "windtfmt.h"
26 #include "winutil.h"
27 #include "windttst.h"
28 
29 #include "cmemory.h"
30 #include "cstring.h"
31 #include "locmap.h"
32 #include "wintzimpl.h"
33 
34 #   define WIN32_LEAN_AND_MEAN
35 #   define VC_EXTRALEAN
36 #   define NOUSER
37 #   define NOSERVICE
38 #   define NOIME
39 #   define NOMCX
40 #   include <windows.h>
41 
42 #define ARRAY_SIZE(array) (sizeof array / sizeof array[0])
43 
getCalendarType(int32_t type)44 static const char *getCalendarType(int32_t type)
45 {
46     switch (type)
47     {
48     case 1:
49     case 2:
50         return "@calendar=gregorian";
51 
52     case 3:
53         return "@calendar=japanese";
54 
55     case 6:
56         return "@calendar=islamic";
57 
58     case 7:
59         return "@calendar=buddhist";
60 
61     case 8:
62         return "@calendar=hebrew";
63 
64     default:
65         return "";
66     }
67 }
68 
testLocales(TestLog * log)69 void Win32DateTimeTest::testLocales(TestLog *log)
70 {
71     SYSTEMTIME winNow;
72     UDate icuNow = 0;
73     SYSTEMTIME st;
74     FILETIME ft;
75     UnicodeString zoneID;
76     const TimeZone *tz = TimeZone::createDefault();
77     TIME_ZONE_INFORMATION tzi;
78 
79     tz->getID(zoneID);
80     if (! uprv_getWindowsTimeZoneInfo(&tzi, zoneID.getBuffer(), zoneID.length())) {
81         UBool found = FALSE;
82         int32_t ec = TimeZone::countEquivalentIDs(zoneID);
83 
84         for (int z = 0; z < ec; z += 1) {
85             UnicodeString equiv = TimeZone::getEquivalentID(zoneID, z);
86 
87             if (found = uprv_getWindowsTimeZoneInfo(&tzi, equiv.getBuffer(), equiv.length())) {
88                 break;
89             }
90         }
91 
92         if (! found) {
93             GetTimeZoneInformation(&tzi);
94         }
95     }
96 
97     GetSystemTime(&st);
98     SystemTimeToFileTime(&st, &ft);
99     SystemTimeToTzSpecificLocalTime(&tzi, &st, &winNow);
100 
101     int64_t wftNow = ((int64_t) ft.dwHighDateTime << 32) + ft.dwLowDateTime;
102     UErrorCode status = U_ZERO_ERROR;
103 
104     int64_t udtsNow = utmscale_fromInt64(wftNow, UDTS_WINDOWS_FILE_TIME, &status);
105 
106     icuNow = (UDate) utmscale_toInt64(udtsNow, UDTS_ICU4C_TIME, &status);
107 
108     int32_t lcidCount = 0;
109     Win32Utilities::LCIDRecord *lcidRecords = Win32Utilities::getLocales(lcidCount);
110 
111     for(int i = 0; i < lcidCount; i += 1) {
112         UErrorCode status = U_ZERO_ERROR;
113         WCHAR longDateFormat[81], longTimeFormat[81], wdBuffer[256], wtBuffer[256];
114         int32_t calType = 0;
115 
116         // NULL localeID means ICU didn't recognize this locale
117         if (lcidRecords[i].localeID == NULL) {
118             continue;
119         }
120 
121         GetLocaleInfoW(lcidRecords[i].lcid, LOCALE_SLONGDATE,   longDateFormat, 81);
122         GetLocaleInfoW(lcidRecords[i].lcid, LOCALE_STIMEFORMAT, longTimeFormat, 81);
123         GetLocaleInfoW(lcidRecords[i].lcid, LOCALE_RETURN_NUMBER|LOCALE_ICALENDARTYPE, (LPWSTR) calType, sizeof(int32_t));
124 
125         char localeID[64];
126 
127         uprv_strcpy(localeID, lcidRecords[i].localeID);
128         uprv_strcat(localeID, getCalendarType(calType));
129 
130         UnicodeString ubBuffer, udBuffer, utBuffer;
131         Locale ulocale(localeID);
132         int32_t wdLength, wtLength;
133 
134         wdLength = GetDateFormatW(lcidRecords[i].lcid, DATE_LONGDATE, &winNow, NULL, wdBuffer, ARRAY_SIZE(wdBuffer));
135         wtLength = GetTimeFormatW(lcidRecords[i].lcid, 0, &winNow, NULL, wtBuffer, ARRAY_SIZE(wtBuffer));
136 
137         if (uprv_strchr(localeID, '@') > 0) {
138             uprv_strcat(localeID, ";");
139         } else {
140             uprv_strcat(localeID, "@");
141         }
142 
143         uprv_strcat(localeID, "compat=host");
144 
145         Locale wlocale(localeID);
146         DateFormat *wbf = DateFormat::createDateTimeInstance(DateFormat::kFull, DateFormat::kFull, wlocale);
147         DateFormat *wdf = DateFormat::createDateInstance(DateFormat::kFull, wlocale);
148         DateFormat *wtf = DateFormat::createTimeInstance(DateFormat::kFull, wlocale);
149 
150         wbf->format(icuNow, ubBuffer);
151         wdf->format(icuNow, udBuffer);
152         wtf->format(icuNow, utBuffer);
153 
154         if (ubBuffer.indexOf(wdBuffer, wdLength - 1, 0) < 0) {
155             UnicodeString baseName(wlocale.getBaseName());
156             UnicodeString expected(wdBuffer);
157 
158             log->errln("DateTime format error for locale " + baseName + ": expected date \"" + expected +
159                        "\" got \"" + ubBuffer + "\"");
160         }
161 
162         if (ubBuffer.indexOf(wtBuffer, wtLength - 1, 0) < 0) {
163             UnicodeString baseName(wlocale.getBaseName());
164             UnicodeString expected(wtBuffer);
165 
166             log->errln("DateTime format error for locale " + baseName + ": expected time \"" + expected +
167                        "\" got \"" + ubBuffer + "\"");
168         }
169 
170         if (udBuffer.compare(wdBuffer) != 0) {
171             UnicodeString baseName(wlocale.getBaseName());
172             UnicodeString expected(wdBuffer);
173 
174             log->errln("Date format error for locale " + baseName + ": expected \"" + expected +
175                        "\" got \"" + udBuffer + "\"");
176         }
177 
178         if (utBuffer.compare(wtBuffer) != 0) {
179             UnicodeString baseName(wlocale.getBaseName());
180             UnicodeString expected(wtBuffer);
181 
182             log->errln("Time format error for locale " + baseName + ": expected \"" + expected +
183                        "\" got \"" + utBuffer + "\"");
184         }
185         delete wbf;
186         delete wdf;
187         delete wtf;
188     }
189 
190     Win32Utilities::freeLocales(lcidRecords);
191     delete tz;
192 }
193 
194 #endif /* #if !UCONFIG_NO_FORMATTING */
195 
196 #endif /* U_PLATFORM_HAS_WIN32_API */
197