1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/sizer.cpp
3 // Purpose: provide new wxSizer class for layout
4 // Author: Robert Roebling and Robin Dunn, contributions by
5 // Dirk Holtwick, Ron Lee
6 // Modified by: Ron Lee
9 // Copyright: (c) Robin Dunn, Robert Roebling
10 // Licence: wxWindows licence
11 /////////////////////////////////////////////////////////////////////////////
13 // For compilers that support precompilation, includes "wx.h".
14 #include "wx/wxprec.h"
23 #include "wx/string.h"
27 #include "wx/settings.h"
28 #include "wx/button.h"
29 #include "wx/statbox.h"
30 #include "wx/toplevel.h"
33 #include "wx/listimpl.cpp"
35 #if WXWIN_COMPATIBILITY_2_4
36 #include "wx/notebook.h"
39 //---------------------------------------------------------------------------
41 IMPLEMENT_CLASS(wxSizerItem
, wxObject
)
42 IMPLEMENT_CLASS(wxSizer
, wxObject
)
43 IMPLEMENT_CLASS(wxGridSizer
, wxSizer
)
44 IMPLEMENT_CLASS(wxFlexGridSizer
, wxGridSizer
)
45 IMPLEMENT_CLASS(wxBoxSizer
, wxSizer
)
47 IMPLEMENT_CLASS(wxStaticBoxSizer
, wxBoxSizer
)
50 IMPLEMENT_CLASS(wxStdDialogButtonSizer
, wxBoxSizer
)
53 WX_DEFINE_EXPORTED_LIST( wxSizerItemList
)
88 // ----------------------------------------------------------------------------
90 // ----------------------------------------------------------------------------
92 void wxSizerItem::Init(const wxSizerFlags
& flags
)
96 m_proportion
= flags
.GetProportion();
97 m_flag
= flags
.GetFlags();
98 m_border
= flags
.GetBorderInPixels();
101 wxSizerItem::wxSizerItem()
113 void wxSizerItem::SetWindow(wxWindow
*window
)
115 wxCHECK_RET( window
, _T("NULL window in wxSizerItem::SetWindow()") );
117 m_kind
= Item_Window
;
120 // window doesn't become smaller than its initial size, whatever happens
121 m_minSize
= window
->GetSize();
123 if ( m_flag
& wxFIXED_MINSIZE
)
124 window
->SetMinSize(m_minSize
);
126 // aspect ratio calculated from initial size
130 wxSizerItem::wxSizerItem(wxWindow
*window
,
135 : m_proportion(proportion
),
144 void wxSizerItem::SetSizer(wxSizer
*sizer
)
150 wxSizerItem::wxSizerItem(wxSizer
*sizer
,
155 : m_proportion(proportion
),
163 // m_minSize is set later
167 void wxSizerItem::SetSpacer(const wxSize
& size
)
169 m_kind
= Item_Spacer
;
170 m_spacer
= new wxSizerSpacer(size
);
175 wxSizerItem::wxSizerItem(int width
,
181 : m_minSize(width
, height
), // minimal size is the initial size
182 m_proportion(proportion
),
187 SetSpacer(width
, height
);
190 wxSizerItem::~wxSizerItem()
200 m_window
->SetContainingSizer(NULL
);
213 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
217 wxSize
wxSizerItem::GetSpacer() const
220 if ( m_kind
== Item_Spacer
)
221 size
= m_spacer
->GetSize();
227 wxSize
wxSizerItem::GetSize() const
236 ret
= m_window
->GetSize();
240 ret
= m_sizer
->GetSize();
244 ret
= m_spacer
->GetSize();
249 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
256 if (m_flag
& wxNORTH
)
258 if (m_flag
& wxSOUTH
)
264 wxSize
wxSizerItem::CalcMin()
268 m_minSize
= m_sizer
->GetMinSize();
270 // if we have to preserve aspect ratio _AND_ this is
271 // the first-time calculation, consider ret to be initial size
272 if ( (m_flag
& wxSHAPED
) && wxIsNullDouble(m_ratio
) )
275 else if ( IsWindow() )
277 // Since the size of the window may change during runtime, we
278 // should use the current minimal/best size.
279 m_minSize
= m_window
->GetBestFittingSize();
282 return GetMinSizeWithBorder();
285 wxSize
wxSizerItem::GetMinSizeWithBorder() const
287 wxSize ret
= m_minSize
;
293 if (m_flag
& wxNORTH
)
295 if (m_flag
& wxSOUTH
)
302 void wxSizerItem::SetDimension( const wxPoint
& pos_
, const wxSize
& size_
)
306 if (m_flag
& wxSHAPED
)
308 // adjust aspect ratio
309 int rwidth
= (int) (size
.y
* m_ratio
);
313 int rheight
= (int) (size
.x
/ m_ratio
);
314 // add vertical space
315 if (m_flag
& wxALIGN_CENTER_VERTICAL
)
316 pos
.y
+= (size
.y
- rheight
) / 2;
317 else if (m_flag
& wxALIGN_BOTTOM
)
318 pos
.y
+= (size
.y
- rheight
);
319 // use reduced dimensions
322 else if (rwidth
< size
.x
)
324 // add horizontal space
325 if (m_flag
& wxALIGN_CENTER_HORIZONTAL
)
326 pos
.x
+= (size
.x
- rwidth
) / 2;
327 else if (m_flag
& wxALIGN_RIGHT
)
328 pos
.x
+= (size
.x
- rwidth
);
333 // This is what GetPosition() returns. Since we calculate
334 // borders afterwards, GetPosition() will be the left/top
335 // corner of the surrounding border.
347 if (m_flag
& wxNORTH
)
352 if (m_flag
& wxSOUTH
)
357 m_rect
= wxRect(pos
, size
);
362 wxFAIL_MSG( _T("can't set size of uninitialized sizer item") );
366 m_window
->SetSize(pos
.x
, pos
.y
, size
.x
, size
.y
,
367 wxSIZE_ALLOW_MINUS_ONE
);
371 m_sizer
->SetDimension(pos
.x
, pos
.y
, size
.x
, size
.y
);
375 m_spacer
->SetSize(size
);
380 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
384 void wxSizerItem::DeleteWindows()
393 //We are deleting the window from this sizer - normally
394 //the window destroys the sizer associated with it,
395 //which might destroy this, which we don't want
396 m_window
->SetContainingSizer(NULL
);
398 //Putting this after the switch will result in a spacer
399 //not being deleted properly on destruction
404 m_sizer
->DeleteWindows();
409 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
414 void wxSizerItem::Show( bool show
)
419 wxFAIL_MSG( _T("can't show uninitialized sizer item") );
423 m_window
->Show(show
);
431 m_spacer
->Show(show
);
436 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
440 bool wxSizerItem::IsShown() const
445 // we may be called from CalcMin(), just return false so that we're
450 return m_window
->IsShown();
453 // arbitrarily decide that if at least one of our elements is
454 // shown, so are we (this arbitrariness is the reason for
455 // deprecating this function)
457 for ( wxSizerItemList::compatibility_iterator
458 node
= m_sizer
->GetChildren().GetFirst();
460 node
= node
->GetNext() )
462 if ( node
->GetData()->IsShown() )
469 return m_spacer
->IsShown();
473 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
479 #if WXWIN_COMPATIBILITY_2_6
480 void wxSizerItem::SetOption( int option
)
482 SetProportion( option
);
485 int wxSizerItem::GetOption() const
487 return GetProportion();
489 #endif // WXWIN_COMPATIBILITY_2_6
492 //---------------------------------------------------------------------------
494 //---------------------------------------------------------------------------
498 WX_CLEAR_LIST(wxSizerItemList
, m_children
);
501 wxSizerItem
* wxSizer::Insert( size_t index
, wxSizerItem
*item
)
503 m_children
.Insert( index
, item
);
505 if ( item
->GetWindow() )
506 item
->GetWindow()->SetContainingSizer( this );
511 void wxSizer::SetContainingWindow(wxWindow
*win
)
513 if ( win
== m_containingWindow
)
516 m_containingWindow
= win
;
518 // set the same window for all nested sizers as well, they also are in the
520 for ( wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
522 node
= node
->GetNext() )
524 wxSizerItem
*const item
= node
->GetData();
525 wxSizer
*const sizer
= item
->GetSizer();
529 sizer
->SetContainingWindow(win
);
534 #if WXWIN_COMPATIBILITY_2_6
535 bool wxSizer::Remove( wxWindow
*window
)
537 return Detach( window
);
539 #endif // WXWIN_COMPATIBILITY_2_6
541 bool wxSizer::Remove( wxSizer
*sizer
)
543 wxASSERT_MSG( sizer
, _T("Removing NULL sizer") );
545 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
548 wxSizerItem
*item
= node
->GetData();
550 if (item
->GetSizer() == sizer
)
553 m_children
.Erase( node
);
557 node
= node
->GetNext();
563 bool wxSizer::Remove( int index
)
565 wxCHECK_MSG( index
>= 0 && (size_t)index
< m_children
.GetCount(),
567 _T("Remove index is out of range") );
569 wxSizerItemList::compatibility_iterator node
= m_children
.Item( index
);
571 wxCHECK_MSG( node
, false, _T("Failed to find child node") );
573 wxSizerItem
*item
= node
->GetData();
575 if ( item
->IsWindow() )
576 item
->GetWindow()->SetContainingSizer( NULL
);
579 m_children
.Erase( node
);
583 bool wxSizer::Detach( wxSizer
*sizer
)
585 wxASSERT_MSG( sizer
, _T("Detaching NULL sizer") );
587 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
590 wxSizerItem
*item
= node
->GetData();
592 if (item
->GetSizer() == sizer
)
596 m_children
.Erase( node
);
599 node
= node
->GetNext();
605 bool wxSizer::Detach( wxWindow
*window
)
607 wxASSERT_MSG( window
, _T("Detaching NULL window") );
609 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
612 wxSizerItem
*item
= node
->GetData();
614 if (item
->GetWindow() == window
)
616 item
->GetWindow()->SetContainingSizer( NULL
);
618 m_children
.Erase( node
);
621 node
= node
->GetNext();
627 bool wxSizer::Detach( int index
)
629 wxCHECK_MSG( index
>= 0 && (size_t)index
< m_children
.GetCount(),
631 _T("Detach index is out of range") );
633 wxSizerItemList::compatibility_iterator node
= m_children
.Item( index
);
635 wxCHECK_MSG( node
, false, _T("Failed to find child node") );
637 wxSizerItem
*item
= node
->GetData();
639 if ( item
->IsSizer() )
641 else if ( item
->IsWindow() )
642 item
->GetWindow()->SetContainingSizer( NULL
);
645 m_children
.Erase( node
);
649 bool wxSizer::Replace( wxWindow
*oldwin
, wxWindow
*newwin
, bool recursive
)
651 wxASSERT_MSG( oldwin
, _T("Replacing NULL window") );
652 wxASSERT_MSG( newwin
, _T("Replacing with NULL window") );
654 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
657 wxSizerItem
*item
= node
->GetData();
659 if (item
->GetWindow() == oldwin
)
661 item
->GetWindow()->SetContainingSizer( NULL
);
662 item
->SetWindow(newwin
);
663 newwin
->SetContainingSizer( this );
666 else if (recursive
&& item
->IsSizer())
668 if (item
->GetSizer()->Replace( oldwin
, newwin
, true ))
672 node
= node
->GetNext();
678 bool wxSizer::Replace( wxSizer
*oldsz
, wxSizer
*newsz
, bool recursive
)
680 wxASSERT_MSG( oldsz
, _T("Replacing NULL sizer") );
681 wxASSERT_MSG( newsz
, _T("Replacing with NULL sizer") );
683 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
686 wxSizerItem
*item
= node
->GetData();
688 if (item
->GetSizer() == oldsz
)
690 wxSizer
*old
= item
->GetSizer();
691 item
->SetSizer(newsz
);
695 else if (recursive
&& item
->IsSizer())
697 if (item
->GetSizer()->Replace( oldsz
, newsz
, true ))
701 node
= node
->GetNext();
707 bool wxSizer::Replace( size_t old
, wxSizerItem
*newitem
)
709 wxCHECK_MSG( old
< m_children
.GetCount(), false, _T("Replace index is out of range") );
710 wxASSERT_MSG( newitem
, _T("Replacing with NULL item") );
712 wxSizerItemList::compatibility_iterator node
= m_children
.Item( old
);
714 wxCHECK_MSG( node
, false, _T("Failed to find child node") );
716 wxSizerItem
*item
= node
->GetData();
717 node
->SetData(newitem
);
723 void wxSizer::Clear( bool delete_windows
)
725 // First clear the ContainingSizer pointers
726 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
729 wxSizerItem
*item
= node
->GetData();
731 if (item
->IsWindow())
732 item
->GetWindow()->SetContainingSizer( NULL
);
733 node
= node
->GetNext();
736 // Destroy the windows if needed
740 // Now empty the list
741 WX_CLEAR_LIST(wxSizerItemList
, m_children
);
744 void wxSizer::DeleteWindows()
746 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
749 wxSizerItem
*item
= node
->GetData();
751 item
->DeleteWindows();
752 node
= node
->GetNext();
756 wxSize
wxSizer::Fit( wxWindow
*window
)
758 wxSize
size(window
->IsTopLevel() ? FitSize(window
)
759 : GetMinWindowSize(window
));
761 window
->SetSize( size
);
766 void wxSizer::FitInside( wxWindow
*window
)
769 if (window
->IsTopLevel())
770 size
= VirtualFitSize( window
);
772 size
= GetMinClientSize( window
);
774 window
->SetVirtualSize( size
);
777 void wxSizer::Layout()
779 // (re)calculates minimums needed for each item and other preparations
783 // Applies the layout and repositions/resizes the items
787 void wxSizer::SetSizeHints( wxWindow
*window
)
789 // Preserve the window's max size hints, but set the
790 // lower bound according to the sizer calculations.
792 wxSize size
= Fit( window
);
794 window
->SetSizeHints( size
.x
,
796 window
->GetMaxWidth(),
797 window
->GetMaxHeight() );
800 void wxSizer::SetVirtualSizeHints( wxWindow
*window
)
802 // Preserve the window's max size hints, but set the
803 // lower bound according to the sizer calculations.
806 wxSize
size( window
->GetVirtualSize() );
807 window
->SetVirtualSizeHints( size
.x
,
809 window
->GetMaxWidth(),
810 window
->GetMaxHeight() );
813 wxSize
wxSizer::GetMaxWindowSize( wxWindow
*window
) const
815 return window
->GetMaxSize();
818 wxSize
wxSizer::GetMinWindowSize( wxWindow
*window
)
820 wxSize
minSize( GetMinSize() );
821 wxSize
size( window
->GetSize() );
822 wxSize
client_size( window
->GetClientSize() );
824 return wxSize( minSize
.x
+size
.x
-client_size
.x
,
825 minSize
.y
+size
.y
-client_size
.y
);
828 // TODO on mac we need a function that determines how much free space this
829 // min size contains, in order to make sure that we have 20 pixels of free
830 // space around the controls
832 // Return a window size that will fit within the screens dimensions
833 wxSize
wxSizer::FitSize( wxWindow
*window
)
835 if ( window
->IsTopLevel() )
837 wxTopLevelWindow
*tlw
= wxDynamicCast(window
, wxTopLevelWindow
);
838 if ( tlw
&& tlw
->IsAlwaysMaximized() )
840 return tlw
->GetClientSize();
844 wxSize size
= GetMinWindowSize( window
);
845 wxSize sizeMax
= GetMaxWindowSize( window
);
847 // Limit the size if sizeMax != wxDefaultSize
849 if ( size
.x
> sizeMax
.x
&& sizeMax
.x
!= wxDefaultCoord
)
851 if ( size
.y
> sizeMax
.y
&& sizeMax
.y
!= wxDefaultCoord
)
857 wxSize
wxSizer::GetMaxClientSize( wxWindow
*window
) const
859 wxSize
maxSize( window
->GetMaxSize() );
861 if ( maxSize
!= wxDefaultSize
)
863 wxSize
size( window
->GetSize() );
864 wxSize
client_size( window
->GetClientSize() );
866 return wxSize( maxSize
.x
+ client_size
.x
- size
.x
,
867 maxSize
.y
+ client_size
.y
- size
.y
);
870 return wxDefaultSize
;
873 wxSize
wxSizer::GetMinClientSize( wxWindow
*WXUNUSED(window
) )
875 return GetMinSize(); // Already returns client size.
878 wxSize
wxSizer::VirtualFitSize( wxWindow
*window
)
880 wxSize size
= GetMinClientSize( window
);
881 wxSize sizeMax
= GetMaxClientSize( window
);
883 // Limit the size if sizeMax != wxDefaultSize
885 if ( size
.x
> sizeMax
.x
&& sizeMax
.x
!= wxDefaultCoord
)
887 if ( size
.y
> sizeMax
.y
&& sizeMax
.y
!= wxDefaultCoord
)
893 void wxSizer::SetDimension( int x
, int y
, int width
, int height
)
902 wxSize
wxSizer::GetMinSize()
904 wxSize
ret( CalcMin() );
905 if (ret
.x
< m_minSize
.x
) ret
.x
= m_minSize
.x
;
906 if (ret
.y
< m_minSize
.y
) ret
.y
= m_minSize
.y
;
910 void wxSizer::DoSetMinSize( int width
, int height
)
913 m_minSize
.y
= height
;
916 bool wxSizer::DoSetItemMinSize( wxWindow
*window
, int width
, int height
)
918 wxASSERT_MSG( window
, _T("SetMinSize for NULL window") );
920 // Is it our immediate child?
922 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
925 wxSizerItem
*item
= node
->GetData();
927 if (item
->GetWindow() == window
)
929 item
->SetMinSize( width
, height
);
932 node
= node
->GetNext();
935 // No? Search any subsizers we own then
937 node
= m_children
.GetFirst();
940 wxSizerItem
*item
= node
->GetData();
942 if ( item
->GetSizer() &&
943 item
->GetSizer()->DoSetItemMinSize( window
, width
, height
) )
945 // A child sizer found the requested windw, exit.
948 node
= node
->GetNext();
954 bool wxSizer::DoSetItemMinSize( wxSizer
*sizer
, int width
, int height
)
956 wxASSERT_MSG( sizer
, _T("SetMinSize for NULL sizer") );
958 // Is it our immediate child?
960 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
963 wxSizerItem
*item
= node
->GetData();
965 if (item
->GetSizer() == sizer
)
967 item
->GetSizer()->DoSetMinSize( width
, height
);
970 node
= node
->GetNext();
973 // No? Search any subsizers we own then
975 node
= m_children
.GetFirst();
978 wxSizerItem
*item
= node
->GetData();
980 if ( item
->GetSizer() &&
981 item
->GetSizer()->DoSetItemMinSize( sizer
, width
, height
) )
983 // A child found the requested sizer, exit.
986 node
= node
->GetNext();
992 bool wxSizer::DoSetItemMinSize( size_t index
, int width
, int height
)
994 wxSizerItemList::compatibility_iterator node
= m_children
.Item( index
);
996 wxCHECK_MSG( node
, false, _T("Failed to find child node") );
998 wxSizerItem
*item
= node
->GetData();
1000 if (item
->GetSizer())
1002 // Sizers contains the minimal size in them, if not calculated ...
1003 item
->GetSizer()->DoSetMinSize( width
, height
);
1007 // ... but the minimal size of spacers and windows is stored via the item
1008 item
->SetMinSize( width
, height
);
1014 wxSizerItem
* wxSizer::GetItem( wxWindow
*window
, bool recursive
)
1016 wxASSERT_MSG( window
, _T("GetItem for NULL window") );
1018 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1021 wxSizerItem
*item
= node
->GetData();
1023 if (item
->GetWindow() == window
)
1027 else if (recursive
&& item
->IsSizer())
1029 wxSizerItem
*subitem
= item
->GetSizer()->GetItem( window
, true );
1034 node
= node
->GetNext();
1040 wxSizerItem
* wxSizer::GetItem( wxSizer
*sizer
, bool recursive
)
1042 wxASSERT_MSG( sizer
, _T("GetItem for NULL sizer") );
1044 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1047 wxSizerItem
*item
= node
->GetData();
1049 if (item
->GetSizer() == sizer
)
1053 else if (recursive
&& item
->IsSizer())
1055 wxSizerItem
*subitem
= item
->GetSizer()->GetItem( sizer
, true );
1060 node
= node
->GetNext();
1066 wxSizerItem
* wxSizer::GetItem( size_t index
)
1068 wxCHECK_MSG( index
< m_children
.GetCount(),
1070 _T("GetItem index is out of range") );
1072 return m_children
.Item( index
)->GetData();
1075 bool wxSizer::Show( wxWindow
*window
, bool show
, bool recursive
)
1077 wxSizerItem
*item
= GetItem( window
, recursive
);
1088 bool wxSizer::Show( wxSizer
*sizer
, bool show
, bool recursive
)
1090 wxSizerItem
*item
= GetItem( sizer
, recursive
);
1101 bool wxSizer::Show( size_t index
, bool show
)
1103 wxSizerItem
*item
= GetItem( index
);
1114 void wxSizer::ShowItems( bool show
)
1116 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1119 node
->GetData()->Show( show
);
1120 node
= node
->GetNext();
1124 bool wxSizer::IsShown( wxWindow
*window
) const
1126 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1129 wxSizerItem
*item
= node
->GetData();
1131 if (item
->GetWindow() == window
)
1133 return item
->IsShown();
1135 node
= node
->GetNext();
1138 wxFAIL_MSG( _T("IsShown failed to find sizer item") );
1143 bool wxSizer::IsShown( wxSizer
*sizer
) const
1145 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1148 wxSizerItem
*item
= node
->GetData();
1150 if (item
->GetSizer() == sizer
)
1152 return item
->IsShown();
1154 node
= node
->GetNext();
1157 wxFAIL_MSG( _T("IsShown failed to find sizer item") );
1162 bool wxSizer::IsShown( size_t index
) const
1164 wxCHECK_MSG( index
< m_children
.GetCount(),
1166 _T("IsShown index is out of range") );
1168 return m_children
.Item( index
)->GetData()->IsShown();
1172 //---------------------------------------------------------------------------
1174 //---------------------------------------------------------------------------
1176 wxGridSizer::wxGridSizer( int rows
, int cols
, int vgap
, int hgap
)
1177 : m_rows( ( cols
== 0 && rows
== 0 ) ? 1 : rows
)
1184 wxGridSizer::wxGridSizer( int cols
, int vgap
, int hgap
)
1185 : m_rows( cols
== 0 ? 1 : 0 )
1192 int wxGridSizer::CalcRowsCols(int& nrows
, int& ncols
) const
1194 int nitems
= m_children
.GetCount();
1200 nrows
= (nitems
+ m_cols
- 1) / m_cols
;
1204 ncols
= (nitems
+ m_rows
- 1) / m_rows
;
1207 else // 0 columns, 0 rows?
1209 wxFAIL_MSG( _T("grid sizer must have either rows or columns fixed") );
1218 void wxGridSizer::RecalcSizes()
1220 int nitems
, nrows
, ncols
;
1221 if ( (nitems
= CalcRowsCols(nrows
, ncols
)) == 0 )
1224 wxSize
sz( GetSize() );
1225 wxPoint
pt( GetPosition() );
1227 int w
= (sz
.x
- (ncols
- 1) * m_hgap
) / ncols
;
1228 int h
= (sz
.y
- (nrows
- 1) * m_vgap
) / nrows
;
1231 for (int c
= 0; c
< ncols
; c
++)
1234 for (int r
= 0; r
< nrows
; r
++)
1236 int i
= r
* ncols
+ c
;
1239 wxSizerItemList::compatibility_iterator node
= m_children
.Item( i
);
1241 wxASSERT_MSG( node
, _T("Failed to find SizerItemList node") );
1243 SetItemBounds( node
->GetData(), x
, y
, w
, h
);
1251 wxSize
wxGridSizer::CalcMin()
1254 if ( CalcRowsCols(nrows
, ncols
) == 0 )
1257 // Find the max width and height for any component
1261 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1264 wxSizerItem
*item
= node
->GetData();
1265 wxSize
sz( item
->CalcMin() );
1267 w
= wxMax( w
, sz
.x
);
1268 h
= wxMax( h
, sz
.y
);
1270 node
= node
->GetNext();
1273 return wxSize( ncols
* w
+ (ncols
-1) * m_hgap
,
1274 nrows
* h
+ (nrows
-1) * m_vgap
);
1277 void wxGridSizer::SetItemBounds( wxSizerItem
*item
, int x
, int y
, int w
, int h
)
1280 wxSize
sz( item
->GetMinSizeWithBorder() );
1281 int flag
= item
->GetFlag();
1283 if ((flag
& wxEXPAND
) || (flag
& wxSHAPED
))
1289 if (flag
& wxALIGN_CENTER_HORIZONTAL
)
1291 pt
.x
= x
+ (w
- sz
.x
) / 2;
1293 else if (flag
& wxALIGN_RIGHT
)
1295 pt
.x
= x
+ (w
- sz
.x
);
1298 if (flag
& wxALIGN_CENTER_VERTICAL
)
1300 pt
.y
= y
+ (h
- sz
.y
) / 2;
1302 else if (flag
& wxALIGN_BOTTOM
)
1304 pt
.y
= y
+ (h
- sz
.y
);
1308 item
->SetDimension(pt
, sz
);
1311 //---------------------------------------------------------------------------
1313 //---------------------------------------------------------------------------
1315 wxFlexGridSizer::wxFlexGridSizer( int rows
, int cols
, int vgap
, int hgap
)
1316 : wxGridSizer( rows
, cols
, vgap
, hgap
),
1317 m_flexDirection(wxBOTH
),
1318 m_growMode(wxFLEX_GROWMODE_SPECIFIED
)
1322 wxFlexGridSizer::wxFlexGridSizer( int cols
, int vgap
, int hgap
)
1323 : wxGridSizer( cols
, vgap
, hgap
),
1324 m_flexDirection(wxBOTH
),
1325 m_growMode(wxFLEX_GROWMODE_SPECIFIED
)
1329 wxFlexGridSizer::~wxFlexGridSizer()
1333 void wxFlexGridSizer::RecalcSizes()
1335 int nitems
, nrows
, ncols
;
1336 if ( (nitems
= CalcRowsCols(nrows
, ncols
)) == 0 )
1339 wxPoint
pt( GetPosition() );
1340 wxSize
sz( GetSize() );
1342 AdjustForGrowables(sz
, m_calculatedMinSize
, nrows
, ncols
);
1344 sz
= wxSize( pt
.x
+ sz
.x
, pt
.y
+ sz
.y
);
1347 for (int c
= 0; c
< ncols
; c
++)
1350 for (int r
= 0; r
< nrows
; r
++)
1352 int i
= r
* ncols
+ c
;
1355 wxSizerItemList::compatibility_iterator node
= m_children
.Item( i
);
1357 wxASSERT_MSG( node
, _T("Failed to find node") );
1359 int w
= wxMax( 0, wxMin( m_colWidths
[c
], sz
.x
- x
) );
1360 int h
= wxMax( 0, wxMin( m_rowHeights
[r
], sz
.y
- y
) );
1362 SetItemBounds( node
->GetData(), x
, y
, w
, h
);
1364 if (m_rowHeights
[r
] != -1)
1365 y
= y
+ m_rowHeights
[r
] + m_vgap
;
1367 if (m_colWidths
[c
] != -1)
1368 x
= x
+ m_colWidths
[c
] + m_hgap
;
1372 wxSize
wxFlexGridSizer::CalcMin()
1378 // Number of rows/columns can change as items are added or removed.
1379 if ( !CalcRowsCols(nrows
, ncols
) )
1382 m_rowHeights
.SetCount(nrows
);
1383 m_colWidths
.SetCount(ncols
);
1385 // We have to recalcuate the sizes in case the item minimum size has
1386 // changed since the previous layout, or the item has been hidden using
1387 // wxSizer::Show(). If all the items in a row/column are hidden, the final
1388 // dimension of the row/column will be -1, indicating that the column
1389 // itself is hidden.
1390 for( s
= m_rowHeights
.GetCount(), i
= 0; i
< s
; ++i
)
1391 m_rowHeights
[ i
] = -1;
1392 for( s
= m_colWidths
.GetCount(), i
= 0; i
< s
; ++i
)
1393 m_colWidths
[ i
] = -1;
1395 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1400 wxSizerItem
*item
= node
->GetData();
1401 if ( item
->IsShown() )
1403 wxSize
sz( item
->CalcMin() );
1404 int row
= i
/ ncols
;
1405 int col
= i
% ncols
;
1407 m_rowHeights
[ row
] = wxMax( wxMax( 0, sz
.y
), m_rowHeights
[ row
] );
1408 m_colWidths
[ col
] = wxMax( wxMax( 0, sz
.x
), m_colWidths
[ col
] );
1411 node
= node
->GetNext();
1415 AdjustForFlexDirection();
1417 // Sum total minimum size, including gaps between rows/columns.
1418 // -1 is used as a magic number meaning empty column.
1420 for (int col
= 0; col
< ncols
; col
++)
1421 if ( m_colWidths
[ col
] != -1 )
1422 width
+= m_colWidths
[ col
] + m_hgap
;
1427 for (int row
= 0; row
< nrows
; row
++)
1428 if ( m_rowHeights
[ row
] != -1 )
1429 height
+= m_rowHeights
[ row
] + m_vgap
;
1433 m_calculatedMinSize
= wxSize( width
, height
);
1434 return m_calculatedMinSize
;
1437 void wxFlexGridSizer::AdjustForFlexDirection()
1439 // the logic in CalcMin works when we resize flexibly in both directions
1440 // but maybe this is not the case
1441 if ( m_flexDirection
!= wxBOTH
)
1443 // select the array corresponding to the direction in which we do *not*
1445 wxArrayInt
& array
= m_flexDirection
== wxVERTICAL
? m_colWidths
1448 const size_t count
= array
.GetCount();
1450 // find the largest value in this array
1454 for ( n
= 0; n
< count
; ++n
)
1456 if ( array
[n
] > largest
)
1460 // and now fill it with the largest value
1461 for ( n
= 0; n
< count
; ++n
)
1469 void wxFlexGridSizer::AdjustForGrowables(const wxSize
& sz
, const wxSize
& minsz
,
1470 int nrows
, int ncols
)
1472 // what to do with the rows? by default, resize them proportionally
1473 if ( sz
.y
> minsz
.y
&& ( (m_flexDirection
& wxVERTICAL
) || (m_growMode
== wxFLEX_GROWMODE_SPECIFIED
) ) )
1475 int sum_proportions
= 0;
1476 int growable_space
= 0;
1479 for (idx
= 0; idx
< m_growableRows
.GetCount(); idx
++)
1481 // Since the number of rows/columns can change as items are
1482 // inserted/deleted, we need to verify at runtime that the
1483 // requested growable rows/columns are still valid.
1484 if (m_growableRows
[idx
] >= nrows
)
1487 // If all items in a row/column are hidden, that row/column will
1488 // have a dimension of -1. This causes the row/column to be
1489 // hidden completely.
1490 if (m_rowHeights
[ m_growableRows
[idx
] ] == -1)
1492 sum_proportions
+= m_growableRowsProportions
[idx
];
1493 growable_space
+= m_rowHeights
[ m_growableRows
[idx
] ];
1499 for (idx
= 0; idx
< m_growableRows
.GetCount(); idx
++)
1501 if (m_growableRows
[idx
] >= nrows
)
1503 if (m_rowHeights
[ m_growableRows
[idx
] ] == -1)
1504 m_rowHeights
[ m_growableRows
[idx
] ] = 0;
1507 int delta
= (sz
.y
- minsz
.y
);
1508 if (sum_proportions
== 0)
1509 delta
= (delta
/num
) + m_rowHeights
[ m_growableRows
[idx
] ];
1511 delta
= ((delta
+growable_space
)*m_growableRowsProportions
[idx
]) / sum_proportions
;
1512 m_rowHeights
[ m_growableRows
[idx
] ] = delta
;
1517 else if ( (m_growMode
== wxFLEX_GROWMODE_ALL
) && (sz
.y
> minsz
.y
) )
1519 // rounding problem?
1520 for ( int row
= 0; row
< nrows
; ++row
)
1521 m_rowHeights
[ row
] = sz
.y
/ nrows
;
1524 // the same logic as above but for the columns
1525 if ( sz
.x
> minsz
.x
&& ( (m_flexDirection
& wxHORIZONTAL
) || (m_growMode
== wxFLEX_GROWMODE_SPECIFIED
) ) )
1527 int sum_proportions
= 0;
1528 int growable_space
= 0;
1531 for (idx
= 0; idx
< m_growableCols
.GetCount(); idx
++)
1533 // Since the number of rows/columns can change as items are
1534 // inserted/deleted, we need to verify at runtime that the
1535 // requested growable rows/columns are still valid.
1536 if (m_growableCols
[idx
] >= ncols
)
1539 // If all items in a row/column are hidden, that row/column will
1540 // have a dimension of -1. This causes the column to be hidden
1542 if (m_colWidths
[ m_growableCols
[idx
] ] == -1)
1544 sum_proportions
+= m_growableColsProportions
[idx
];
1545 growable_space
+= m_colWidths
[ m_growableCols
[idx
] ];
1551 for (idx
= 0; idx
< m_growableCols
.GetCount(); idx
++)
1553 if (m_growableCols
[idx
] >= ncols
)
1555 if (m_colWidths
[ m_growableCols
[idx
] ] == -1)
1556 m_colWidths
[ m_growableCols
[idx
] ] = 0;
1559 int delta
= (sz
.x
- minsz
.x
);
1560 if (sum_proportions
== 0)
1561 delta
= (delta
/num
) + m_colWidths
[ m_growableCols
[idx
] ];
1563 delta
= ((delta
+growable_space
)*m_growableColsProportions
[idx
])/sum_proportions
;
1564 m_colWidths
[ m_growableCols
[idx
] ] = delta
;
1569 else if ( (m_growMode
== wxFLEX_GROWMODE_ALL
) && (sz
.x
> minsz
.x
) )
1571 for ( int col
=0; col
< ncols
; ++col
)
1572 m_colWidths
[ col
] = sz
.x
/ ncols
;
1577 void wxFlexGridSizer::AddGrowableRow( size_t idx
, int proportion
)
1579 m_growableRows
.Add( idx
);
1580 m_growableRowsProportions
.Add( proportion
);
1583 void wxFlexGridSizer::AddGrowableCol( size_t idx
, int proportion
)
1585 m_growableCols
.Add( idx
);
1586 m_growableColsProportions
.Add( proportion
);
1589 // helper function for RemoveGrowableCol/Row()
1591 DoRemoveFromArrays(size_t idx
, wxArrayInt
& items
, wxArrayInt
& proportions
)
1593 const size_t count
= items
.size();
1594 for ( size_t n
= 0; n
< count
; n
++ )
1596 if ( (size_t)items
[n
] == idx
)
1599 proportions
.RemoveAt(n
);
1604 wxFAIL_MSG( _T("column/row is already not growable") );
1607 void wxFlexGridSizer::RemoveGrowableCol( size_t idx
)
1609 DoRemoveFromArrays(idx
, m_growableCols
, m_growableColsProportions
);
1612 void wxFlexGridSizer::RemoveGrowableRow( size_t idx
)
1614 DoRemoveFromArrays(idx
, m_growableRows
, m_growableRowsProportions
);
1617 //---------------------------------------------------------------------------
1619 //---------------------------------------------------------------------------
1621 wxBoxSizer::wxBoxSizer( int orient
)
1622 : m_orient( orient
)
1626 void wxBoxSizer::RecalcSizes()
1628 if (m_children
.GetCount() == 0)
1634 if (m_orient
== wxHORIZONTAL
)
1635 delta
= m_size
.x
- m_fixedWidth
;
1637 delta
= m_size
.y
- m_fixedHeight
;
1640 wxPoint
pt( m_position
);
1642 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1645 wxSizerItem
*item
= node
->GetData();
1647 if (item
->IsShown())
1649 wxSize
size( item
->GetMinSizeWithBorder() );
1651 if (m_orient
== wxVERTICAL
)
1653 wxCoord height
= size
.y
;
1654 if (item
->GetProportion())
1656 // Because of at least one visible item has non-zero
1657 // proportion then m_stretchable is not zero
1658 height
= (delta
* item
->GetProportion()) / m_stretchable
;
1661 wxPoint
child_pos( pt
);
1662 wxSize
child_size( size
.x
, height
);
1664 if (item
->GetFlag() & (wxEXPAND
| wxSHAPED
))
1665 child_size
.x
= m_size
.x
;
1666 else if (item
->GetFlag() & wxALIGN_RIGHT
)
1667 child_pos
.x
+= m_size
.x
- size
.x
;
1668 else if (item
->GetFlag() & (wxCENTER
| wxALIGN_CENTER_HORIZONTAL
))
1669 // XXX wxCENTER is added for backward compatibility;
1670 // wxALIGN_CENTER should be used in new code
1671 child_pos
.x
+= (m_size
.x
- size
.x
) / 2;
1673 item
->SetDimension( child_pos
, child_size
);
1679 wxCoord width
= size
.x
;
1680 if (item
->GetProportion())
1682 // Because of at least one visible item has non-zero
1683 // proportion then m_stretchable is not zero
1684 width
= (delta
* item
->GetProportion()) / m_stretchable
;
1687 wxPoint
child_pos( pt
);
1688 wxSize
child_size( width
, size
.y
);
1690 if (item
->GetFlag() & (wxEXPAND
| wxSHAPED
))
1691 child_size
.y
= m_size
.y
;
1692 else if (item
->GetFlag() & wxALIGN_BOTTOM
)
1693 child_pos
.y
+= m_size
.y
- size
.y
;
1694 else if (item
->GetFlag() & (wxCENTER
| wxALIGN_CENTER_VERTICAL
))
1695 // XXX wxCENTER is added for backward compatibility;
1696 // wxALIGN_CENTER should be used in new code
1697 child_pos
.y
+= (m_size
.y
- size
.y
) / 2;
1699 if ( m_containingWindow
)
1701 child_pos
.x
= m_containingWindow
->AdjustForLayoutDirection
1709 item
->SetDimension( child_pos
, child_size
);
1715 node
= node
->GetNext();
1719 wxSize
wxBoxSizer::CalcMin()
1721 if (m_children
.GetCount() == 0)
1730 // precalc item minsizes and count proportions
1731 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1734 wxSizerItem
*item
= node
->GetData();
1736 if ( item
->IsShown() )
1738 item
->CalcMin(); // result is stored in the item
1740 m_stretchable
+= item
->GetProportion();
1743 node
= node
->GetNext();
1746 // Total minimum size (width or height) of sizer
1749 node
= m_children
.GetFirst();
1752 wxSizerItem
*item
= node
->GetData();
1754 if (item
->IsShown() && item
->GetProportion() != 0)
1756 int stretch
= item
->GetProportion();
1757 wxSize
size( item
->GetMinSizeWithBorder() );
1760 // Integer division rounded up is (a + b - 1) / b
1761 // Round up needed in order to guarantee that all
1762 // all items will have size not less then their min size
1763 if (m_orient
== wxHORIZONTAL
)
1764 minSize
= ( size
.x
*m_stretchable
+ stretch
- 1)/stretch
;
1766 minSize
= ( size
.y
*m_stretchable
+ stretch
- 1)/stretch
;
1768 if (minSize
> maxMinSize
)
1769 maxMinSize
= minSize
;
1771 node
= node
->GetNext();
1774 // Calculate overall minimum size
1775 node
= m_children
.GetFirst();
1778 wxSizerItem
*item
= node
->GetData();
1780 if (item
->IsShown())
1782 wxSize
size( item
->GetMinSizeWithBorder() );
1783 if (item
->GetProportion() != 0)
1785 if (m_orient
== wxHORIZONTAL
)
1786 size
.x
= (maxMinSize
*item
->GetProportion())/m_stretchable
;
1788 size
.y
= (maxMinSize
*item
->GetProportion())/m_stretchable
;
1792 if (m_orient
== wxVERTICAL
)
1794 m_fixedHeight
+= size
.y
;
1795 m_fixedWidth
= wxMax( m_fixedWidth
, size
.x
);
1799 m_fixedWidth
+= size
.x
;
1800 m_fixedHeight
= wxMax( m_fixedHeight
, size
.y
);
1804 if (m_orient
== wxHORIZONTAL
)
1806 m_minWidth
+= size
.x
;
1807 m_minHeight
= wxMax( m_minHeight
, size
.y
);
1811 m_minHeight
+= size
.y
;
1812 m_minWidth
= wxMax( m_minWidth
, size
.x
);
1815 node
= node
->GetNext();
1818 return wxSize( m_minWidth
, m_minHeight
);
1821 //---------------------------------------------------------------------------
1823 //---------------------------------------------------------------------------
1827 wxStaticBoxSizer::wxStaticBoxSizer( wxStaticBox
*box
, int orient
)
1828 : wxBoxSizer( orient
),
1831 wxASSERT_MSG( box
, wxT("wxStaticBoxSizer needs a static box") );
1833 // do this so that our Detach() is called if the static box is destroyed
1835 m_staticBox
->SetContainingSizer(this);
1838 wxStaticBoxSizer::wxStaticBoxSizer(int orient
, wxWindow
*win
, const wxString
& s
)
1839 : wxBoxSizer(orient
),
1840 m_staticBox(new wxStaticBox(win
, wxID_ANY
, s
))
1843 m_staticBox
->SetContainingSizer(this);
1846 wxStaticBoxSizer::~wxStaticBoxSizer()
1851 static void GetStaticBoxBorders( wxStaticBox
*box
,
1855 // this has to be done platform by platform as there is no way to
1856 // guess the thickness of a wxStaticBox border
1857 box
->GetBordersForSizer(borderTop
, borderOther
);
1860 void wxStaticBoxSizer::RecalcSizes()
1862 int top_border
, other_border
;
1863 GetStaticBoxBorders(m_staticBox
, &top_border
, &other_border
);
1865 m_staticBox
->SetSize( m_position
.x
, m_position
.y
, m_size
.x
, m_size
.y
);
1867 wxPoint
old_pos( m_position
);
1868 m_position
.x
+= other_border
;
1869 m_position
.y
+= top_border
;
1870 wxSize
old_size( m_size
);
1871 m_size
.x
-= 2*other_border
;
1872 m_size
.y
-= top_border
+ other_border
;
1874 wxBoxSizer::RecalcSizes();
1876 m_position
= old_pos
;
1880 wxSize
wxStaticBoxSizer::CalcMin()
1882 int top_border
, other_border
;
1883 GetStaticBoxBorders(m_staticBox
, &top_border
, &other_border
);
1885 wxSize
ret( wxBoxSizer::CalcMin() );
1886 ret
.x
+= 2*other_border
;
1887 ret
.y
+= other_border
+ top_border
;
1892 void wxStaticBoxSizer::ShowItems( bool show
)
1894 m_staticBox
->Show( show
);
1895 wxBoxSizer::ShowItems( show
);
1898 bool wxStaticBoxSizer::Detach( wxWindow
*window
)
1900 // avoid deleting m_staticBox in our dtor if it's being detached from the
1901 // sizer (which can happen because it's being already destroyed for
1903 if ( window
== m_staticBox
)
1909 return wxSizer::Detach( window
);
1912 #endif // wxUSE_STATBOX
1916 wxStdDialogButtonSizer::wxStdDialogButtonSizer()
1917 : wxBoxSizer(wxHORIZONTAL
)
1919 // Vertical buttons with lots of space on either side
1920 // looks rubbish on WinCE, so let's not do this for now.
1921 // If we are going to use vertical buttons, we should
1922 // put the sizer to the right of other controls in the dialog,
1923 // and that's beyond the scope of this sizer.
1925 bool is_pda
= (wxSystemSettings::GetScreenType() <= wxSYS_SCREEN_PDA
);
1926 // If we have a PDA screen, put yes/no button over
1927 // all other buttons, otherwise on the left side.
1929 m_orient
= wxVERTICAL
;
1932 m_buttonAffirmative
= NULL
;
1933 m_buttonApply
= NULL
;
1934 m_buttonNegative
= NULL
;
1935 m_buttonCancel
= NULL
;
1936 m_buttonHelp
= NULL
;
1939 void wxStdDialogButtonSizer::AddButton(wxButton
*mybutton
)
1941 switch (mybutton
->GetId())
1946 m_buttonAffirmative
= mybutton
;
1949 m_buttonApply
= mybutton
;
1952 m_buttonNegative
= mybutton
;
1955 m_buttonCancel
= mybutton
;
1958 case wxID_CONTEXT_HELP
:
1959 m_buttonHelp
= mybutton
;
1966 void wxStdDialogButtonSizer::SetAffirmativeButton( wxButton
*button
)
1968 m_buttonAffirmative
= button
;
1971 void wxStdDialogButtonSizer::SetNegativeButton( wxButton
*button
)
1973 m_buttonNegative
= button
;
1976 void wxStdDialogButtonSizer::SetCancelButton( wxButton
*button
)
1978 m_buttonCancel
= button
;
1981 void wxStdDialogButtonSizer::Realize()
1984 Add(0, 0, 0, wxLEFT
, 6);
1986 Add((wxWindow
*)m_buttonHelp
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 6);
1988 if (m_buttonNegative
){
1989 // HIG POLICE BULLETIN - destructive buttons need extra padding
1990 // 24 pixels on either side
1991 Add((wxWindow
*)m_buttonNegative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 12);
1994 // extra whitespace between help/negative and cancel/ok buttons
1995 Add(0, 0, 1, wxEXPAND
, 0);
1997 if (m_buttonCancel
){
1998 Add((wxWindow
*)m_buttonCancel
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 6);
1999 // Cancel or help should be default
2000 // m_buttonCancel->SetDefaultButton();
2003 // Ugh, Mac doesn't really have apply dialogs, so I'll just
2004 // figure the best place is between Cancel and OK
2006 Add((wxWindow
*)m_buttonApply
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 6);
2008 if (m_buttonAffirmative
){
2009 Add((wxWindow
*)m_buttonAffirmative
, 0, wxALIGN_CENTRE
| wxLEFT
, 6);
2011 if (m_buttonAffirmative
->GetId() == wxID_SAVE
){
2012 // these buttons have set labels under Mac so we should use them
2013 m_buttonAffirmative
->SetLabel(_("Save"));
2014 m_buttonNegative
->SetLabel(_("Don't Save"));
2018 // Extra space around and at the right
2020 #elif defined(__WXGTK20__)
2021 Add(0, 0, 0, wxLEFT
, 9);
2023 Add((wxWindow
*)m_buttonHelp
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 3);
2025 // extra whitespace between help and cancel/ok buttons
2026 Add(0, 0, 1, wxEXPAND
, 0);
2028 if (m_buttonNegative
){
2029 Add((wxWindow
*)m_buttonNegative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 3);
2032 if (m_buttonCancel
){
2033 Add((wxWindow
*)m_buttonCancel
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 3);
2034 // Cancel or help should be default
2035 // m_buttonCancel->SetDefaultButton();
2039 Add((wxWindow
*)m_buttonApply
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 3);
2041 if (m_buttonAffirmative
)
2042 Add((wxWindow
*)m_buttonAffirmative
, 0, wxALIGN_CENTRE
| wxLEFT
, 6);
2043 #elif defined(__WXMSW__)
2046 // right-justify buttons
2047 Add(0, 0, 1, wxEXPAND
, 0);
2049 if (m_buttonAffirmative
){
2050 Add((wxWindow
*)m_buttonAffirmative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonAffirmative
->ConvertDialogToPixels(wxSize(2, 0)).x
);
2053 if (m_buttonNegative
){
2054 Add((wxWindow
*)m_buttonNegative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonNegative
->ConvertDialogToPixels(wxSize(2, 0)).x
);
2057 if (m_buttonCancel
){
2058 Add((wxWindow
*)m_buttonCancel
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonCancel
->ConvertDialogToPixels(wxSize(2, 0)).x
);
2061 Add((wxWindow
*)m_buttonApply
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonApply
->ConvertDialogToPixels(wxSize(2, 0)).x
);
2064 Add((wxWindow
*)m_buttonHelp
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonHelp
->ConvertDialogToPixels(wxSize(2, 0)).x
);
2066 // GTK+1 and any other platform
2068 // Add(0, 0, 0, wxLEFT, 5); // Not sure what this was for but it unbalances the dialog
2070 Add((wxWindow
*)m_buttonHelp
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonHelp
->ConvertDialogToPixels(wxSize(4, 0)).x
);
2072 // extra whitespace between help and cancel/ok buttons
2073 Add(0, 0, 1, wxEXPAND
, 0);
2076 Add((wxWindow
*)m_buttonApply
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonApply
->ConvertDialogToPixels(wxSize(4, 0)).x
);
2078 if (m_buttonAffirmative
){
2079 Add((wxWindow
*)m_buttonAffirmative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonAffirmative
->ConvertDialogToPixels(wxSize(4, 0)).x
);
2082 if (m_buttonNegative
){
2083 Add((wxWindow
*)m_buttonNegative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonNegative
->ConvertDialogToPixels(wxSize(4, 0)).x
);
2086 if (m_buttonCancel
){
2087 Add((wxWindow
*)m_buttonCancel
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonCancel
->ConvertDialogToPixels(wxSize(4, 0)).x
);
2088 // Cancel or help should be default
2089 // m_buttonCancel->SetDefaultButton();
2095 #endif // wxUSE_BUTTON
2097 #if WXWIN_COMPATIBILITY_2_4
2099 // ----------------------------------------------------------------------------
2101 // ----------------------------------------------------------------------------
2104 IMPLEMENT_CLASS(wxBookCtrlSizer
, wxSizer
)
2106 IMPLEMENT_CLASS(wxNotebookSizer
, wxBookCtrlSizer
)
2107 #endif // wxUSE_NOTEBOOK
2108 #endif // wxUSE_BOOKCTRL
2112 #if WXWIN_COMPATIBILITY_2_6
2114 wxBookCtrlSizer::wxBookCtrlSizer(wxBookCtrlBase
*bookctrl
)
2115 : m_bookctrl(bookctrl
)
2117 wxASSERT_MSG( bookctrl
, wxT("wxBookCtrlSizer needs a control") );
2120 #endif // WXWIN_COMPATIBILITY_2_6
2122 void wxBookCtrlSizer::RecalcSizes()
2124 m_bookctrl
->SetSize( m_position
.x
, m_position
.y
, m_size
.x
, m_size
.y
);
2127 wxSize
wxBookCtrlSizer::CalcMin()
2129 wxSize sizeBorder
= m_bookctrl
->CalcSizeFromPage(wxSize(0,0));
2134 if ( m_bookctrl
->GetPageCount() == 0 )
2136 return wxSize(sizeBorder
.x
+ 10, sizeBorder
.y
+ 10);
2142 wxWindowList::compatibility_iterator
2143 node
= m_bookctrl
->GetChildren().GetFirst();
2146 wxWindow
*item
= node
->GetData();
2147 wxSizer
*itemsizer
= item
->GetSizer();
2151 wxSize
subsize( itemsizer
->CalcMin() );
2153 if (subsize
.x
> maxX
)
2155 if (subsize
.y
> maxY
)
2159 node
= node
->GetNext();
2162 return wxSize( maxX
, maxY
) + sizeBorder
;
2167 #if WXWIN_COMPATIBILITY_2_6
2169 wxNotebookSizer::wxNotebookSizer(wxNotebook
*nb
)
2171 wxASSERT_MSG( nb
, wxT("wxNotebookSizer needs a control") );
2175 #endif // WXWIN_COMPATIBILITY_2_6
2177 #endif // wxUSE_NOTEBOOOK
2178 #endif // wxUSE_BOOKCTRL
2180 #endif // WXWIN_COMPATIBILITY_2_4