// 22 may 2015 #include "uipriv_windows.hpp" struct uiDateTimePicker { uiWindowsControl c; HWND hwnd; void(*onChanged)(uiDateTimePicker *, void *); void *onChangedData; }; // utility functions #define GLI(what, buf, n) GetLocaleInfoEx(LOCALE_NAME_USER_DEFAULT, what, buf, n) // The real date/time picker does a manual replacement of "yy" with "yyyy" for DTS_SHORTDATECENTURYFORMAT. // Because we're also duplicating its functionality (see below), we have to do it too. static WCHAR *expandYear(WCHAR *dts, int n) { WCHAR *out; WCHAR *p, *q; int ny = 0; // allocate more than we need to be safe out = (WCHAR *) uiprivAlloc((n * 3) * sizeof (WCHAR), "WCHAR[]"); q = out; for (p = dts; *p != L'\0'; p++) { // first, if the current character is a y, increment the number of consecutive ys // otherwise, stop counting, and if there were only two, add two more to make four if (*p != L'y') { if (ny == 2) { *q++ = L'y'; *q++ = L'y'; } ny = 0; } else ny++; // next, handle quoted blocks // we do this AFTER the above so yy'abc' becomes yyyy'abc' and not yy'abc'yy // this handles the case of 'a''b' elegantly as well if (*p == L'\'') { // copy the opening quote *q++ = *p; // copy the contents for (;;) { p++; if (*p == L'\'') break; if (*p == L'\0') uiprivImplBug("unterminated quote in system-provided locale date string in expandYear()"); *q++ = *p; } // and fall through to copy the closing quote } // copy the current character *q++ = *p; } // handle trailing yy if (ny == 2) { *q++ = L'y'; *q++ = L'y'; } *q++ = L'\0'; return out; } // Windows has no combined date/time prebuilt constant; we have to build the format string ourselves // TODO use a default format if one fails static void setDateTimeFormat(HWND hwnd) { WCHAR *unexpandedDate, *date; WCHAR *time; WCHAR *datetime; int ndate, ntime; ndate = GLI(LOCALE_SSHORTDATE, NULL, 0); if (ndate == 0) logLastError(L"error getting date string length"); date = (WCHAR *) uiprivAlloc(ndate * sizeof (WCHAR), "WCHAR[]"); if (GLI(LOCALE_SSHORTDATE, date, ndate) == 0) logLastError(L"error geting date string"); unexpandedDate = date; // so we can free it date = expandYear(unexpandedDate, ndate); uiprivFree(unexpandedDate); ntime = GLI(LOCALE_STIMEFORMAT, NULL, 0); if (ndate == 0) logLastError(L"error getting time string length"); time = (WCHAR *) uiprivAlloc(ntime * sizeof (WCHAR), "WCHAR[]"); if (GLI(LOCALE_STIMEFORMAT, time, ntime) == 0) logLastError(L"error geting time string"); datetime = strf(L"%s %s", date, time); if (SendMessageW(hwnd, DTM_SETFORMAT, 0, (LPARAM) datetime) == 0) logLastError(L"error applying format string to date/time picker"); uiprivFree(datetime); uiprivFree(time); uiprivFree(date); } // control implementation static void uiDateTimePickerDestroy(uiControl *c) { uiDateTimePicker *d = uiDateTimePicker(c); uiWindowsUnregisterReceiveWM_WININICHANGE(d->hwnd); uiWindowsUnregisterWM_NOTIFYHandler(d->hwnd); uiWindowsEnsureDestroyWindow(d->hwnd); uiFreeControl(uiControl(d)); } uiWindowsControlAllDefaultsExceptDestroy(uiDateTimePicker) // the height returned from DTM_GETIDEALSIZE is unreliable; see http://stackoverflow.com/questions/30626549/what-is-the-proper-use-of-dtm-getidealsize-treating-the-returned-size-as-pixels // from http://msdn.microsoft.com/en-us/library/windows/desktop/dn742486.aspx#sizingandspacing #define entryHeight 14 static void uiDateTimePickerMinimumSize(uiWindowsControl *c, int *width, int *height) { uiDateTimePicker *d = uiDateTimePicker(c); SIZE s; uiWindowsSizing sizing; int y; s.cx = 0; s.cy = 0; SendMessageW(d->hwnd, DTM_GETIDEALSIZE, 0, (LPARAM) (&s)); *width = s.cx; y = entryHeight; uiWindowsGetSizing(d->hwnd, &sizing); uiWindowsSizingDlgUnitsToPixels(&sizing, NULL, &y); *height = y; } static BOOL onWM_NOTIFY(uiControl *c, HWND hwnd, NMHDR *nmhdr, LRESULT *lResult) { uiDateTimePicker *d = uiDateTimePicker(c); if (nmhdr->code != DTN_DATETIMECHANGE) return FALSE; (*(d->onChanged))(d, d->onChangedData); *lResult = 0; return TRUE; } static void fromSystemTime(SYSTEMTIME *systime, struct tm *time) { ZeroMemory(time, sizeof (struct tm)); time->tm_sec = systime->wSecond; time->tm_min = systime->wMinute; time->tm_hour = systime->wHour; time->tm_mday = systime->wDay; time->tm_mon = systime->wMonth - 1; time->tm_year = systime->wYear - 1900; time->tm_wday = systime->wDayOfWeek; time->tm_isdst = -1; } static void toSystemTime(const struct tm *time, SYSTEMTIME *systime) { ZeroMemory(systime, sizeof (SYSTEMTIME)); systime->wYear = time->tm_year + 1900; systime->wMonth = time->tm_mon + 1; systime->wDayOfWeek = time->tm_wday; systime->wDay = time->tm_mday; systime->wHour = time->tm_hour; systime->wMinute = time->tm_min; systime->wSecond = time->tm_sec; } static void defaultOnChanged(uiDateTimePicker *d, void *data) { // do nothing } void uiDateTimePickerTime(uiDateTimePicker *d, struct tm *time) { SYSTEMTIME systime; if (SendMessageW(d->hwnd, DTM_GETSYSTEMTIME, 0, (LPARAM) (&systime)) != GDT_VALID) logLastError(L"error getting date and time"); fromSystemTime(&systime, time); } void uiDateTimePickerSetTime(uiDateTimePicker *d, const struct tm *time) { SYSTEMTIME systime; toSystemTime(time, &systime); if (SendMessageW(d->hwnd, DTM_SETSYSTEMTIME, GDT_VALID, (LPARAM) (&systime)) == 0) logLastError(L"error setting date and time"); (*(d->onChanged))(d, d->onChangedData); } void uiDateTimePickerOnChanged(uiDateTimePicker *d, void(*f)(uiDateTimePicker *, void *), void *data) { d->onChanged = f; d->onChangedData = data; } static uiDateTimePicker *finishNewDateTimePicker(DWORD style) { uiDateTimePicker *d; uiWindowsNewControl(uiDateTimePicker, d); d->hwnd = uiWindowsEnsureCreateControlHWND(WS_EX_CLIENTEDGE, DATETIMEPICK_CLASSW, L"", style | WS_TABSTOP, hInstance, NULL, TRUE); // automatically update date/time format when user changes locale settings // for the standard styles, this is in the date-time picker itself // for our date/time mode, we do it in a subclass assigned in uiNewDateTimePicker() uiWindowsRegisterReceiveWM_WININICHANGE(d->hwnd); uiWindowsRegisterWM_NOTIFYHandler(d->hwnd, onWM_NOTIFY, uiControl(d)); uiDateTimePickerOnChanged(d, defaultOnChanged, NULL); return d; } static LRESULT CALLBACK datetimepickerSubProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam, UINT_PTR uIdSubclass, DWORD_PTR dwRefData) { switch (uMsg) { case WM_WININICHANGE: // we can optimize this by only doing it when the real date/time picker does it // unfortunately, I don't know when that is :/ // hopefully this won't hurt setDateTimeFormat(hwnd); return 0; case WM_NCDESTROY: if (RemoveWindowSubclass(hwnd, datetimepickerSubProc, uIdSubclass) == FALSE) logLastError(L"error removing date-time picker locale change handling subclass"); break; } return DefSubclassProc(hwnd, uMsg, wParam, lParam); } uiDateTimePicker *uiNewDateTimePicker(void) { uiDateTimePicker *d; d = finishNewDateTimePicker(0); setDateTimeFormat(d->hwnd); if (SetWindowSubclass(d->hwnd, datetimepickerSubProc, 0, (DWORD_PTR) d) == FALSE) logLastError(L"error subclassing date-time-picker to assist in locale change handling"); // TODO set a suitable default in this case return d; } uiDateTimePicker *uiNewDatePicker(void) { return finishNewDateTimePicker(DTS_SHORTDATECENTURYFORMAT); } uiDateTimePicker *uiNewTimePicker(void) { return finishNewDateTimePicker(DTS_TIMEFORMAT); }