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"
21 #include "wx/string.h"
25 #include "wx/settings.h"
29 #include "wx/statbox.h"
30 #include "wx/listimpl.cpp"
32 #if WXWIN_COMPATIBILITY_2_4
33 #include "wx/notebook.h"
36 //---------------------------------------------------------------------------
38 IMPLEMENT_CLASS(wxSizerItem
, wxObject
)
39 IMPLEMENT_CLASS(wxSizer
, wxObject
)
40 IMPLEMENT_CLASS(wxGridSizer
, wxSizer
)
41 IMPLEMENT_CLASS(wxFlexGridSizer
, wxGridSizer
)
42 IMPLEMENT_CLASS(wxBoxSizer
, wxSizer
)
44 IMPLEMENT_CLASS(wxStaticBoxSizer
, wxBoxSizer
)
47 IMPLEMENT_CLASS(wxStdDialogButtonSizer
, wxBoxSizer
)
50 WX_DEFINE_EXPORTED_LIST( wxSizerItemList
)
85 // ----------------------------------------------------------------------------
87 // ----------------------------------------------------------------------------
89 void wxSizerItem::Init(const wxSizerFlags
& flags
)
93 m_proportion
= flags
.GetProportion();
94 m_flag
= flags
.GetFlags();
95 m_border
= flags
.GetBorderInPixels();
98 wxSizerItem::wxSizerItem()
110 void wxSizerItem::SetWindow(wxWindow
*window
)
112 wxCHECK_RET( window
, _T("NULL window in wxSizerItem::SetWindow()") );
114 m_kind
= Item_Window
;
117 // window doesn't become smaller than its initial size, whatever happens
118 m_minSize
= window
->GetSize();
120 if ( m_flag
& wxFIXED_MINSIZE
)
121 window
->SetMinSize(m_minSize
);
123 // aspect ratio calculated from initial size
127 wxSizerItem::wxSizerItem(wxWindow
*window
,
132 : m_proportion(proportion
),
141 void wxSizerItem::SetSizer(wxSizer
*sizer
)
147 wxSizerItem::wxSizerItem(wxSizer
*sizer
,
152 : m_proportion(proportion
),
160 // m_minSize is set later
164 void wxSizerItem::SetSpacer(const wxSize
& size
)
166 m_kind
= Item_Spacer
;
167 m_spacer
= new wxSizerSpacer(size
);
172 wxSizerItem::wxSizerItem(int width
,
178 : m_minSize(width
, height
), // minimal size is the initial size
179 m_proportion(proportion
),
184 SetSpacer(width
, height
);
187 wxSizerItem::~wxSizerItem()
197 m_window
->SetContainingSizer(NULL
);
210 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
214 wxSize
wxSizerItem::GetSpacer() const
217 if ( m_kind
== Item_Spacer
)
218 size
= m_spacer
->GetSize();
224 wxSize
wxSizerItem::GetSize() const
233 ret
= m_window
->GetSize();
237 ret
= m_sizer
->GetSize();
241 ret
= m_spacer
->GetSize();
246 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
253 if (m_flag
& wxNORTH
)
255 if (m_flag
& wxSOUTH
)
261 wxSize
wxSizerItem::CalcMin()
265 m_minSize
= m_sizer
->GetMinSize();
267 // if we have to preserve aspect ratio _AND_ this is
268 // the first-time calculation, consider ret to be initial size
269 if ( (m_flag
& wxSHAPED
) && wxIsNullDouble(m_ratio
) )
272 else if ( IsWindow() )
274 // Since the size of the window may change during runtime, we
275 // should use the current minimal/best size.
276 m_minSize
= m_window
->GetBestFittingSize();
279 return GetMinSizeWithBorder();
282 wxSize
wxSizerItem::GetMinSizeWithBorder() const
284 wxSize ret
= m_minSize
;
290 if (m_flag
& wxNORTH
)
292 if (m_flag
& wxSOUTH
)
299 void wxSizerItem::SetDimension( const wxPoint
& pos_
, const wxSize
& size_
)
303 if (m_flag
& wxSHAPED
)
305 // adjust aspect ratio
306 int rwidth
= (int) (size
.y
* m_ratio
);
310 int rheight
= (int) (size
.x
/ m_ratio
);
311 // add vertical space
312 if (m_flag
& wxALIGN_CENTER_VERTICAL
)
313 pos
.y
+= (size
.y
- rheight
) / 2;
314 else if (m_flag
& wxALIGN_BOTTOM
)
315 pos
.y
+= (size
.y
- rheight
);
316 // use reduced dimensions
319 else if (rwidth
< size
.x
)
321 // add horizontal space
322 if (m_flag
& wxALIGN_CENTER_HORIZONTAL
)
323 pos
.x
+= (size
.x
- rwidth
) / 2;
324 else if (m_flag
& wxALIGN_RIGHT
)
325 pos
.x
+= (size
.x
- rwidth
);
330 // This is what GetPosition() returns. Since we calculate
331 // borders afterwards, GetPosition() will be the left/top
332 // corner of the surrounding border.
344 if (m_flag
& wxNORTH
)
349 if (m_flag
& wxSOUTH
)
354 m_rect
= wxRect(pos
, size
);
359 wxFAIL_MSG( _T("can't set size of uninitialized sizer item") );
363 m_window
->SetSize(pos
.x
, pos
.y
, size
.x
, size
.y
,
364 wxSIZE_ALLOW_MINUS_ONE
);
368 m_sizer
->SetDimension(pos
.x
, pos
.y
, size
.x
, size
.y
);
372 m_spacer
->SetSize(size
);
377 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
381 void wxSizerItem::DeleteWindows()
390 //We are deleting the window from this sizer - normally
391 //the window destroys the sizer associated with it,
392 //which might destroy this, which we don't want
393 m_window
->SetContainingSizer(NULL
);
395 //Putting this after the switch will result in a spacer
396 //not being deleted properly on destruction
401 m_sizer
->DeleteWindows();
406 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
411 void wxSizerItem::Show( bool show
)
416 wxFAIL_MSG( _T("can't show uninitialized sizer item") );
420 m_window
->Show(show
);
428 m_spacer
->Show(show
);
433 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
437 bool wxSizerItem::IsShown() const
442 // we may be called from CalcMin(), just return false so that we're
447 return m_window
->IsShown();
450 // arbitrarily decide that if at least one of our elements is
451 // shown, so are we (this arbitrariness is the reason for
452 // deprecating this function)
454 for ( wxSizerItemList::compatibility_iterator
455 node
= m_sizer
->GetChildren().GetFirst();
457 node
= node
->GetNext() )
459 if ( node
->GetData()->IsShown() )
466 return m_spacer
->IsShown();
470 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
476 #if WXWIN_COMPATIBILITY_2_6
477 void wxSizerItem::SetOption( int option
)
479 SetProportion( option
);
482 int wxSizerItem::GetOption() const
484 return GetProportion();
486 #endif // WXWIN_COMPATIBILITY_2_6
489 //---------------------------------------------------------------------------
491 //---------------------------------------------------------------------------
495 WX_CLEAR_LIST(wxSizerItemList
, m_children
);
498 wxSizerItem
* wxSizer::Insert( size_t index
, wxSizerItem
*item
)
500 m_children
.Insert( index
, item
);
502 if ( item
->GetWindow() )
503 item
->GetWindow()->SetContainingSizer( this );
508 #if WXWIN_COMPATIBILITY_2_6
509 bool wxSizer::Remove( wxWindow
*window
)
511 return Detach( window
);
513 #endif // WXWIN_COMPATIBILITY_2_6
515 bool wxSizer::Remove( wxSizer
*sizer
)
517 wxASSERT_MSG( sizer
, _T("Removing NULL sizer") );
519 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
522 wxSizerItem
*item
= node
->GetData();
524 if (item
->GetSizer() == sizer
)
527 m_children
.Erase( node
);
531 node
= node
->GetNext();
537 bool wxSizer::Remove( int index
)
539 wxCHECK_MSG( index
>= 0 && (size_t)index
< m_children
.GetCount(),
541 _T("Remove index is out of range") );
543 wxSizerItemList::compatibility_iterator node
= m_children
.Item( index
);
545 wxCHECK_MSG( node
, false, _T("Failed to find child node") );
547 wxSizerItem
*item
= node
->GetData();
549 if ( item
->IsWindow() )
550 item
->GetWindow()->SetContainingSizer( NULL
);
553 m_children
.Erase( node
);
557 bool wxSizer::Detach( wxSizer
*sizer
)
559 wxASSERT_MSG( sizer
, _T("Detaching NULL sizer") );
561 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
564 wxSizerItem
*item
= node
->GetData();
566 if (item
->GetSizer() == sizer
)
570 m_children
.Erase( node
);
573 node
= node
->GetNext();
579 bool wxSizer::Detach( wxWindow
*window
)
581 wxASSERT_MSG( window
, _T("Detaching NULL window") );
583 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
586 wxSizerItem
*item
= node
->GetData();
588 if (item
->GetWindow() == window
)
590 item
->GetWindow()->SetContainingSizer( NULL
);
592 m_children
.Erase( node
);
595 node
= node
->GetNext();
601 bool wxSizer::Detach( int index
)
603 wxCHECK_MSG( index
>= 0 && (size_t)index
< m_children
.GetCount(),
605 _T("Detach index is out of range") );
607 wxSizerItemList::compatibility_iterator node
= m_children
.Item( index
);
609 wxCHECK_MSG( node
, false, _T("Failed to find child node") );
611 wxSizerItem
*item
= node
->GetData();
613 if ( item
->IsSizer() )
615 else if ( item
->IsWindow() )
616 item
->GetWindow()->SetContainingSizer( NULL
);
619 m_children
.Erase( node
);
623 void wxSizer::Clear( bool delete_windows
)
625 // First clear the ContainingSizer pointers
626 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
629 wxSizerItem
*item
= node
->GetData();
631 if (item
->IsWindow())
632 item
->GetWindow()->SetContainingSizer( NULL
);
633 node
= node
->GetNext();
636 // Destroy the windows if needed
640 // Now empty the list
641 WX_CLEAR_LIST(wxSizerItemList
, m_children
);
644 void wxSizer::DeleteWindows()
646 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
649 wxSizerItem
*item
= node
->GetData();
651 item
->DeleteWindows();
652 node
= node
->GetNext();
656 wxSize
wxSizer::Fit( wxWindow
*window
)
658 wxSize
size(window
->IsTopLevel() ? FitSize(window
)
659 : GetMinWindowSize(window
));
661 window
->SetSize( size
);
666 void wxSizer::FitInside( wxWindow
*window
)
669 if (window
->IsTopLevel())
670 size
= VirtualFitSize( window
);
672 size
= GetMinClientSize( window
);
674 window
->SetVirtualSize( size
);
677 void wxSizer::Layout()
679 // (re)calculates minimums needed for each item and other preparations
683 // Applies the layout and repositions/resizes the items
687 void wxSizer::SetSizeHints( wxWindow
*window
)
689 // Preserve the window's max size hints, but set the
690 // lower bound according to the sizer calculations.
692 wxSize size
= Fit( window
);
694 window
->SetSizeHints( size
.x
,
696 window
->GetMaxWidth(),
697 window
->GetMaxHeight() );
700 void wxSizer::SetVirtualSizeHints( wxWindow
*window
)
702 // Preserve the window's max size hints, but set the
703 // lower bound according to the sizer calculations.
706 wxSize
size( window
->GetVirtualSize() );
707 window
->SetVirtualSizeHints( size
.x
,
709 window
->GetMaxWidth(),
710 window
->GetMaxHeight() );
713 wxSize
wxSizer::GetMaxWindowSize( wxWindow
*window
) const
715 return window
->GetMaxSize();
718 wxSize
wxSizer::GetMinWindowSize( wxWindow
*window
)
720 wxSize
minSize( GetMinSize() );
721 wxSize
size( window
->GetSize() );
722 wxSize
client_size( window
->GetClientSize() );
724 return wxSize( minSize
.x
+size
.x
-client_size
.x
,
725 minSize
.y
+size
.y
-client_size
.y
);
728 // TODO on mac we need a function that determines how much free space this
729 // min size contains, in order to make sure that we have 20 pixels of free
730 // space around the controls
732 // Return a window size that will fit within the screens dimensions
733 wxSize
wxSizer::FitSize( wxWindow
*window
)
735 if ( window
->IsTopLevel() )
737 wxTopLevelWindow
*tlw
= wxDynamicCast(window
, wxTopLevelWindow
);
738 if ( tlw
&& tlw
->IsAlwaysMaximized() )
740 return tlw
->GetClientSize();
744 wxSize size
= GetMinWindowSize( window
);
745 wxSize sizeMax
= GetMaxWindowSize( window
);
747 // Limit the size if sizeMax != wxDefaultSize
749 if ( size
.x
> sizeMax
.x
&& sizeMax
.x
!= wxDefaultCoord
)
751 if ( size
.y
> sizeMax
.y
&& sizeMax
.y
!= wxDefaultCoord
)
757 wxSize
wxSizer::GetMaxClientSize( wxWindow
*window
) const
759 wxSize
maxSize( window
->GetMaxSize() );
761 if ( maxSize
!= wxDefaultSize
)
763 wxSize
size( window
->GetSize() );
764 wxSize
client_size( window
->GetClientSize() );
766 return wxSize( maxSize
.x
+ client_size
.x
- size
.x
,
767 maxSize
.y
+ client_size
.y
- size
.y
);
770 return wxDefaultSize
;
773 wxSize
wxSizer::GetMinClientSize( wxWindow
*WXUNUSED(window
) )
775 return GetMinSize(); // Already returns client size.
778 wxSize
wxSizer::VirtualFitSize( wxWindow
*window
)
780 wxSize size
= GetMinClientSize( window
);
781 wxSize sizeMax
= GetMaxClientSize( window
);
783 // Limit the size if sizeMax != wxDefaultSize
785 if ( size
.x
> sizeMax
.x
&& sizeMax
.x
!= wxDefaultCoord
)
787 if ( size
.y
> sizeMax
.y
&& sizeMax
.y
!= wxDefaultCoord
)
793 void wxSizer::SetDimension( int x
, int y
, int width
, int height
)
802 wxSize
wxSizer::GetMinSize()
804 wxSize
ret( CalcMin() );
805 if (ret
.x
< m_minSize
.x
) ret
.x
= m_minSize
.x
;
806 if (ret
.y
< m_minSize
.y
) ret
.y
= m_minSize
.y
;
810 void wxSizer::DoSetMinSize( int width
, int height
)
813 m_minSize
.y
= height
;
816 bool wxSizer::DoSetItemMinSize( wxWindow
*window
, int width
, int height
)
818 wxASSERT_MSG( window
, _T("SetMinSize for NULL window") );
820 // Is it our immediate child?
822 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
825 wxSizerItem
*item
= node
->GetData();
827 if (item
->GetWindow() == window
)
829 item
->SetMinSize( width
, height
);
832 node
= node
->GetNext();
835 // No? Search any subsizers we own then
837 node
= m_children
.GetFirst();
840 wxSizerItem
*item
= node
->GetData();
842 if ( item
->GetSizer() &&
843 item
->GetSizer()->DoSetItemMinSize( window
, width
, height
) )
845 // A child sizer found the requested windw, exit.
848 node
= node
->GetNext();
854 bool wxSizer::DoSetItemMinSize( wxSizer
*sizer
, int width
, int height
)
856 wxASSERT_MSG( sizer
, _T("SetMinSize for NULL sizer") );
858 // Is it our immediate child?
860 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
863 wxSizerItem
*item
= node
->GetData();
865 if (item
->GetSizer() == sizer
)
867 item
->GetSizer()->DoSetMinSize( width
, height
);
870 node
= node
->GetNext();
873 // No? Search any subsizers we own then
875 node
= m_children
.GetFirst();
878 wxSizerItem
*item
= node
->GetData();
880 if ( item
->GetSizer() &&
881 item
->GetSizer()->DoSetItemMinSize( sizer
, width
, height
) )
883 // A child found the requested sizer, exit.
886 node
= node
->GetNext();
892 bool wxSizer::DoSetItemMinSize( size_t index
, int width
, int height
)
894 wxSizerItemList::compatibility_iterator node
= m_children
.Item( index
);
896 wxCHECK_MSG( node
, false, _T("Failed to find child node") );
898 wxSizerItem
*item
= node
->GetData();
900 if (item
->GetSizer())
902 // Sizers contains the minimal size in them, if not calculated ...
903 item
->GetSizer()->DoSetMinSize( width
, height
);
907 // ... but the minimal size of spacers and windows is stored via the item
908 item
->SetMinSize( width
, height
);
914 wxSizerItem
* wxSizer::GetItem( wxWindow
*window
, bool recursive
)
916 wxASSERT_MSG( window
, _T("GetItem for NULL window") );
918 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
921 wxSizerItem
*item
= node
->GetData();
923 if (item
->GetWindow() == window
)
927 else if (recursive
&& item
->IsSizer())
929 wxSizerItem
*subitem
= item
->GetSizer()->GetItem( window
, true );
934 node
= node
->GetNext();
940 wxSizerItem
* wxSizer::GetItem( wxSizer
*sizer
, bool recursive
)
942 wxASSERT_MSG( sizer
, _T("GetItem for NULL sizer") );
944 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
947 wxSizerItem
*item
= node
->GetData();
949 if (item
->GetSizer() == sizer
)
953 else if (recursive
&& item
->IsSizer())
955 wxSizerItem
*subitem
= item
->GetSizer()->GetItem( sizer
, true );
960 node
= node
->GetNext();
966 wxSizerItem
* wxSizer::GetItem( size_t index
)
968 wxCHECK_MSG( index
< m_children
.GetCount(),
970 _T("GetItem index is out of range") );
972 return m_children
.Item( index
)->GetData();
975 bool wxSizer::Show( wxWindow
*window
, bool show
, bool recursive
)
977 wxSizerItem
*item
= GetItem( window
, recursive
);
988 bool wxSizer::Show( wxSizer
*sizer
, bool show
, bool recursive
)
990 wxSizerItem
*item
= GetItem( sizer
, recursive
);
1001 bool wxSizer::Show( size_t index
, bool show
)
1003 wxSizerItem
*item
= GetItem( index
);
1014 void wxSizer::ShowItems( bool show
)
1016 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1019 node
->GetData()->Show( show
);
1020 node
= node
->GetNext();
1024 bool wxSizer::IsShown( wxWindow
*window
) const
1026 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1029 wxSizerItem
*item
= node
->GetData();
1031 if (item
->GetWindow() == window
)
1033 return item
->IsShown();
1035 node
= node
->GetNext();
1038 wxFAIL_MSG( _T("IsShown failed to find sizer item") );
1043 bool wxSizer::IsShown( wxSizer
*sizer
) const
1045 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1048 wxSizerItem
*item
= node
->GetData();
1050 if (item
->GetSizer() == sizer
)
1052 return item
->IsShown();
1054 node
= node
->GetNext();
1057 wxFAIL_MSG( _T("IsShown failed to find sizer item") );
1062 bool wxSizer::IsShown( size_t index
) const
1064 wxCHECK_MSG( index
< m_children
.GetCount(),
1066 _T("IsShown index is out of range") );
1068 return m_children
.Item( index
)->GetData()->IsShown();
1072 //---------------------------------------------------------------------------
1074 //---------------------------------------------------------------------------
1076 wxGridSizer::wxGridSizer( int rows
, int cols
, int vgap
, int hgap
)
1077 : m_rows( ( cols
== 0 && rows
== 0 ) ? 1 : rows
)
1084 wxGridSizer::wxGridSizer( int cols
, int vgap
, int hgap
)
1085 : m_rows( cols
== 0 ? 1 : 0 )
1092 int wxGridSizer::CalcRowsCols(int& nrows
, int& ncols
) const
1094 int nitems
= m_children
.GetCount();
1100 nrows
= (nitems
+ m_cols
- 1) / m_cols
;
1104 ncols
= (nitems
+ m_rows
- 1) / m_rows
;
1107 else // 0 columns, 0 rows?
1109 wxFAIL_MSG( _T("grid sizer must have either rows or columns fixed") );
1118 void wxGridSizer::RecalcSizes()
1120 int nitems
, nrows
, ncols
;
1121 if ( (nitems
= CalcRowsCols(nrows
, ncols
)) == 0 )
1124 wxSize
sz( GetSize() );
1125 wxPoint
pt( GetPosition() );
1127 int w
= (sz
.x
- (ncols
- 1) * m_hgap
) / ncols
;
1128 int h
= (sz
.y
- (nrows
- 1) * m_vgap
) / nrows
;
1131 for (int c
= 0; c
< ncols
; c
++)
1134 for (int r
= 0; r
< nrows
; r
++)
1136 int i
= r
* ncols
+ c
;
1139 wxSizerItemList::compatibility_iterator node
= m_children
.Item( i
);
1141 wxASSERT_MSG( node
, _T("Failed to find SizerItemList node") );
1143 SetItemBounds( node
->GetData(), x
, y
, w
, h
);
1151 wxSize
wxGridSizer::CalcMin()
1154 if ( CalcRowsCols(nrows
, ncols
) == 0 )
1155 return wxSize(10, 10);
1157 // Find the max width and height for any component
1161 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1164 wxSizerItem
*item
= node
->GetData();
1165 wxSize
sz( item
->CalcMin() );
1167 w
= wxMax( w
, sz
.x
);
1168 h
= wxMax( h
, sz
.y
);
1170 node
= node
->GetNext();
1173 return wxSize( ncols
* w
+ (ncols
-1) * m_hgap
,
1174 nrows
* h
+ (nrows
-1) * m_vgap
);
1177 void wxGridSizer::SetItemBounds( wxSizerItem
*item
, int x
, int y
, int w
, int h
)
1180 wxSize
sz( item
->GetMinSizeWithBorder() );
1181 int flag
= item
->GetFlag();
1183 if ((flag
& wxEXPAND
) || (flag
& wxSHAPED
))
1189 if (flag
& wxALIGN_CENTER_HORIZONTAL
)
1191 pt
.x
= x
+ (w
- sz
.x
) / 2;
1193 else if (flag
& wxALIGN_RIGHT
)
1195 pt
.x
= x
+ (w
- sz
.x
);
1198 if (flag
& wxALIGN_CENTER_VERTICAL
)
1200 pt
.y
= y
+ (h
- sz
.y
) / 2;
1202 else if (flag
& wxALIGN_BOTTOM
)
1204 pt
.y
= y
+ (h
- sz
.y
);
1208 item
->SetDimension(pt
, sz
);
1211 //---------------------------------------------------------------------------
1213 //---------------------------------------------------------------------------
1215 wxFlexGridSizer::wxFlexGridSizer( int rows
, int cols
, int vgap
, int hgap
)
1216 : wxGridSizer( rows
, cols
, vgap
, hgap
),
1217 m_flexDirection(wxBOTH
),
1218 m_growMode(wxFLEX_GROWMODE_SPECIFIED
)
1222 wxFlexGridSizer::wxFlexGridSizer( int cols
, int vgap
, int hgap
)
1223 : wxGridSizer( cols
, vgap
, hgap
),
1224 m_flexDirection(wxBOTH
),
1225 m_growMode(wxFLEX_GROWMODE_SPECIFIED
)
1229 wxFlexGridSizer::~wxFlexGridSizer()
1233 void wxFlexGridSizer::RecalcSizes()
1235 int nitems
, nrows
, ncols
;
1236 if ( (nitems
= CalcRowsCols(nrows
, ncols
)) == 0 )
1239 wxPoint
pt( GetPosition() );
1240 wxSize
sz( GetSize() );
1242 AdjustForGrowables(sz
, m_calculatedMinSize
, nrows
, ncols
);
1244 sz
= wxSize( pt
.x
+ sz
.x
, pt
.y
+ sz
.y
);
1247 for (int c
= 0; c
< ncols
; c
++)
1250 for (int r
= 0; r
< nrows
; r
++)
1252 int i
= r
* ncols
+ c
;
1255 wxSizerItemList::compatibility_iterator node
= m_children
.Item( i
);
1257 wxASSERT_MSG( node
, _T("Failed to find node") );
1259 int w
= wxMax( 0, wxMin( m_colWidths
[c
], sz
.x
- x
) );
1260 int h
= wxMax( 0, wxMin( m_rowHeights
[r
], sz
.y
- y
) );
1262 SetItemBounds( node
->GetData(), x
, y
, w
, h
);
1264 if (m_rowHeights
[r
] != -1)
1265 y
= y
+ m_rowHeights
[r
] + m_vgap
;
1267 if (m_colWidths
[c
] != -1)
1268 x
= x
+ m_colWidths
[c
] + m_hgap
;
1272 wxSize
wxFlexGridSizer::CalcMin()
1278 // Number of rows/columns can change as items are added or removed.
1279 if ( !CalcRowsCols(nrows
, ncols
) )
1280 return wxSize(10, 10);
1282 m_rowHeights
.SetCount(nrows
);
1283 m_colWidths
.SetCount(ncols
);
1285 // We have to recalcuate the sizes in case the item minimum size has
1286 // changed since the previous layout, or the item has been hidden using
1287 // wxSizer::Show(). If all the items in a row/column are hidden, the final
1288 // dimension of the row/column will be -1, indicating that the column
1289 // itself is hidden.
1290 for( s
= m_rowHeights
.GetCount(), i
= 0; i
< s
; ++i
)
1291 m_rowHeights
[ i
] = -1;
1292 for( s
= m_colWidths
.GetCount(), i
= 0; i
< s
; ++i
)
1293 m_colWidths
[ i
] = -1;
1295 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1300 wxSizerItem
*item
= node
->GetData();
1301 if ( item
->IsShown() )
1303 wxSize
sz( item
->CalcMin() );
1304 int row
= i
/ ncols
;
1305 int col
= i
% ncols
;
1307 m_rowHeights
[ row
] = wxMax( wxMax( 0, sz
.y
), m_rowHeights
[ row
] );
1308 m_colWidths
[ col
] = wxMax( wxMax( 0, sz
.x
), m_colWidths
[ col
] );
1311 node
= node
->GetNext();
1315 AdjustForFlexDirection();
1317 // Sum total minimum size, including gaps between rows/columns.
1318 // -1 is used as a magic number meaning empty column.
1320 for (int col
= 0; col
< ncols
; col
++)
1321 if ( m_colWidths
[ col
] != -1 )
1322 width
+= m_colWidths
[ col
] + m_hgap
;
1327 for (int row
= 0; row
< nrows
; row
++)
1328 if ( m_rowHeights
[ row
] != -1 )
1329 height
+= m_rowHeights
[ row
] + m_vgap
;
1333 m_calculatedMinSize
= wxSize( width
, height
);
1334 return m_calculatedMinSize
;
1337 void wxFlexGridSizer::AdjustForFlexDirection()
1339 // the logic in CalcMin works when we resize flexibly in both directions
1340 // but maybe this is not the case
1341 if ( m_flexDirection
!= wxBOTH
)
1343 // select the array corresponding to the direction in which we do *not*
1345 wxArrayInt
& array
= m_flexDirection
== wxVERTICAL
? m_colWidths
1348 const size_t count
= array
.GetCount();
1350 // find the largest value in this array
1354 for ( n
= 0; n
< count
; ++n
)
1356 if ( array
[n
] > largest
)
1360 // and now fill it with the largest value
1361 for ( n
= 0; n
< count
; ++n
)
1369 void wxFlexGridSizer::AdjustForGrowables(const wxSize
& sz
, const wxSize
& minsz
,
1370 int nrows
, int ncols
)
1372 // what to do with the rows? by default, resize them proportionally
1373 if ( sz
.y
> minsz
.y
&& ( (m_flexDirection
& wxVERTICAL
) || (m_growMode
== wxFLEX_GROWMODE_SPECIFIED
) ) )
1375 int sum_proportions
= 0;
1376 int growable_space
= 0;
1379 for (idx
= 0; idx
< m_growableRows
.GetCount(); idx
++)
1381 // Since the number of rows/columns can change as items are
1382 // inserted/deleted, we need to verify at runtime that the
1383 // requested growable rows/columns are still valid.
1384 if (m_growableRows
[idx
] >= nrows
)
1387 // If all items in a row/column are hidden, that row/column will
1388 // have a dimension of -1. This causes the row/column to be
1389 // hidden completely.
1390 if (m_rowHeights
[ m_growableRows
[idx
] ] == -1)
1392 sum_proportions
+= m_growableRowsProportions
[idx
];
1393 growable_space
+= m_rowHeights
[ m_growableRows
[idx
] ];
1399 for (idx
= 0; idx
< m_growableRows
.GetCount(); idx
++)
1401 if (m_growableRows
[idx
] >= nrows
)
1403 if (m_rowHeights
[ m_growableRows
[idx
] ] == -1)
1404 m_rowHeights
[ m_growableRows
[idx
] ] = 0;
1407 int delta
= (sz
.y
- minsz
.y
);
1408 if (sum_proportions
== 0)
1409 delta
= (delta
/num
) + m_rowHeights
[ m_growableRows
[idx
] ];
1411 delta
= ((delta
+growable_space
)*m_growableRowsProportions
[idx
]) / sum_proportions
;
1412 m_rowHeights
[ m_growableRows
[idx
] ] = delta
;
1417 else if ( (m_growMode
== wxFLEX_GROWMODE_ALL
) && (sz
.y
> minsz
.y
) )
1419 // rounding problem?
1420 for ( int row
= 0; row
< nrows
; ++row
)
1421 m_rowHeights
[ row
] = sz
.y
/ nrows
;
1424 // the same logic as above but for the columns
1425 if ( sz
.x
> minsz
.x
&& ( (m_flexDirection
& wxHORIZONTAL
) || (m_growMode
== wxFLEX_GROWMODE_SPECIFIED
) ) )
1427 int sum_proportions
= 0;
1428 int growable_space
= 0;
1431 for (idx
= 0; idx
< m_growableCols
.GetCount(); idx
++)
1433 // Since the number of rows/columns can change as items are
1434 // inserted/deleted, we need to verify at runtime that the
1435 // requested growable rows/columns are still valid.
1436 if (m_growableCols
[idx
] >= ncols
)
1439 // If all items in a row/column are hidden, that row/column will
1440 // have a dimension of -1. This causes the column to be hidden
1442 if (m_colWidths
[ m_growableCols
[idx
] ] == -1)
1444 sum_proportions
+= m_growableColsProportions
[idx
];
1445 growable_space
+= m_colWidths
[ m_growableCols
[idx
] ];
1451 for (idx
= 0; idx
< m_growableCols
.GetCount(); idx
++)
1453 if (m_growableCols
[idx
] >= ncols
)
1455 if (m_colWidths
[ m_growableCols
[idx
] ] == -1)
1456 m_colWidths
[ m_growableCols
[idx
] ] = 0;
1459 int delta
= (sz
.x
- minsz
.x
);
1460 if (sum_proportions
== 0)
1461 delta
= (delta
/num
) + m_colWidths
[ m_growableCols
[idx
] ];
1463 delta
= ((delta
+growable_space
)*m_growableColsProportions
[idx
])/sum_proportions
;
1464 m_colWidths
[ m_growableCols
[idx
] ] = delta
;
1469 else if ( (m_growMode
== wxFLEX_GROWMODE_ALL
) && (sz
.x
> minsz
.x
) )
1471 for ( int col
=0; col
< ncols
; ++col
)
1472 m_colWidths
[ col
] = sz
.x
/ ncols
;
1477 void wxFlexGridSizer::AddGrowableRow( size_t idx
, int proportion
)
1479 m_growableRows
.Add( idx
);
1480 m_growableRowsProportions
.Add( proportion
);
1483 void wxFlexGridSizer::RemoveGrowableRow( size_t idx
)
1485 m_growableRows
.Remove( idx
);
1488 void wxFlexGridSizer::AddGrowableCol( size_t idx
, int proportion
)
1490 m_growableCols
.Add( idx
);
1491 m_growableColsProportions
.Add( proportion
);
1494 void wxFlexGridSizer::RemoveGrowableCol( size_t idx
)
1496 m_growableCols
.Remove( idx
);
1499 //---------------------------------------------------------------------------
1501 //---------------------------------------------------------------------------
1503 wxBoxSizer::wxBoxSizer( int orient
)
1504 : m_orient( orient
)
1508 void wxBoxSizer::RecalcSizes()
1510 if (m_children
.GetCount() == 0)
1516 if (m_orient
== wxHORIZONTAL
)
1517 delta
= m_size
.x
- m_fixedWidth
;
1519 delta
= m_size
.y
- m_fixedHeight
;
1522 wxPoint
pt( m_position
);
1524 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1527 wxSizerItem
*item
= node
->GetData();
1529 if (item
->IsShown())
1531 wxSize
size( item
->GetMinSizeWithBorder() );
1533 if (m_orient
== wxVERTICAL
)
1535 wxCoord height
= size
.y
;
1536 if (item
->GetProportion())
1538 // Because of at least one visible item has non-zero
1539 // proportion then m_stretchable is not zero
1540 height
= (delta
* item
->GetProportion()) / m_stretchable
;
1543 wxPoint
child_pos( pt
);
1544 wxSize
child_size( size
.x
, height
);
1546 if (item
->GetFlag() & (wxEXPAND
| wxSHAPED
))
1547 child_size
.x
= m_size
.x
;
1548 else if (item
->GetFlag() & wxALIGN_RIGHT
)
1549 child_pos
.x
+= m_size
.x
- size
.x
;
1550 else if (item
->GetFlag() & (wxCENTER
| wxALIGN_CENTER_HORIZONTAL
))
1551 // XXX wxCENTER is added for backward compatibility;
1552 // wxALIGN_CENTER should be used in new code
1553 child_pos
.x
+= (m_size
.x
- size
.x
) / 2;
1555 item
->SetDimension( child_pos
, child_size
);
1561 wxCoord width
= size
.x
;
1562 if (item
->GetProportion())
1564 // Because of at least one visible item has non-zero
1565 // proportion then m_stretchable is not zero
1566 width
= (delta
* item
->GetProportion()) / m_stretchable
;
1569 wxPoint
child_pos( pt
);
1570 wxSize
child_size( width
, size
.y
);
1572 if (item
->GetFlag() & (wxEXPAND
| wxSHAPED
))
1573 child_size
.y
= m_size
.y
;
1574 else if (item
->GetFlag() & wxALIGN_BOTTOM
)
1575 child_pos
.y
+= m_size
.y
- size
.y
;
1576 else if (item
->GetFlag() & (wxCENTER
| wxALIGN_CENTER_VERTICAL
))
1577 // XXX wxCENTER is added for backward compatibility;
1578 // wxALIGN_CENTER should be used in new code
1579 child_pos
.y
+= (m_size
.y
- size
.y
) / 2;
1581 item
->SetDimension( child_pos
, child_size
);
1587 node
= node
->GetNext();
1591 wxSize
wxBoxSizer::CalcMin()
1593 if (m_children
.GetCount() == 0)
1594 return wxSize(10,10);
1602 // precalc item minsizes and count proportions
1603 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1606 wxSizerItem
*item
= node
->GetData();
1608 if ( item
->IsShown() )
1610 item
->CalcMin(); // result is stored in the item
1612 m_stretchable
+= item
->GetProportion();
1615 node
= node
->GetNext();
1618 // Total minimum size (width or height) of sizer
1621 node
= m_children
.GetFirst();
1624 wxSizerItem
*item
= node
->GetData();
1626 if (item
->IsShown() && item
->GetProportion() != 0)
1628 int stretch
= item
->GetProportion();
1629 wxSize
size( item
->GetMinSizeWithBorder() );
1632 // Integer division rounded up is (a + b - 1) / b
1633 // Round up needed in order to guarantee that all
1634 // all items will have size not less then their min size
1635 if (m_orient
== wxHORIZONTAL
)
1636 minSize
= ( size
.x
*m_stretchable
+ stretch
- 1)/stretch
;
1638 minSize
= ( size
.y
*m_stretchable
+ stretch
- 1)/stretch
;
1640 if (minSize
> maxMinSize
)
1641 maxMinSize
= minSize
;
1643 node
= node
->GetNext();
1646 // Calculate overall minimum size
1647 node
= m_children
.GetFirst();
1650 wxSizerItem
*item
= node
->GetData();
1652 if (item
->IsShown())
1654 wxSize
size( item
->GetMinSizeWithBorder() );
1655 if (item
->GetProportion() != 0)
1657 if (m_orient
== wxHORIZONTAL
)
1658 size
.x
= (maxMinSize
*item
->GetProportion())/m_stretchable
;
1660 size
.y
= (maxMinSize
*item
->GetProportion())/m_stretchable
;
1664 if (m_orient
== wxVERTICAL
)
1666 m_fixedHeight
+= size
.y
;
1667 m_fixedWidth
= wxMax( m_fixedWidth
, size
.x
);
1671 m_fixedWidth
+= size
.x
;
1672 m_fixedHeight
= wxMax( m_fixedHeight
, size
.y
);
1676 if (m_orient
== wxHORIZONTAL
)
1678 m_minWidth
+= size
.x
;
1679 m_minHeight
= wxMax( m_minHeight
, size
.y
);
1683 m_minHeight
+= size
.y
;
1684 m_minWidth
= wxMax( m_minWidth
, size
.x
);
1687 node
= node
->GetNext();
1690 return wxSize( m_minWidth
, m_minHeight
);
1693 //---------------------------------------------------------------------------
1695 //---------------------------------------------------------------------------
1699 wxStaticBoxSizer::wxStaticBoxSizer( wxStaticBox
*box
, int orient
)
1700 : wxBoxSizer( orient
),
1703 wxASSERT_MSG( box
, wxT("wxStaticBoxSizer needs a static box") );
1705 // do this so that our Detach() is called if the static box is destroyed
1707 m_staticBox
->SetContainingSizer(this);
1710 wxStaticBoxSizer::wxStaticBoxSizer(int orient
, wxWindow
*win
, const wxString
& s
)
1711 : wxBoxSizer(orient
),
1712 m_staticBox(new wxStaticBox(win
, wxID_ANY
, s
))
1715 m_staticBox
->SetContainingSizer(this);
1718 wxStaticBoxSizer::~wxStaticBoxSizer()
1723 static void GetStaticBoxBorders( wxStaticBox
*box
,
1727 // this has to be done platform by platform as there is no way to
1728 // guess the thickness of a wxStaticBox border
1729 box
->GetBordersForSizer(borderTop
, borderOther
);
1732 void wxStaticBoxSizer::RecalcSizes()
1734 int top_border
, other_border
;
1735 GetStaticBoxBorders(m_staticBox
, &top_border
, &other_border
);
1737 m_staticBox
->SetSize( m_position
.x
, m_position
.y
, m_size
.x
, m_size
.y
);
1739 wxPoint
old_pos( m_position
);
1740 m_position
.x
+= other_border
;
1741 m_position
.y
+= top_border
;
1742 wxSize
old_size( m_size
);
1743 m_size
.x
-= 2*other_border
;
1744 m_size
.y
-= top_border
+ other_border
;
1746 wxBoxSizer::RecalcSizes();
1748 m_position
= old_pos
;
1752 wxSize
wxStaticBoxSizer::CalcMin()
1754 int top_border
, other_border
;
1755 GetStaticBoxBorders(m_staticBox
, &top_border
, &other_border
);
1757 wxSize
ret( wxBoxSizer::CalcMin() );
1758 ret
.x
+= 2*other_border
;
1759 ret
.y
+= other_border
+ top_border
;
1764 void wxStaticBoxSizer::ShowItems( bool show
)
1766 m_staticBox
->Show( show
);
1767 wxBoxSizer::ShowItems( show
);
1770 bool wxStaticBoxSizer::Detach( wxWindow
*window
)
1772 // avoid deleting m_staticBox in our dtor if it's being detached from the
1773 // sizer (which can happen because it's being already destroyed for
1775 if ( window
== m_staticBox
)
1781 return wxSizer::Detach( window
);
1784 #endif // wxUSE_STATBOX
1788 wxStdDialogButtonSizer::wxStdDialogButtonSizer()
1789 : wxBoxSizer(wxHORIZONTAL
)
1791 // Vertical buttons with lots of space on either side
1792 // looks rubbish on WinCE, so let's not do this for now.
1793 // If we are going to use vertical buttons, we should
1794 // put the sizer to the right of other controls in the dialog,
1795 // and that's beyond the scope of this sizer.
1797 bool is_pda
= (wxSystemSettings::GetScreenType() <= wxSYS_SCREEN_PDA
);
1798 // If we have a PDA screen, put yes/no button over
1799 // all other buttons, otherwise on the left side.
1801 m_orient
= wxVERTICAL
;
1804 m_buttonAffirmative
= NULL
;
1805 m_buttonApply
= NULL
;
1806 m_buttonNegative
= NULL
;
1807 m_buttonCancel
= NULL
;
1808 m_buttonHelp
= NULL
;
1811 void wxStdDialogButtonSizer::AddButton(wxButton
*mybutton
)
1813 switch (mybutton
->GetId())
1818 m_buttonAffirmative
= mybutton
;
1821 m_buttonApply
= mybutton
;
1824 m_buttonNegative
= mybutton
;
1827 m_buttonCancel
= mybutton
;
1830 case wxID_CONTEXT_HELP
:
1831 m_buttonHelp
= mybutton
;
1838 void wxStdDialogButtonSizer::SetAffirmativeButton( wxButton
*button
)
1840 m_buttonAffirmative
= button
;
1843 void wxStdDialogButtonSizer::SetNegativeButton( wxButton
*button
)
1845 m_buttonNegative
= button
;
1848 void wxStdDialogButtonSizer::SetCancelButton( wxButton
*button
)
1850 m_buttonCancel
= button
;
1853 void wxStdDialogButtonSizer::Realize()
1856 Add(0, 0, 0, wxLEFT
, 6);
1858 Add((wxWindow
*)m_buttonHelp
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 6);
1860 if (m_buttonNegative
){
1861 // HIG POLICE BULLETIN - destructive buttons need extra padding
1862 // 24 pixels on either side
1863 Add((wxWindow
*)m_buttonNegative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 12);
1866 // extra whitespace between help/negative and cancel/ok buttons
1867 Add(0, 0, 1, wxEXPAND
, 0);
1869 if (m_buttonCancel
){
1870 Add((wxWindow
*)m_buttonCancel
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 6);
1871 // Cancel or help should be default
1872 // m_buttonCancel->SetDefaultButton();
1875 // Ugh, Mac doesn't really have apply dialogs, so I'll just
1876 // figure the best place is between Cancel and OK
1878 Add((wxWindow
*)m_buttonApply
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 6);
1880 if (m_buttonAffirmative
){
1881 Add((wxWindow
*)m_buttonAffirmative
, 0, wxALIGN_CENTRE
| wxLEFT
, 6);
1883 if (m_buttonAffirmative
->GetId() == wxID_SAVE
){
1884 // these buttons have set labels under Mac so we should use them
1885 m_buttonAffirmative
->SetLabel(_("Save"));
1886 m_buttonNegative
->SetLabel(_("Don't Save"));
1890 // Extra space around and at the right
1892 #elif defined(__WXGTK20__)
1893 Add(0, 0, 0, wxLEFT
, 9);
1895 Add((wxWindow
*)m_buttonHelp
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 3);
1897 // extra whitespace between help and cancel/ok buttons
1898 Add(0, 0, 1, wxEXPAND
, 0);
1900 if (m_buttonNegative
){
1901 Add((wxWindow
*)m_buttonNegative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 3);
1904 if (m_buttonCancel
){
1905 Add((wxWindow
*)m_buttonCancel
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 3);
1906 // Cancel or help should be default
1907 // m_buttonCancel->SetDefaultButton();
1911 Add((wxWindow
*)m_buttonApply
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 3);
1913 if (m_buttonAffirmative
)
1914 Add((wxWindow
*)m_buttonAffirmative
, 0, wxALIGN_CENTRE
| wxLEFT
, 6);
1915 #elif defined(__WXMSW__)
1918 // right-justify buttons
1919 Add(0, 0, 1, wxEXPAND
, 0);
1921 if (m_buttonAffirmative
){
1922 Add((wxWindow
*)m_buttonAffirmative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonAffirmative
->ConvertDialogToPixels(wxSize(2, 0)).x
);
1925 if (m_buttonNegative
){
1926 Add((wxWindow
*)m_buttonNegative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonNegative
->ConvertDialogToPixels(wxSize(2, 0)).x
);
1929 if (m_buttonCancel
){
1930 Add((wxWindow
*)m_buttonCancel
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonCancel
->ConvertDialogToPixels(wxSize(2, 0)).x
);
1933 Add((wxWindow
*)m_buttonApply
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonApply
->ConvertDialogToPixels(wxSize(2, 0)).x
);
1936 Add((wxWindow
*)m_buttonHelp
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonHelp
->ConvertDialogToPixels(wxSize(2, 0)).x
);
1938 // GTK+1 and any other platform
1940 // Add(0, 0, 0, wxLEFT, 5); // Not sure what this was for but it unbalances the dialog
1942 Add((wxWindow
*)m_buttonHelp
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonHelp
->ConvertDialogToPixels(wxSize(4, 0)).x
);
1944 // extra whitespace between help and cancel/ok buttons
1945 Add(0, 0, 1, wxEXPAND
, 0);
1948 Add((wxWindow
*)m_buttonApply
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonApply
->ConvertDialogToPixels(wxSize(4, 0)).x
);
1950 if (m_buttonAffirmative
){
1951 Add((wxWindow
*)m_buttonAffirmative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonAffirmative
->ConvertDialogToPixels(wxSize(4, 0)).x
);
1954 if (m_buttonNegative
){
1955 Add((wxWindow
*)m_buttonNegative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonNegative
->ConvertDialogToPixels(wxSize(4, 0)).x
);
1958 if (m_buttonCancel
){
1959 Add((wxWindow
*)m_buttonCancel
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonCancel
->ConvertDialogToPixels(wxSize(4, 0)).x
);
1960 // Cancel or help should be default
1961 // m_buttonCancel->SetDefaultButton();
1967 #endif // wxUSE_BUTTON
1969 #if WXWIN_COMPATIBILITY_2_4
1971 // ----------------------------------------------------------------------------
1973 // ----------------------------------------------------------------------------
1976 IMPLEMENT_CLASS(wxBookCtrlSizer
, wxSizer
)
1978 IMPLEMENT_CLASS(wxNotebookSizer
, wxBookCtrlSizer
)
1979 #endif // wxUSE_NOTEBOOK
1980 #endif // wxUSE_BOOKCTRL
1984 #if WXWIN_COMPATIBILITY_2_6
1986 wxBookCtrlSizer::wxBookCtrlSizer(wxBookCtrlBase
*bookctrl
)
1987 : m_bookctrl(bookctrl
)
1989 wxASSERT_MSG( bookctrl
, wxT("wxBookCtrlSizer needs a control") );
1992 #endif // WXWIN_COMPATIBILITY_2_6
1994 void wxBookCtrlSizer::RecalcSizes()
1996 m_bookctrl
->SetSize( m_position
.x
, m_position
.y
, m_size
.x
, m_size
.y
);
1999 wxSize
wxBookCtrlSizer::CalcMin()
2001 wxSize sizeBorder
= m_bookctrl
->CalcSizeFromPage(wxSize(0,0));
2006 if ( m_bookctrl
->GetPageCount() == 0 )
2008 return wxSize(sizeBorder
.x
+ 10, sizeBorder
.y
+ 10);
2014 wxWindowList::compatibility_iterator
2015 node
= m_bookctrl
->GetChildren().GetFirst();
2018 wxWindow
*item
= node
->GetData();
2019 wxSizer
*itemsizer
= item
->GetSizer();
2023 wxSize
subsize( itemsizer
->CalcMin() );
2025 if (subsize
.x
> maxX
)
2027 if (subsize
.y
> maxY
)
2031 node
= node
->GetNext();
2034 return wxSize( maxX
, maxY
) + sizeBorder
;
2039 #if WXWIN_COMPATIBILITY_2_6
2041 wxNotebookSizer::wxNotebookSizer(wxNotebook
*nb
)
2043 wxASSERT_MSG( nb
, wxT("wxNotebookSizer needs a control") );
2047 #endif // WXWIN_COMPATIBILITY_2_6
2049 #endif // wxUSE_NOTEBOOOK
2050 #endif // wxUSE_BOOKCTRL
2052 #endif // WXWIN_COMPATIBILITY_2_4