2015-05-22 09:36:24 -05:00
// 22 may 2015
# include "uipriv_windows.h"
2015-08-30 11:25:53 -05:00
struct uiDateTimePicker {
uiWindowsControl c ;
2015-05-22 09:36:24 -05:00
HWND hwnd ;
} ;
2015-08-30 11:25:53 -05:00
uiWindowsDefineControlWithOnDestroy (
uiDateTimePicker , // type name
uiDateTimePickerType , // type function
uiWindowsUnregisterReceiveWM_WININICHANGE ( this - > hwnd ) ; // on destroy
)
2015-05-22 09:36:24 -05:00
2015-06-04 08:55:28 -05:00
// utility functions
# define GLI(what, buf, n) GetLocaleInfoEx(LOCALE_NAME_USER_DEFAULT, what, buf, n)
2015-06-05 14:43:18 -05:00
// 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 * ) uiAlloc ( ( 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 ' )
complain ( " 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 ;
}
2015-06-04 08:55:28 -05:00
// Windows has no combined date/time prebuilt constant; we have to build the format string ourselves
static void setDateTimeFormat ( HWND hwnd )
{
2015-06-05 14:43:18 -05:00
WCHAR * unexpandedDate , * date ;
WCHAR * time ;
WCHAR * datetime ;
2015-06-04 08:55:28 -05:00
int ndate , ntime ;
int n ;
ndate = GLI ( LOCALE_SSHORTDATE , NULL , 0 ) ;
if ( ndate = = 0 )
logLastError ( " error getting date string length in setDateTimeFormat() " ) ;
date = ( WCHAR * ) uiAlloc ( ndate * sizeof ( WCHAR ) , " WCHAR[] " ) ;
if ( GLI ( LOCALE_SSHORTDATE , date , ndate ) = = 0 )
logLastError ( " error geting date string in setDateTimeFormat() " ) ;
2015-06-05 14:43:18 -05:00
unexpandedDate = date ; // so we can free it
date = expandYear ( unexpandedDate , ndate ) ;
uiFree ( unexpandedDate ) ;
2015-06-04 08:55:28 -05:00
ntime = GLI ( LOCALE_STIMEFORMAT , NULL , 0 ) ;
if ( ndate = = 0 )
logLastError ( " error getting time string length in setDateTimeFormat() " ) ;
time = ( WCHAR * ) uiAlloc ( ntime * sizeof ( WCHAR ) , " WCHAR[] " ) ;
if ( GLI ( LOCALE_STIMEFORMAT , time , ntime ) = = 0 )
logLastError ( " error geting time string in setDateTimeFormat() " ) ;
n = _scwprintf ( L " %s %s " , date , time ) ;
datetime = ( WCHAR * ) uiAlloc ( ( n + 1 ) * sizeof ( WCHAR ) , " WCHAR[] " ) ;
2015-11-27 18:59:46 -06:00
_snwprintf ( datetime , n + 1 , L " %s %s " , date , time ) ;
2015-06-04 08:55:28 -05:00
if ( SendMessageW ( hwnd , DTM_SETFORMAT , 0 , ( LPARAM ) datetime ) = = 0 )
logLastError ( " error applying format string to date/time picker in setDateTimeFormat() " ) ;
uiFree ( datetime ) ;
uiFree ( time ) ;
uiFree ( date ) ;
}
// control implementation
2015-06-05 14:53:26 -05:00
// 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
2015-05-22 09:36:24 -05:00
// from http://msdn.microsoft.com/en-us/library/windows/desktop/dn742486.aspx#sizingandspacing
# define entryHeight 14
2015-08-31 16:50:23 -05:00
static void minimumSize ( uiWindowsControl * c , uiWindowsSizing * dd , intmax_t * width , intmax_t * height )
2015-05-22 09:36:24 -05:00
{
2015-08-30 11:25:53 -05:00
uiDateTimePicker * d = uiDateTimePicker ( c ) ;
2015-06-05 14:53:26 -05:00
SIZE s ;
s . cx = 0 ;
s . cy = 0 ;
2015-08-31 16:50:23 -05:00
SendMessageW ( d - > hwnd , DTM_GETIDEALSIZE , 0 , ( LPARAM ) ( & s ) ) ;
2015-06-05 14:53:26 -05:00
* width = s . cx ;
2015-08-31 16:50:23 -05:00
* height = uiWindowsDlgUnitsToY ( entryHeight , dd - > BaseY ) ;
2015-05-22 09:36:24 -05:00
}
2015-08-30 11:25:53 -05:00
static uiDateTimePicker * finishNewDateTimePicker ( DWORD style )
2015-05-22 09:36:24 -05:00
{
2015-08-30 11:25:53 -05:00
uiDateTimePicker * d ;
2015-05-22 09:36:24 -05:00
2015-08-30 11:25:53 -05:00
d = ( uiDateTimePicker * ) uiNewControl ( uiDateTimePickerType ( ) ) ;
2015-05-22 09:36:24 -05:00
2015-08-31 11:33:44 -05:00
d - > hwnd = uiWindowsEnsureCreateControlHWND ( WS_EX_CLIENTEDGE ,
2015-05-29 17:03:24 -05:00
DATETIMEPICK_CLASSW , L " " ,
style | WS_TABSTOP ,
hInstance , NULL ,
TRUE ) ;
2015-05-22 09:36:24 -05:00
2015-06-04 09:47:24 -05:00
// 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 ) ;
2015-08-30 11:25:53 -05:00
uiWindowsFinishNewControl ( d , uiDateTimePicker ) ;
2015-05-22 09:36:24 -05:00
2015-08-30 11:25:53 -05:00
return d ;
2015-05-22 09:36:24 -05:00
}
2015-06-04 09:55:54 -05:00
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 ( " error removing date-time picker locale change handling subclass in datetimepickerSubProc() " ) ;
break ;
}
return DefSubclassProc ( hwnd , uMsg , wParam , lParam ) ;
}
2015-05-22 09:36:24 -05:00
uiDateTimePicker * uiNewDateTimePicker ( void )
{
2015-08-30 11:25:53 -05:00
uiDateTimePicker * d ;
2015-05-22 12:24:07 -05:00
2015-08-30 11:25:53 -05:00
d = finishNewDateTimePicker ( 0 ) ;
2015-06-04 08:55:28 -05:00
setDateTimeFormat ( d - > hwnd ) ;
2015-06-04 09:55:54 -05:00
if ( SetWindowSubclass ( d - > hwnd , datetimepickerSubProc , 0 , ( DWORD_PTR ) d ) = = FALSE )
logLastError ( " error subclassing date-time-picker to assist in locale change handling in uiNewDateTimePicker() " ) ;
2015-08-30 11:25:53 -05:00
return d ;
2015-05-22 09:36:24 -05:00
}
uiDateTimePicker * uiNewDatePicker ( void )
{
2015-06-04 08:55:28 -05:00
return finishNewDateTimePicker ( DTS_SHORTDATECENTURYFORMAT ) ;
2015-05-22 09:36:24 -05:00
}
uiDateTimePicker * uiNewTimePicker ( void )
{
2015-06-04 08:55:28 -05:00
return finishNewDateTimePicker ( DTS_TIMEFORMAT ) ;
2015-05-22 09:36:24 -05:00
}