2016-06-08 22:33:32 -05:00
// 8 june 2016
# include "uipriv_windows.hpp"
struct formChild {
uiControl * c ;
HWND label ;
int stretchy ;
intmax_t height ;
} ;
struct uiForm {
uiWindowsControl c ;
HWND hwnd ;
std : : vector < struct formChild > * controls ;
int padded ;
} ;
static void formPadding ( uiForm * f , int * xpadding , int * ypadding )
{
uiWindowsSizing sizing ;
* xpadding = 0 ;
* ypadding = 0 ;
if ( f - > padded ) {
uiWindowsGetSizing ( f - > hwnd , & sizing ) ;
uiWindowsSizingStandardPadding ( & sizing , xpadding , ypadding ) ;
}
}
// via http://msdn.microsoft.com/en-us/library/windows/desktop/dn742486.aspx#sizingandspacing
# define labelHeight 8
# define labelYOffset 3
static void formRelayout ( uiForm * f )
{
RECT r ;
intmax_t x , y , width , height ;
int xpadding , ypadding ;
uintmax_t nStretchy ;
intmax_t labelwid , stretchyht ;
intmax_t thiswid ;
uintmax_t i ;
intmax_t minimumWidth , minimumHeight ;
uiWindowsSizing sizing ;
int labelht , labelyoff ;
if ( f - > controls - > size ( ) = = 0 )
return ;
uiWindowsEnsureGetClientRect ( f - > hwnd , & r ) ;
x = r . left ;
y = r . top ;
width = r . right - r . left ;
height = r . bottom - r . top ;
// -1) get this Form's padding
formPadding ( f , & xpadding , & ypadding ) ;
// 0) inset the available rect by the needed padding
// TODO this is incorrect if any controls are hidden
width - = xpadding ;
height - = ( f - > controls - > size ( ) - 1 ) * ypadding ;
// 1) get width of labels and height of non-stretchy controls
// this will tell us how much space will be left for controls
labelwid = 0 ;
stretchyht = height ;
nStretchy = 0 ;
for ( struct formChild & fc : * ( f - > controls ) ) {
if ( ! uiControlVisible ( fc . c ) )
continue ;
thiswid = uiWindowsWindowTextWidth ( fc . label ) ;
if ( labelwid < thiswid )
labelwid = thiswid ;
if ( fc . stretchy ) {
nStretchy + + ;
continue ;
}
uiWindowsControlMinimumSize ( uiWindowsControl ( fc . c ) , & minimumWidth , & minimumHeight ) ;
fc . height = minimumHeight ;
stretchyht - = minimumHeight ;
}
// 2) now get the width of controls and the height of stretchy controls
width - = labelwid ;
if ( nStretchy ! = 0 ) {
stretchyht / = nStretchy ;
for ( struct formChild & fc : * ( f - > controls ) ) {
if ( ! uiControlVisible ( fc . c ) )
continue ;
if ( fc . stretchy )
fc . height = stretchyht ;
}
}
// 3) get the y offset
labelyoff = labelYOffset ;
uiWindowsGetSizing ( f - > hwnd , & sizing ) ;
uiWindowsSizingDlgUnitsToPixels ( & sizing , NULL , & labelyoff ) ;
// 4) now we can position controls
// first, make relative to the top-left corner of the container
// also prefer left alignment on Windows
x = labelwid + xpadding ;
y = 0 ;
for ( const struct formChild & fc : * ( f - > controls ) ) {
if ( ! uiControlVisible ( fc . c ) )
continue ;
labelht = labelHeight ;
uiWindowsGetSizing ( f - > hwnd , & sizing ) ;
uiWindowsSizingDlgUnitsToPixels ( & sizing , NULL , & labelht ) ;
uiWindowsEnsureMoveWindowDuringResize ( fc . label , 0 , y + labelyoff - sizing . InternalLeading , labelwid , labelht ) ;
uiWindowsEnsureMoveWindowDuringResize ( ( HWND ) uiControlHandle ( fc . c ) , x , y , width , fc . height ) ;
y + = fc . height + ypadding ;
}
}
static void uiFormDestroy ( uiControl * c )
{
uiForm * f = uiForm ( c ) ;
for ( const struct formChild & fc : * ( f - > controls ) ) {
uiControlSetParent ( fc . c , NULL ) ;
uiControlDestroy ( fc . c ) ;
uiWindowsEnsureDestroyWindow ( fc . label ) ;
}
delete f - > controls ;
uiWindowsEnsureDestroyWindow ( f - > hwnd ) ;
uiFreeControl ( uiControl ( f ) ) ;
}
uiWindowsControlDefaultHandle ( uiForm )
uiWindowsControlDefaultParent ( uiForm )
uiWindowsControlDefaultSetParent ( uiForm )
uiWindowsControlDefaultToplevel ( uiForm )
uiWindowsControlDefaultVisible ( uiForm )
uiWindowsControlDefaultShow ( uiForm )
uiWindowsControlDefaultHide ( uiForm )
uiWindowsControlDefaultEnabled ( uiForm )
uiWindowsControlDefaultEnable ( uiForm )
uiWindowsControlDefaultDisable ( uiForm )
static void uiFormSyncEnableState ( uiWindowsControl * c , int enabled )
{
uiForm * f = uiForm ( c ) ;
if ( uiWindowsShouldStopSyncEnableState ( uiWindowsControl ( f ) , enabled ) )
return ;
for ( const struct formChild & fc : * ( f - > controls ) )
uiWindowsControlSyncEnableState ( uiWindowsControl ( fc . c ) , enabled ) ;
}
uiWindowsControlDefaultSetParentHWND ( uiForm )
static void uiFormMinimumSize ( uiWindowsControl * c , intmax_t * width , intmax_t * height )
{
uiForm * f = uiForm ( c ) ;
int xpadding , ypadding ;
uintmax_t nStretchy ;
// these two contain the largest minimum width and height of all stretchy controls in the form
// all stretchy controls will use this value to determine the final minimum size
intmax_t maxLabelWidth , maxControlWidth ;
intmax_t maxStretchyHeight ;
intmax_t labelwid ;
uintmax_t i ;
intmax_t minimumWidth , minimumHeight ;
uiWindowsSizing sizing ;
* width = 0 ;
* height = 0 ;
if ( f - > controls - > size ( ) = = 0 )
return ;
// 0) get this Form's padding
formPadding ( f , & xpadding , & ypadding ) ;
// 1) initialize the desired rect with the needed padding
// TODO this is wrong if any controls are hidden
* width = xpadding ;
* height = ( f - > controls - > size ( ) - 1 ) * ypadding ;
// 2) determine the longest width of all controls and labels; add in the height of non-stretchy controls and get (but not add in) the largest heights of stretchy controls
// we still add in like direction of stretchy controls
nStretchy = 0 ;
maxLabelWidth = 0 ;
maxControlWidth = 0 ;
maxStretchyHeight = 0 ;
for ( const struct formChild & fc : * ( f - > controls ) ) {
if ( ! uiControlVisible ( fc . c ) )
continue ;
labelwid = uiWindowsWindowTextWidth ( fc . label ) ;
if ( maxLabelWidth < labelwid )
maxLabelWidth = labelwid ;
uiWindowsControlMinimumSize ( uiWindowsControl ( fc . c ) , & minimumWidth , & minimumHeight ) ;
if ( fc . stretchy ) {
nStretchy + + ;
if ( maxStretchyHeight < minimumHeight )
maxStretchyHeight = minimumHeight ;
}
if ( maxControlWidth < minimumWidth )
maxControlWidth = minimumWidth ;
if ( ! fc . stretchy )
* height + = minimumHeight ;
}
* width + = maxLabelWidth + maxControlWidth ;
// 3) and now we can add in stretchy controls
* height + = nStretchy * maxStretchyHeight ;
}
static void uiFormMinimumSizeChanged ( uiWindowsControl * c )
{
uiForm * f = uiForm ( c ) ;
if ( uiWindowsControlTooSmall ( uiWindowsControl ( f ) ) ) {
uiWindowsControlContinueMinimumSizeChanged ( uiWindowsControl ( f ) ) ;
return ;
}
formRelayout ( f ) ;
}
uiWindowsControlDefaultLayoutRect ( uiForm )
uiWindowsControlDefaultAssignControlIDZOrder ( uiForm )
static void formArrangeChildren ( uiForm * f )
{
LONG_PTR controlID ;
HWND insertAfter ;
uintmax_t i ;
controlID = 100 ;
insertAfter = NULL ;
for ( const struct formChild & fc : * ( f - > controls ) ) {
// TODO assign label ID and z-order
uiWindowsControlAssignControlIDZOrder ( uiWindowsControl ( fc . c ) , & controlID , & insertAfter ) ;
}
}
void uiFormAppend ( uiForm * f , const char * label , uiControl * c , int stretchy )
{
struct formChild fc ;
WCHAR * wlabel ;
fc . c = c ;
wlabel = toUTF16 ( label ) ;
fc . label = uiWindowsEnsureCreateControlHWND ( 0 ,
L " STATIC " , wlabel ,
SS_LEFT | SS_NOPREFIX ,
hInstance , NULL ,
TRUE ) ;
2016-06-10 21:29:20 -05:00
uiFree ( wlabel ) ;
2016-06-08 22:33:32 -05:00
uiWindowsEnsureSetParentHWND ( fc . label , f - > hwnd ) ;
fc . stretchy = stretchy ;
uiControlSetParent ( fc . c , uiControl ( f ) ) ;
uiWindowsControlSetParentHWND ( uiWindowsControl ( fc . c ) , f - > hwnd ) ;
f - > controls - > push_back ( fc ) ;
formArrangeChildren ( f ) ;
uiWindowsControlMinimumSizeChanged ( uiWindowsControl ( f ) ) ;
}
int uiFormPadded ( uiForm * f )
{
return f - > padded ;
}
void uiFormSetPadded ( uiForm * f , int padded )
{
f - > padded = padded ;
uiWindowsControlMinimumSizeChanged ( uiWindowsControl ( f ) ) ;
}
static void onResize ( uiWindowsControl * c )
{
formRelayout ( uiForm ( c ) ) ;
}
uiForm * uiNewForm ( void )
{
uiForm * f ;
uiWindowsNewControl ( uiForm , f ) ;
f - > hwnd = uiWindowsMakeContainer ( uiWindowsControl ( f ) , onResize ) ;
f - > controls = new std : : vector < struct formChild > ;
return f ;
}