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"
20 #include "wx/display.h"
24 #include "wx/string.h"
28 #include "wx/settings.h"
29 #include "wx/button.h"
30 #include "wx/statbox.h"
31 #include "wx/toplevel.h"
34 #include "wx/listimpl.cpp"
37 //---------------------------------------------------------------------------
39 IMPLEMENT_CLASS(wxSizerItem
, wxObject
)
40 IMPLEMENT_CLASS(wxSizer
, wxObject
)
41 IMPLEMENT_CLASS(wxGridSizer
, wxSizer
)
42 IMPLEMENT_CLASS(wxFlexGridSizer
, wxGridSizer
)
43 IMPLEMENT_CLASS(wxBoxSizer
, wxSizer
)
45 IMPLEMENT_CLASS(wxStaticBoxSizer
, wxBoxSizer
)
48 IMPLEMENT_CLASS(wxStdDialogButtonSizer
, wxBoxSizer
)
51 WX_DEFINE_EXPORTED_LIST( wxSizerItemList
)
86 // ----------------------------------------------------------------------------
88 // ----------------------------------------------------------------------------
90 void wxSizerItem::Init(const wxSizerFlags
& flags
)
94 m_proportion
= flags
.GetProportion();
95 m_flag
= flags
.GetFlags();
96 m_border
= flags
.GetBorderInPixels();
99 wxSizerItem::wxSizerItem()
109 void wxSizerItem::DoSetWindow(wxWindow
*window
)
111 wxCHECK_RET( window
, _T("NULL window in wxSizerItem::SetWindow()") );
113 m_kind
= Item_Window
;
116 // window doesn't become smaller than its initial size, whatever happens
117 m_minSize
= window
->GetSize();
119 if ( m_flag
& wxFIXED_MINSIZE
)
120 window
->SetMinSize(m_minSize
);
122 // aspect ratio calculated from initial size
126 wxSizerItem::wxSizerItem(wxWindow
*window
,
132 m_proportion(proportion
),
141 void wxSizerItem::DoSetSizer(wxSizer
*sizer
)
147 wxSizerItem::wxSizerItem(wxSizer
*sizer
,
154 m_proportion(proportion
),
162 // m_minSize is set later
166 void wxSizerItem::DoSetSpacer(const wxSize
& size
)
168 m_kind
= Item_Spacer
;
169 m_spacer
= new wxSizerSpacer(size
);
174 wxSizerItem::wxSizerItem(int width
,
182 m_minSize(width
, height
), // minimal size is the initial size
183 m_proportion(proportion
),
188 DoSetSpacer(wxSize(width
, height
));
191 wxSizerItem::~wxSizerItem()
197 void wxSizerItem::Free()
205 m_window
->SetContainingSizer(NULL
);
218 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
224 wxSize
wxSizerItem::GetSpacer() const
227 if ( m_kind
== Item_Spacer
)
228 size
= m_spacer
->GetSize();
234 wxSize
wxSizerItem::GetSize() const
243 ret
= m_window
->GetSize();
247 ret
= m_sizer
->GetSize();
251 ret
= m_spacer
->GetSize();
256 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
263 if (m_flag
& wxNORTH
)
265 if (m_flag
& wxSOUTH
)
271 wxSize
wxSizerItem::CalcMin()
275 m_minSize
= m_sizer
->GetMinSize();
277 // if we have to preserve aspect ratio _AND_ this is
278 // the first-time calculation, consider ret to be initial size
279 if ( (m_flag
& wxSHAPED
) && wxIsNullDouble(m_ratio
) )
282 else if ( IsWindow() )
284 // Since the size of the window may change during runtime, we
285 // should use the current minimal/best size.
286 m_minSize
= m_window
->GetEffectiveMinSize();
289 return GetMinSizeWithBorder();
292 wxSize
wxSizerItem::GetMinSizeWithBorder() const
294 wxSize ret
= m_minSize
;
300 if (m_flag
& wxNORTH
)
302 if (m_flag
& wxSOUTH
)
309 void wxSizerItem::SetDimension( const wxPoint
& pos_
, const wxSize
& size_
)
313 if (m_flag
& wxSHAPED
)
315 // adjust aspect ratio
316 int rwidth
= (int) (size
.y
* m_ratio
);
320 int rheight
= (int) (size
.x
/ m_ratio
);
321 // add vertical space
322 if (m_flag
& wxALIGN_CENTER_VERTICAL
)
323 pos
.y
+= (size
.y
- rheight
) / 2;
324 else if (m_flag
& wxALIGN_BOTTOM
)
325 pos
.y
+= (size
.y
- rheight
);
326 // use reduced dimensions
329 else if (rwidth
< size
.x
)
331 // add horizontal space
332 if (m_flag
& wxALIGN_CENTER_HORIZONTAL
)
333 pos
.x
+= (size
.x
- rwidth
) / 2;
334 else if (m_flag
& wxALIGN_RIGHT
)
335 pos
.x
+= (size
.x
- rwidth
);
340 // This is what GetPosition() returns. Since we calculate
341 // borders afterwards, GetPosition() will be the left/top
342 // corner of the surrounding border.
354 if (m_flag
& wxNORTH
)
359 if (m_flag
& wxSOUTH
)
369 m_rect
= wxRect(pos
, size
);
374 wxFAIL_MSG( _T("can't set size of uninitialized sizer item") );
378 m_window
->SetSize(pos
.x
, pos
.y
, size
.x
, size
.y
,
379 wxSIZE_ALLOW_MINUS_ONE
);
383 m_sizer
->SetDimension(pos
.x
, pos
.y
, size
.x
, size
.y
);
387 m_spacer
->SetSize(size
);
392 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
396 void wxSizerItem::DeleteWindows()
405 //We are deleting the window from this sizer - normally
406 //the window destroys the sizer associated with it,
407 //which might destroy this, which we don't want
408 m_window
->SetContainingSizer(NULL
);
410 //Putting this after the switch will result in a spacer
411 //not being deleted properly on destruction
416 m_sizer
->DeleteWindows();
421 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
426 void wxSizerItem::Show( bool show
)
431 wxFAIL_MSG( _T("can't show uninitialized sizer item") );
435 m_window
->Show(show
);
443 m_spacer
->Show(show
);
448 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
452 bool wxSizerItem::IsShown() const
457 // we may be called from CalcMin(), just return false so that we're
462 return m_window
->IsShown();
465 // arbitrarily decide that if at least one of our elements is
466 // shown, so are we (this arbitrariness is the reason for
467 // deprecating this function)
469 // Some apps (such as dialog editors) depend on an empty sizer still
470 // being laid out correctly and reporting the correct size and position.
471 if (m_sizer
->GetChildren().GetCount() == 0)
474 for ( wxSizerItemList::compatibility_iterator
475 node
= m_sizer
->GetChildren().GetFirst();
477 node
= node
->GetNext() )
479 if ( node
->GetData()->IsShown() )
486 return m_spacer
->IsShown();
490 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
496 #if WXWIN_COMPATIBILITY_2_6
497 void wxSizerItem::SetOption( int option
)
499 SetProportion( option
);
502 int wxSizerItem::GetOption() const
504 return GetProportion();
506 #endif // WXWIN_COMPATIBILITY_2_6
509 //---------------------------------------------------------------------------
511 //---------------------------------------------------------------------------
515 WX_CLEAR_LIST(wxSizerItemList
, m_children
);
518 wxSizerItem
* wxSizer::Insert( size_t index
, wxSizerItem
*item
)
520 m_children
.Insert( index
, item
);
522 if ( item
->GetWindow() )
523 item
->GetWindow()->SetContainingSizer( this );
528 void wxSizer::SetContainingWindow(wxWindow
*win
)
530 if ( win
== m_containingWindow
)
533 m_containingWindow
= win
;
535 // set the same window for all nested sizers as well, they also are in the
537 for ( wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
539 node
= node
->GetNext() )
541 wxSizerItem
*const item
= node
->GetData();
542 wxSizer
*const sizer
= item
->GetSizer();
546 sizer
->SetContainingWindow(win
);
551 #if WXWIN_COMPATIBILITY_2_6
552 bool wxSizer::Remove( wxWindow
*window
)
554 return Detach( window
);
556 #endif // WXWIN_COMPATIBILITY_2_6
558 bool wxSizer::Remove( wxSizer
*sizer
)
560 wxASSERT_MSG( sizer
, _T("Removing NULL sizer") );
562 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
565 wxSizerItem
*item
= node
->GetData();
567 if (item
->GetSizer() == sizer
)
570 m_children
.Erase( node
);
574 node
= node
->GetNext();
580 bool wxSizer::Remove( int index
)
582 wxCHECK_MSG( index
>= 0 && (size_t)index
< m_children
.GetCount(),
584 _T("Remove index is out of range") );
586 wxSizerItemList::compatibility_iterator node
= m_children
.Item( index
);
588 wxCHECK_MSG( node
, false, _T("Failed to find child node") );
590 delete node
->GetData();
591 m_children
.Erase( node
);
596 bool wxSizer::Detach( wxSizer
*sizer
)
598 wxASSERT_MSG( sizer
, _T("Detaching NULL sizer") );
600 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
603 wxSizerItem
*item
= node
->GetData();
605 if (item
->GetSizer() == sizer
)
609 m_children
.Erase( node
);
612 node
= node
->GetNext();
618 bool wxSizer::Detach( wxWindow
*window
)
620 wxASSERT_MSG( window
, _T("Detaching NULL window") );
622 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
625 wxSizerItem
*item
= node
->GetData();
627 if (item
->GetWindow() == window
)
630 m_children
.Erase( node
);
633 node
= node
->GetNext();
639 bool wxSizer::Detach( int index
)
641 wxCHECK_MSG( index
>= 0 && (size_t)index
< m_children
.GetCount(),
643 _T("Detach index is out of range") );
645 wxSizerItemList::compatibility_iterator node
= m_children
.Item( index
);
647 wxCHECK_MSG( node
, false, _T("Failed to find child node") );
649 wxSizerItem
*item
= node
->GetData();
651 if ( item
->IsSizer() )
655 m_children
.Erase( node
);
659 bool wxSizer::Replace( wxWindow
*oldwin
, wxWindow
*newwin
, bool recursive
)
661 wxASSERT_MSG( oldwin
, _T("Replacing NULL window") );
662 wxASSERT_MSG( newwin
, _T("Replacing with NULL window") );
664 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
667 wxSizerItem
*item
= node
->GetData();
669 if (item
->GetWindow() == oldwin
)
671 item
->AssignWindow(newwin
);
672 newwin
->SetContainingSizer( this );
675 else if (recursive
&& item
->IsSizer())
677 if (item
->GetSizer()->Replace( oldwin
, newwin
, true ))
681 node
= node
->GetNext();
687 bool wxSizer::Replace( wxSizer
*oldsz
, wxSizer
*newsz
, bool recursive
)
689 wxASSERT_MSG( oldsz
, _T("Replacing NULL sizer") );
690 wxASSERT_MSG( newsz
, _T("Replacing with NULL sizer") );
692 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
695 wxSizerItem
*item
= node
->GetData();
697 if (item
->GetSizer() == oldsz
)
699 item
->AssignSizer(newsz
);
702 else if (recursive
&& item
->IsSizer())
704 if (item
->GetSizer()->Replace( oldsz
, newsz
, true ))
708 node
= node
->GetNext();
714 bool wxSizer::Replace( size_t old
, wxSizerItem
*newitem
)
716 wxCHECK_MSG( old
< m_children
.GetCount(), false, _T("Replace index is out of range") );
717 wxASSERT_MSG( newitem
, _T("Replacing with NULL item") );
719 wxSizerItemList::compatibility_iterator node
= m_children
.Item( old
);
721 wxCHECK_MSG( node
, false, _T("Failed to find child node") );
723 wxSizerItem
*item
= node
->GetData();
724 node
->SetData(newitem
);
730 void wxSizer::Clear( bool delete_windows
)
732 // First clear the ContainingSizer pointers
733 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
736 wxSizerItem
*item
= node
->GetData();
738 if (item
->IsWindow())
739 item
->GetWindow()->SetContainingSizer( NULL
);
740 node
= node
->GetNext();
743 // Destroy the windows if needed
747 // Now empty the list
748 WX_CLEAR_LIST(wxSizerItemList
, m_children
);
751 void wxSizer::DeleteWindows()
753 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
756 wxSizerItem
*item
= node
->GetData();
758 item
->DeleteWindows();
759 node
= node
->GetNext();
763 wxSize
wxSizer::Fit( wxWindow
*window
)
765 // take the min size by default and limit it by max size
766 wxSize size
= GetMinWindowSize(window
);
767 wxSize sizeMax
= GetMaxWindowSize(window
);
769 wxTopLevelWindow
*tlw
= wxDynamicCast(window
, wxTopLevelWindow
);
772 // hack for small screen devices where TLWs are always full screen
773 if ( tlw
->IsAlwaysMaximized() )
775 size
= tlw
->GetSize();
777 else // normal situation
779 // limit the window to the size of the display it is on
780 int disp
= wxDisplay::GetFromWindow(window
);
781 if ( disp
== wxNOT_FOUND
)
783 // or, if we don't know which one it is, of the main one
787 sizeMax
= wxDisplay(disp
).GetClientArea().GetSize();
791 if ( sizeMax
.x
!= wxDefaultCoord
&& size
.x
> sizeMax
.x
)
793 if ( sizeMax
.y
!= wxDefaultCoord
&& size
.y
> sizeMax
.y
)
797 window
->SetSize( size
);
802 void wxSizer::FitInside( wxWindow
*window
)
805 if (window
->IsTopLevel())
806 size
= VirtualFitSize( window
);
808 size
= GetMinClientSize( window
);
810 window
->SetVirtualSize( size
);
813 void wxSizer::Layout()
815 // (re)calculates minimums needed for each item and other preparations
819 // Applies the layout and repositions/resizes the items
823 void wxSizer::SetSizeHints( wxWindow
*window
)
825 // Preserve the window's max size hints, but set the
826 // lower bound according to the sizer calculations.
828 wxSize size
= Fit( window
);
830 window
->SetSizeHints( size
.x
,
832 window
->GetMaxWidth(),
833 window
->GetMaxHeight() );
836 void wxSizer::SetVirtualSizeHints( wxWindow
*window
)
838 // Preserve the window's max size hints, but set the
839 // lower bound according to the sizer calculations.
842 wxSize
size( window
->GetVirtualSize() );
843 window
->SetVirtualSizeHints( size
.x
,
845 window
->GetMaxWidth(),
846 window
->GetMaxHeight() );
849 wxSize
wxSizer::GetMaxWindowSize( wxWindow
*window
) const
851 return window
->GetMaxSize();
854 wxSize
wxSizer::GetMinWindowSize( wxWindow
*window
)
856 wxSize
minSize( GetMinSize() );
857 wxSize
size( window
->GetSize() );
858 wxSize
client_size( window
->GetClientSize() );
860 return wxSize( minSize
.x
+size
.x
-client_size
.x
,
861 minSize
.y
+size
.y
-client_size
.y
);
864 // TODO on mac we need a function that determines how much free space this
865 // min size contains, in order to make sure that we have 20 pixels of free
866 // space around the controls
867 wxSize
wxSizer::GetMaxClientSize( wxWindow
*window
) const
869 wxSize
maxSize( window
->GetMaxSize() );
871 if ( maxSize
!= wxDefaultSize
)
873 wxSize
size( window
->GetSize() );
874 wxSize
client_size( window
->GetClientSize() );
876 return wxSize( maxSize
.x
+ client_size
.x
- size
.x
,
877 maxSize
.y
+ client_size
.y
- size
.y
);
880 return wxDefaultSize
;
883 wxSize
wxSizer::GetMinClientSize( wxWindow
*WXUNUSED(window
) )
885 return GetMinSize(); // Already returns client size.
888 wxSize
wxSizer::VirtualFitSize( wxWindow
*window
)
890 wxSize size
= GetMinClientSize( window
);
891 wxSize sizeMax
= GetMaxClientSize( window
);
893 // Limit the size if sizeMax != wxDefaultSize
895 if ( size
.x
> sizeMax
.x
&& sizeMax
.x
!= wxDefaultCoord
)
897 if ( size
.y
> sizeMax
.y
&& sizeMax
.y
!= wxDefaultCoord
)
903 void wxSizer::SetDimension( int x
, int y
, int width
, int height
)
912 wxSize
wxSizer::GetMinSize()
914 wxSize
ret( CalcMin() );
915 if (ret
.x
< m_minSize
.x
) ret
.x
= m_minSize
.x
;
916 if (ret
.y
< m_minSize
.y
) ret
.y
= m_minSize
.y
;
920 void wxSizer::DoSetMinSize( int width
, int height
)
923 m_minSize
.y
= height
;
926 bool wxSizer::DoSetItemMinSize( wxWindow
*window
, int width
, int height
)
928 wxASSERT_MSG( window
, _T("SetMinSize for NULL window") );
930 // Is it our immediate child?
932 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
935 wxSizerItem
*item
= node
->GetData();
937 if (item
->GetWindow() == window
)
939 item
->SetMinSize( width
, height
);
942 node
= node
->GetNext();
945 // No? Search any subsizers we own then
947 node
= m_children
.GetFirst();
950 wxSizerItem
*item
= node
->GetData();
952 if ( item
->GetSizer() &&
953 item
->GetSizer()->DoSetItemMinSize( window
, width
, height
) )
955 // A child sizer found the requested windw, exit.
958 node
= node
->GetNext();
964 bool wxSizer::DoSetItemMinSize( wxSizer
*sizer
, int width
, int height
)
966 wxASSERT_MSG( sizer
, _T("SetMinSize for NULL sizer") );
968 // Is it our immediate child?
970 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
973 wxSizerItem
*item
= node
->GetData();
975 if (item
->GetSizer() == sizer
)
977 item
->GetSizer()->DoSetMinSize( width
, height
);
980 node
= node
->GetNext();
983 // No? Search any subsizers we own then
985 node
= m_children
.GetFirst();
988 wxSizerItem
*item
= node
->GetData();
990 if ( item
->GetSizer() &&
991 item
->GetSizer()->DoSetItemMinSize( sizer
, width
, height
) )
993 // A child found the requested sizer, exit.
996 node
= node
->GetNext();
1002 bool wxSizer::DoSetItemMinSize( size_t index
, int width
, int height
)
1004 wxSizerItemList::compatibility_iterator node
= m_children
.Item( index
);
1006 wxCHECK_MSG( node
, false, _T("Failed to find child node") );
1008 wxSizerItem
*item
= node
->GetData();
1010 if (item
->GetSizer())
1012 // Sizers contains the minimal size in them, if not calculated ...
1013 item
->GetSizer()->DoSetMinSize( width
, height
);
1017 // ... but the minimal size of spacers and windows is stored via the item
1018 item
->SetMinSize( width
, height
);
1024 wxSizerItem
* wxSizer::GetItem( wxWindow
*window
, bool recursive
)
1026 wxASSERT_MSG( window
, _T("GetItem for NULL window") );
1028 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1031 wxSizerItem
*item
= node
->GetData();
1033 if (item
->GetWindow() == window
)
1037 else if (recursive
&& item
->IsSizer())
1039 wxSizerItem
*subitem
= item
->GetSizer()->GetItem( window
, true );
1044 node
= node
->GetNext();
1050 wxSizerItem
* wxSizer::GetItem( wxSizer
*sizer
, bool recursive
)
1052 wxASSERT_MSG( sizer
, _T("GetItem for NULL sizer") );
1054 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1057 wxSizerItem
*item
= node
->GetData();
1059 if (item
->GetSizer() == sizer
)
1063 else if (recursive
&& item
->IsSizer())
1065 wxSizerItem
*subitem
= item
->GetSizer()->GetItem( sizer
, true );
1070 node
= node
->GetNext();
1076 wxSizerItem
* wxSizer::GetItem( size_t index
)
1078 wxCHECK_MSG( index
< m_children
.GetCount(),
1080 _T("GetItem index is out of range") );
1082 return m_children
.Item( index
)->GetData();
1085 bool wxSizer::Show( wxWindow
*window
, bool show
, bool recursive
)
1087 wxSizerItem
*item
= GetItem( window
, recursive
);
1098 bool wxSizer::Show( wxSizer
*sizer
, bool show
, bool recursive
)
1100 wxSizerItem
*item
= GetItem( sizer
, recursive
);
1111 bool wxSizer::Show( size_t index
, bool show
)
1113 wxSizerItem
*item
= GetItem( index
);
1124 void wxSizer::ShowItems( bool show
)
1126 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1129 node
->GetData()->Show( show
);
1130 node
= node
->GetNext();
1134 bool wxSizer::IsShown( wxWindow
*window
) const
1136 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1139 wxSizerItem
*item
= node
->GetData();
1141 if (item
->GetWindow() == window
)
1143 return item
->IsShown();
1145 node
= node
->GetNext();
1148 wxFAIL_MSG( _T("IsShown failed to find sizer item") );
1153 bool wxSizer::IsShown( wxSizer
*sizer
) const
1155 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1158 wxSizerItem
*item
= node
->GetData();
1160 if (item
->GetSizer() == sizer
)
1162 return item
->IsShown();
1164 node
= node
->GetNext();
1167 wxFAIL_MSG( _T("IsShown failed to find sizer item") );
1172 bool wxSizer::IsShown( size_t index
) const
1174 wxCHECK_MSG( index
< m_children
.GetCount(),
1176 _T("IsShown index is out of range") );
1178 return m_children
.Item( index
)->GetData()->IsShown();
1182 //---------------------------------------------------------------------------
1184 //---------------------------------------------------------------------------
1186 wxGridSizer::wxGridSizer( int rows
, int cols
, int vgap
, int hgap
)
1187 : m_rows( ( cols
== 0 && rows
== 0 ) ? 1 : rows
)
1194 wxGridSizer::wxGridSizer( int cols
, int vgap
, int hgap
)
1195 : m_rows( cols
== 0 ? 1 : 0 )
1202 int wxGridSizer::CalcRowsCols(int& nrows
, int& ncols
) const
1204 int nitems
= m_children
.GetCount();
1210 nrows
= (nitems
+ m_cols
- 1) / m_cols
;
1214 ncols
= (nitems
+ m_rows
- 1) / m_rows
;
1217 else // 0 columns, 0 rows?
1219 wxFAIL_MSG( _T("grid sizer must have either rows or columns fixed") );
1228 void wxGridSizer::RecalcSizes()
1230 int nitems
, nrows
, ncols
;
1231 if ( (nitems
= CalcRowsCols(nrows
, ncols
)) == 0 )
1234 wxSize
sz( GetSize() );
1235 wxPoint
pt( GetPosition() );
1237 int w
= (sz
.x
- (ncols
- 1) * m_hgap
) / ncols
;
1238 int h
= (sz
.y
- (nrows
- 1) * m_vgap
) / nrows
;
1241 for (int c
= 0; c
< ncols
; c
++)
1244 for (int r
= 0; r
< nrows
; r
++)
1246 int i
= r
* ncols
+ c
;
1249 wxSizerItemList::compatibility_iterator node
= m_children
.Item( i
);
1251 wxASSERT_MSG( node
, _T("Failed to find SizerItemList node") );
1253 SetItemBounds( node
->GetData(), x
, y
, w
, h
);
1261 wxSize
wxGridSizer::CalcMin()
1264 if ( CalcRowsCols(nrows
, ncols
) == 0 )
1267 // Find the max width and height for any component
1271 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1274 wxSizerItem
*item
= node
->GetData();
1275 wxSize
sz( item
->CalcMin() );
1277 w
= wxMax( w
, sz
.x
);
1278 h
= wxMax( h
, sz
.y
);
1280 node
= node
->GetNext();
1283 return wxSize( ncols
* w
+ (ncols
-1) * m_hgap
,
1284 nrows
* h
+ (nrows
-1) * m_vgap
);
1287 void wxGridSizer::SetItemBounds( wxSizerItem
*item
, int x
, int y
, int w
, int h
)
1290 wxSize
sz( item
->GetMinSizeWithBorder() );
1291 int flag
= item
->GetFlag();
1293 if ((flag
& wxEXPAND
) || (flag
& wxSHAPED
))
1299 if (flag
& wxALIGN_CENTER_HORIZONTAL
)
1301 pt
.x
= x
+ (w
- sz
.x
) / 2;
1303 else if (flag
& wxALIGN_RIGHT
)
1305 pt
.x
= x
+ (w
- sz
.x
);
1308 if (flag
& wxALIGN_CENTER_VERTICAL
)
1310 pt
.y
= y
+ (h
- sz
.y
) / 2;
1312 else if (flag
& wxALIGN_BOTTOM
)
1314 pt
.y
= y
+ (h
- sz
.y
);
1318 item
->SetDimension(pt
, sz
);
1321 //---------------------------------------------------------------------------
1323 //---------------------------------------------------------------------------
1325 wxFlexGridSizer::wxFlexGridSizer( int rows
, int cols
, int vgap
, int hgap
)
1326 : wxGridSizer( rows
, cols
, vgap
, hgap
),
1327 m_flexDirection(wxBOTH
),
1328 m_growMode(wxFLEX_GROWMODE_SPECIFIED
)
1332 wxFlexGridSizer::wxFlexGridSizer( int cols
, int vgap
, int hgap
)
1333 : wxGridSizer( cols
, vgap
, hgap
),
1334 m_flexDirection(wxBOTH
),
1335 m_growMode(wxFLEX_GROWMODE_SPECIFIED
)
1339 wxFlexGridSizer::~wxFlexGridSizer()
1343 void wxFlexGridSizer::RecalcSizes()
1345 int nitems
, nrows
, ncols
;
1346 if ( (nitems
= CalcRowsCols(nrows
, ncols
)) == 0 )
1349 wxPoint
pt( GetPosition() );
1350 wxSize
sz( GetSize() );
1352 AdjustForGrowables(sz
, m_calculatedMinSize
, nrows
, ncols
);
1354 sz
= wxSize( pt
.x
+ sz
.x
, pt
.y
+ sz
.y
);
1357 for (int c
= 0; c
< ncols
; c
++)
1360 for (int r
= 0; r
< nrows
; r
++)
1362 int i
= r
* ncols
+ c
;
1365 wxSizerItemList::compatibility_iterator node
= m_children
.Item( i
);
1367 wxASSERT_MSG( node
, _T("Failed to find node") );
1369 int w
= wxMax( 0, wxMin( m_colWidths
[c
], sz
.x
- x
) );
1370 int h
= wxMax( 0, wxMin( m_rowHeights
[r
], sz
.y
- y
) );
1372 SetItemBounds( node
->GetData(), x
, y
, w
, h
);
1374 if (m_rowHeights
[r
] != -1)
1375 y
= y
+ m_rowHeights
[r
] + m_vgap
;
1377 if (m_colWidths
[c
] != -1)
1378 x
= x
+ m_colWidths
[c
] + m_hgap
;
1382 wxSize
wxFlexGridSizer::CalcMin()
1388 // Number of rows/columns can change as items are added or removed.
1389 if ( !CalcRowsCols(nrows
, ncols
) )
1392 m_rowHeights
.SetCount(nrows
);
1393 m_colWidths
.SetCount(ncols
);
1395 // We have to recalcuate the sizes in case the item minimum size has
1396 // changed since the previous layout, or the item has been hidden using
1397 // wxSizer::Show(). If all the items in a row/column are hidden, the final
1398 // dimension of the row/column will be -1, indicating that the column
1399 // itself is hidden.
1400 for( s
= m_rowHeights
.GetCount(), i
= 0; i
< s
; ++i
)
1401 m_rowHeights
[ i
] = -1;
1402 for( s
= m_colWidths
.GetCount(), i
= 0; i
< s
; ++i
)
1403 m_colWidths
[ i
] = -1;
1405 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1410 wxSizerItem
*item
= node
->GetData();
1411 if ( item
->IsShown() )
1413 wxSize
sz( item
->CalcMin() );
1414 int row
= i
/ ncols
;
1415 int col
= i
% ncols
;
1417 m_rowHeights
[ row
] = wxMax( wxMax( 0, sz
.y
), m_rowHeights
[ row
] );
1418 m_colWidths
[ col
] = wxMax( wxMax( 0, sz
.x
), m_colWidths
[ col
] );
1421 node
= node
->GetNext();
1425 AdjustForFlexDirection();
1427 // Sum total minimum size, including gaps between rows/columns.
1428 // -1 is used as a magic number meaning empty column.
1430 for (int col
= 0; col
< ncols
; col
++)
1431 if ( m_colWidths
[ col
] != -1 )
1432 width
+= m_colWidths
[ col
] + m_hgap
;
1437 for (int row
= 0; row
< nrows
; row
++)
1438 if ( m_rowHeights
[ row
] != -1 )
1439 height
+= m_rowHeights
[ row
] + m_vgap
;
1443 m_calculatedMinSize
= wxSize( width
, height
);
1444 return m_calculatedMinSize
;
1447 void wxFlexGridSizer::AdjustForFlexDirection()
1449 // the logic in CalcMin works when we resize flexibly in both directions
1450 // but maybe this is not the case
1451 if ( m_flexDirection
!= wxBOTH
)
1453 // select the array corresponding to the direction in which we do *not*
1455 wxArrayInt
& array
= m_flexDirection
== wxVERTICAL
? m_colWidths
1458 const size_t count
= array
.GetCount();
1460 // find the largest value in this array
1464 for ( n
= 0; n
< count
; ++n
)
1466 if ( array
[n
] > largest
)
1470 // and now fill it with the largest value
1471 for ( n
= 0; n
< count
; ++n
)
1479 void wxFlexGridSizer::AdjustForGrowables(const wxSize
& sz
, const wxSize
& minsz
,
1480 int nrows
, int ncols
)
1482 // what to do with the rows? by default, resize them proportionally
1483 if ( sz
.y
> minsz
.y
&& ( (m_flexDirection
& wxVERTICAL
) || (m_growMode
== wxFLEX_GROWMODE_SPECIFIED
) ) )
1485 int sum_proportions
= 0;
1486 int growable_space
= 0;
1489 for (idx
= 0; idx
< m_growableRows
.GetCount(); idx
++)
1491 // Since the number of rows/columns can change as items are
1492 // inserted/deleted, we need to verify at runtime that the
1493 // requested growable rows/columns are still valid.
1494 if (m_growableRows
[idx
] >= nrows
)
1497 // If all items in a row/column are hidden, that row/column will
1498 // have a dimension of -1. This causes the row/column to be
1499 // hidden completely.
1500 if (m_rowHeights
[ m_growableRows
[idx
] ] == -1)
1502 sum_proportions
+= m_growableRowsProportions
[idx
];
1503 growable_space
+= m_rowHeights
[ m_growableRows
[idx
] ];
1509 for (idx
= 0; idx
< m_growableRows
.GetCount(); idx
++)
1511 if (m_growableRows
[idx
] >= nrows
)
1513 if (m_rowHeights
[ m_growableRows
[idx
] ] == -1)
1514 m_rowHeights
[ m_growableRows
[idx
] ] = 0;
1517 int delta
= (sz
.y
- minsz
.y
);
1518 if (sum_proportions
== 0)
1519 delta
= (delta
/num
) + m_rowHeights
[ m_growableRows
[idx
] ];
1521 delta
= ((delta
+growable_space
)*m_growableRowsProportions
[idx
]) / sum_proportions
;
1522 m_rowHeights
[ m_growableRows
[idx
] ] = delta
;
1527 else if ( (m_growMode
== wxFLEX_GROWMODE_ALL
) && (sz
.y
> minsz
.y
) )
1529 // rounding problem?
1530 for ( int row
= 0; row
< nrows
; ++row
)
1531 m_rowHeights
[ row
] = sz
.y
/ nrows
;
1534 // the same logic as above but for the columns
1535 if ( sz
.x
> minsz
.x
&& ( (m_flexDirection
& wxHORIZONTAL
) || (m_growMode
== wxFLEX_GROWMODE_SPECIFIED
) ) )
1537 int sum_proportions
= 0;
1538 int growable_space
= 0;
1541 for (idx
= 0; idx
< m_growableCols
.GetCount(); idx
++)
1543 // Since the number of rows/columns can change as items are
1544 // inserted/deleted, we need to verify at runtime that the
1545 // requested growable rows/columns are still valid.
1546 if (m_growableCols
[idx
] >= ncols
)
1549 // If all items in a row/column are hidden, that row/column will
1550 // have a dimension of -1. This causes the column to be hidden
1552 if (m_colWidths
[ m_growableCols
[idx
] ] == -1)
1554 sum_proportions
+= m_growableColsProportions
[idx
];
1555 growable_space
+= m_colWidths
[ m_growableCols
[idx
] ];
1561 for (idx
= 0; idx
< m_growableCols
.GetCount(); idx
++)
1563 if (m_growableCols
[idx
] >= ncols
)
1565 if (m_colWidths
[ m_growableCols
[idx
] ] == -1)
1566 m_colWidths
[ m_growableCols
[idx
] ] = 0;
1569 int delta
= (sz
.x
- minsz
.x
);
1570 if (sum_proportions
== 0)
1571 delta
= (delta
/num
) + m_colWidths
[ m_growableCols
[idx
] ];
1573 delta
= ((delta
+growable_space
)*m_growableColsProportions
[idx
])/sum_proportions
;
1574 m_colWidths
[ m_growableCols
[idx
] ] = delta
;
1579 else if ( (m_growMode
== wxFLEX_GROWMODE_ALL
) && (sz
.x
> minsz
.x
) )
1581 for ( int col
=0; col
< ncols
; ++col
)
1582 m_colWidths
[ col
] = sz
.x
/ ncols
;
1587 void wxFlexGridSizer::AddGrowableRow( size_t idx
, int proportion
)
1589 m_growableRows
.Add( idx
);
1590 m_growableRowsProportions
.Add( proportion
);
1593 void wxFlexGridSizer::AddGrowableCol( size_t idx
, int proportion
)
1595 m_growableCols
.Add( idx
);
1596 m_growableColsProportions
.Add( proportion
);
1599 // helper function for RemoveGrowableCol/Row()
1601 DoRemoveFromArrays(size_t idx
, wxArrayInt
& items
, wxArrayInt
& proportions
)
1603 const size_t count
= items
.size();
1604 for ( size_t n
= 0; n
< count
; n
++ )
1606 if ( (size_t)items
[n
] == idx
)
1609 proportions
.RemoveAt(n
);
1614 wxFAIL_MSG( _T("column/row is already not growable") );
1617 void wxFlexGridSizer::RemoveGrowableCol( size_t idx
)
1619 DoRemoveFromArrays(idx
, m_growableCols
, m_growableColsProportions
);
1622 void wxFlexGridSizer::RemoveGrowableRow( size_t idx
)
1624 DoRemoveFromArrays(idx
, m_growableRows
, m_growableRowsProportions
);
1627 //---------------------------------------------------------------------------
1629 //---------------------------------------------------------------------------
1631 wxBoxSizer::wxBoxSizer( int orient
)
1632 : m_orient( orient
)
1636 void wxBoxSizer::RecalcSizes()
1638 if (m_children
.GetCount() == 0)
1644 if (m_orient
== wxHORIZONTAL
)
1645 delta
= m_size
.x
- m_fixedWidth
;
1647 delta
= m_size
.y
- m_fixedHeight
;
1650 wxPoint
pt( m_position
);
1652 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1655 wxSizerItem
*item
= node
->GetData();
1657 if (item
->IsShown())
1659 wxSize
size( item
->GetMinSizeWithBorder() );
1661 if (m_orient
== wxVERTICAL
)
1663 wxCoord height
= size
.y
;
1664 if (item
->GetProportion())
1666 // Because of at least one visible item has non-zero
1667 // proportion then m_stretchable is not zero
1668 height
= (delta
* item
->GetProportion()) / m_stretchable
;
1671 wxPoint
child_pos( pt
);
1672 wxSize
child_size( size
.x
, height
);
1674 if (item
->GetFlag() & (wxEXPAND
| wxSHAPED
))
1675 child_size
.x
= m_size
.x
;
1676 else if (item
->GetFlag() & wxALIGN_RIGHT
)
1677 child_pos
.x
+= m_size
.x
- size
.x
;
1678 else if (item
->GetFlag() & (wxCENTER
| wxALIGN_CENTER_HORIZONTAL
))
1679 // XXX wxCENTER is added for backward compatibility;
1680 // wxALIGN_CENTER should be used in new code
1681 child_pos
.x
+= (m_size
.x
- size
.x
) / 2;
1683 item
->SetDimension( child_pos
, child_size
);
1689 wxCoord width
= size
.x
;
1690 if (item
->GetProportion())
1692 // Because of at least one visible item has non-zero
1693 // proportion then m_stretchable is not zero
1694 width
= (delta
* item
->GetProportion()) / m_stretchable
;
1697 wxPoint
child_pos( pt
);
1698 wxSize
child_size( width
, size
.y
);
1700 if (item
->GetFlag() & (wxEXPAND
| wxSHAPED
))
1701 child_size
.y
= m_size
.y
;
1702 else if (item
->GetFlag() & wxALIGN_BOTTOM
)
1703 child_pos
.y
+= m_size
.y
- size
.y
;
1704 else if (item
->GetFlag() & (wxCENTER
| wxALIGN_CENTER_VERTICAL
))
1705 // XXX wxCENTER is added for backward compatibility;
1706 // wxALIGN_CENTER should be used in new code
1707 child_pos
.y
+= (m_size
.y
- size
.y
) / 2;
1709 if ( m_containingWindow
)
1711 child_pos
.x
= m_containingWindow
->AdjustForLayoutDirection
1719 item
->SetDimension( child_pos
, child_size
);
1725 node
= node
->GetNext();
1729 wxSize
wxBoxSizer::CalcMin()
1731 if (m_children
.GetCount() == 0)
1740 // precalc item minsizes and count proportions
1741 wxSizerItemList::compatibility_iterator node
= m_children
.GetFirst();
1744 wxSizerItem
*item
= node
->GetData();
1746 if ( item
->IsShown() )
1748 item
->CalcMin(); // result is stored in the item
1750 m_stretchable
+= item
->GetProportion();
1753 node
= node
->GetNext();
1756 // Total minimum size (width or height) of sizer
1759 node
= m_children
.GetFirst();
1762 wxSizerItem
*item
= node
->GetData();
1764 if (item
->IsShown() && item
->GetProportion() != 0)
1766 int stretch
= item
->GetProportion();
1767 wxSize
size( item
->GetMinSizeWithBorder() );
1770 // Integer division rounded up is (a + b - 1) / b
1771 // Round up needed in order to guarantee that all
1772 // all items will have size not less then their min size
1773 if (m_orient
== wxHORIZONTAL
)
1774 minSize
= ( size
.x
*m_stretchable
+ stretch
- 1)/stretch
;
1776 minSize
= ( size
.y
*m_stretchable
+ stretch
- 1)/stretch
;
1778 if (minSize
> maxMinSize
)
1779 maxMinSize
= minSize
;
1781 node
= node
->GetNext();
1784 // Calculate overall minimum size
1785 node
= m_children
.GetFirst();
1788 wxSizerItem
*item
= node
->GetData();
1790 if (item
->IsShown())
1792 wxSize
size( item
->GetMinSizeWithBorder() );
1793 if (item
->GetProportion() != 0)
1795 if (m_orient
== wxHORIZONTAL
)
1796 size
.x
= (maxMinSize
*item
->GetProportion())/m_stretchable
;
1798 size
.y
= (maxMinSize
*item
->GetProportion())/m_stretchable
;
1802 if (m_orient
== wxVERTICAL
)
1804 m_fixedHeight
+= size
.y
;
1805 m_fixedWidth
= wxMax( m_fixedWidth
, size
.x
);
1809 m_fixedWidth
+= size
.x
;
1810 m_fixedHeight
= wxMax( m_fixedHeight
, size
.y
);
1814 if (m_orient
== wxHORIZONTAL
)
1816 m_minWidth
+= size
.x
;
1817 m_minHeight
= wxMax( m_minHeight
, size
.y
);
1821 m_minHeight
+= size
.y
;
1822 m_minWidth
= wxMax( m_minWidth
, size
.x
);
1825 node
= node
->GetNext();
1828 return wxSize( m_minWidth
, m_minHeight
);
1831 //---------------------------------------------------------------------------
1833 //---------------------------------------------------------------------------
1837 wxStaticBoxSizer::wxStaticBoxSizer( wxStaticBox
*box
, int orient
)
1838 : wxBoxSizer( orient
),
1841 wxASSERT_MSG( box
, wxT("wxStaticBoxSizer needs a static box") );
1843 // do this so that our Detach() is called if the static box is destroyed
1845 m_staticBox
->SetContainingSizer(this);
1848 wxStaticBoxSizer::wxStaticBoxSizer(int orient
, wxWindow
*win
, const wxString
& s
)
1849 : wxBoxSizer(orient
),
1850 m_staticBox(new wxStaticBox(win
, wxID_ANY
, s
))
1853 m_staticBox
->SetContainingSizer(this);
1856 wxStaticBoxSizer::~wxStaticBoxSizer()
1861 static void GetStaticBoxBorders( wxStaticBox
*box
,
1865 // this has to be done platform by platform as there is no way to
1866 // guess the thickness of a wxStaticBox border
1867 box
->GetBordersForSizer(borderTop
, borderOther
);
1870 void wxStaticBoxSizer::RecalcSizes()
1872 int top_border
, other_border
;
1873 GetStaticBoxBorders(m_staticBox
, &top_border
, &other_border
);
1875 m_staticBox
->SetSize( m_position
.x
, m_position
.y
, m_size
.x
, m_size
.y
);
1877 wxPoint
old_pos( m_position
);
1878 m_position
.x
+= other_border
;
1879 m_position
.y
+= top_border
;
1880 wxSize
old_size( m_size
);
1881 m_size
.x
-= 2*other_border
;
1882 m_size
.y
-= top_border
+ other_border
;
1884 wxBoxSizer::RecalcSizes();
1886 m_position
= old_pos
;
1890 wxSize
wxStaticBoxSizer::CalcMin()
1892 int top_border
, other_border
;
1893 GetStaticBoxBorders(m_staticBox
, &top_border
, &other_border
);
1895 wxSize
ret( wxBoxSizer::CalcMin() );
1896 ret
.x
+= 2*other_border
;
1897 ret
.y
+= other_border
+ top_border
;
1902 void wxStaticBoxSizer::ShowItems( bool show
)
1904 m_staticBox
->Show( show
);
1905 wxBoxSizer::ShowItems( show
);
1908 bool wxStaticBoxSizer::Detach( wxWindow
*window
)
1910 // avoid deleting m_staticBox in our dtor if it's being detached from the
1911 // sizer (which can happen because it's being already destroyed for
1913 if ( window
== m_staticBox
)
1919 return wxSizer::Detach( window
);
1922 #endif // wxUSE_STATBOX
1926 wxStdDialogButtonSizer::wxStdDialogButtonSizer()
1927 : wxBoxSizer(wxHORIZONTAL
)
1929 // Vertical buttons with lots of space on either side
1930 // looks rubbish on WinCE, so let's not do this for now.
1931 // If we are going to use vertical buttons, we should
1932 // put the sizer to the right of other controls in the dialog,
1933 // and that's beyond the scope of this sizer.
1935 bool is_pda
= (wxSystemSettings::GetScreenType() <= wxSYS_SCREEN_PDA
);
1936 // If we have a PDA screen, put yes/no button over
1937 // all other buttons, otherwise on the left side.
1939 m_orient
= wxVERTICAL
;
1942 m_buttonAffirmative
= NULL
;
1943 m_buttonApply
= NULL
;
1944 m_buttonNegative
= NULL
;
1945 m_buttonCancel
= NULL
;
1946 m_buttonHelp
= NULL
;
1949 void wxStdDialogButtonSizer::AddButton(wxButton
*mybutton
)
1951 switch (mybutton
->GetId())
1956 m_buttonAffirmative
= mybutton
;
1959 m_buttonApply
= mybutton
;
1962 m_buttonNegative
= mybutton
;
1965 m_buttonCancel
= mybutton
;
1968 case wxID_CONTEXT_HELP
:
1969 m_buttonHelp
= mybutton
;
1976 void wxStdDialogButtonSizer::SetAffirmativeButton( wxButton
*button
)
1978 m_buttonAffirmative
= button
;
1981 void wxStdDialogButtonSizer::SetNegativeButton( wxButton
*button
)
1983 m_buttonNegative
= button
;
1986 void wxStdDialogButtonSizer::SetCancelButton( wxButton
*button
)
1988 m_buttonCancel
= button
;
1991 void wxStdDialogButtonSizer::Realize()
1994 Add(0, 0, 0, wxLEFT
, 6);
1996 Add((wxWindow
*)m_buttonHelp
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 6);
1998 if (m_buttonNegative
){
1999 // HIG POLICE BULLETIN - destructive buttons need extra padding
2000 // 24 pixels on either side
2001 Add((wxWindow
*)m_buttonNegative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 12);
2004 // extra whitespace between help/negative and cancel/ok buttons
2005 Add(0, 0, 1, wxEXPAND
, 0);
2007 if (m_buttonCancel
){
2008 Add((wxWindow
*)m_buttonCancel
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 6);
2009 // Cancel or help should be default
2010 // m_buttonCancel->SetDefaultButton();
2013 // Ugh, Mac doesn't really have apply dialogs, so I'll just
2014 // figure the best place is between Cancel and OK
2016 Add((wxWindow
*)m_buttonApply
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 6);
2018 if (m_buttonAffirmative
){
2019 Add((wxWindow
*)m_buttonAffirmative
, 0, wxALIGN_CENTRE
| wxLEFT
, 6);
2021 if (m_buttonAffirmative
->GetId() == wxID_SAVE
){
2022 // these buttons have set labels under Mac so we should use them
2023 m_buttonAffirmative
->SetLabel(_("Save"));
2024 if (m_buttonNegative
)
2025 m_buttonNegative
->SetLabel(_("Don't Save"));
2029 // Extra space around and at the right
2031 #elif defined(__WXGTK20__)
2032 Add(0, 0, 0, wxLEFT
, 9);
2034 Add((wxWindow
*)m_buttonHelp
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 3);
2036 // extra whitespace between help and cancel/ok buttons
2037 Add(0, 0, 1, wxEXPAND
, 0);
2039 if (m_buttonNegative
){
2040 Add((wxWindow
*)m_buttonNegative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 3);
2043 if (m_buttonCancel
){
2044 Add((wxWindow
*)m_buttonCancel
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 3);
2045 // Cancel or help should be default
2046 // m_buttonCancel->SetDefaultButton();
2050 Add((wxWindow
*)m_buttonApply
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, 3);
2052 if (m_buttonAffirmative
)
2053 Add((wxWindow
*)m_buttonAffirmative
, 0, wxALIGN_CENTRE
| wxLEFT
, 6);
2054 #elif defined(__WXMSW__)
2057 // right-justify buttons
2058 Add(0, 0, 1, wxEXPAND
, 0);
2060 if (m_buttonAffirmative
){
2061 Add((wxWindow
*)m_buttonAffirmative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonAffirmative
->ConvertDialogToPixels(wxSize(2, 0)).x
);
2064 if (m_buttonNegative
){
2065 Add((wxWindow
*)m_buttonNegative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonNegative
->ConvertDialogToPixels(wxSize(2, 0)).x
);
2068 if (m_buttonCancel
){
2069 Add((wxWindow
*)m_buttonCancel
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonCancel
->ConvertDialogToPixels(wxSize(2, 0)).x
);
2072 Add((wxWindow
*)m_buttonApply
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonApply
->ConvertDialogToPixels(wxSize(2, 0)).x
);
2075 Add((wxWindow
*)m_buttonHelp
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonHelp
->ConvertDialogToPixels(wxSize(2, 0)).x
);
2077 // GTK+1 and any other platform
2079 // Add(0, 0, 0, wxLEFT, 5); // Not sure what this was for but it unbalances the dialog
2081 Add((wxWindow
*)m_buttonHelp
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonHelp
->ConvertDialogToPixels(wxSize(4, 0)).x
);
2083 // extra whitespace between help and cancel/ok buttons
2084 Add(0, 0, 1, wxEXPAND
, 0);
2087 Add((wxWindow
*)m_buttonApply
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonApply
->ConvertDialogToPixels(wxSize(4, 0)).x
);
2089 if (m_buttonAffirmative
){
2090 Add((wxWindow
*)m_buttonAffirmative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonAffirmative
->ConvertDialogToPixels(wxSize(4, 0)).x
);
2093 if (m_buttonNegative
){
2094 Add((wxWindow
*)m_buttonNegative
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonNegative
->ConvertDialogToPixels(wxSize(4, 0)).x
);
2097 if (m_buttonCancel
){
2098 Add((wxWindow
*)m_buttonCancel
, 0, wxALIGN_CENTRE
| wxLEFT
| wxRIGHT
, m_buttonCancel
->ConvertDialogToPixels(wxSize(4, 0)).x
);
2099 // Cancel or help should be default
2100 // m_buttonCancel->SetDefaultButton();
2106 #endif // wxUSE_BUTTON