]> git.saurik.com Git - wxWidgets.git/blame - src/common/sizer.cpp
use GetPath() instead of GetFullPath() in the directory-oriented functions (patch...
[wxWidgets.git] / src / common / sizer.cpp
CommitLineData
5279a24d 1/////////////////////////////////////////////////////////////////////////////
ca3e85cf 2// Name: src/common/sizer.cpp
1044a386 3// Purpose: provide new wxSizer class for layout
aa5973ee
JS
4// Author: Robert Roebling and Robin Dunn, contributions by
5// Dirk Holtwick, Ron Lee
566d84a7 6// Modified by: Ron Lee
0c0d686f 7// Created:
5279a24d 8// RCS-ID: $Id$
aa5973ee 9// Copyright: (c) Robin Dunn, Robert Roebling
65571936 10// Licence: wxWindows licence
5279a24d
RR
11/////////////////////////////////////////////////////////////////////////////
12
77671fd2
VZ
13// For compilers that support precompilation, includes "wx.h".
14#include "wx/wxprec.h"
15
16#ifdef __BORLANDC__
17 #pragma hdrstop
18#endif
19
0f769aab
WS
20#ifndef WX_PRECOMP
21 #include "wx/string.h"
22 #include "wx/intl.h"
2fb18bf4 23 #include "wx/math.h"
de6185e2 24 #include "wx/utils.h"
9eddec69 25 #include "wx/settings.h"
0f769aab
WS
26#endif // WX_PRECOMP
27
5279a24d 28#include "wx/sizer.h"
27ea1d8a 29#include "wx/statbox.h"
c54b92d3 30#include "wx/listimpl.cpp"
88a7a4e1 31
adbf2d73
VS
32#if WXWIN_COMPATIBILITY_2_4
33 #include "wx/notebook.h"
34#endif
5279a24d 35
0c0d686f
RD
36//---------------------------------------------------------------------------
37
9cbee2ce
RL
38IMPLEMENT_CLASS(wxSizerItem, wxObject)
39IMPLEMENT_CLASS(wxSizer, wxObject)
40IMPLEMENT_CLASS(wxGridSizer, wxSizer)
41IMPLEMENT_CLASS(wxFlexGridSizer, wxGridSizer)
42IMPLEMENT_CLASS(wxBoxSizer, wxSizer)
1e6feb95 43#if wxUSE_STATBOX
9cbee2ce 44IMPLEMENT_CLASS(wxStaticBoxSizer, wxBoxSizer)
1e6feb95 45#endif
974c2a59 46#if wxUSE_BUTTON
acf2ac37 47IMPLEMENT_CLASS(wxStdDialogButtonSizer, wxBoxSizer)
974c2a59 48#endif
0c0d686f 49
259c43f6 50WX_DEFINE_EXPORTED_LIST( wxSizerItemList )
12a3f227 51
066f1b7a 52/*
8b2bac62
WS
53 TODO PROPERTIES
54 sizeritem
55 object
56 object_ref
57 minsize
58 option
59 flag
60 border
066f1b7a 61 spacer
8b2bac62
WS
62 option
63 flag
64 borfder
65 boxsizer
66 orient
066f1b7a 67 staticboxsizer
8b2bac62
WS
68 orient
69 label
70 gridsizer
71 rows
72 cols
73 vgap
74 hgap
75 flexgridsizer
76 rows
77 cols
78 vgap
79 hgap
80 growablerows
81 growablecols
066f1b7a
SC
82 minsize
83*/
ccbc8038 84
50c06297 85// ----------------------------------------------------------------------------
3417c2cd 86// wxSizerItem
50c06297 87// ----------------------------------------------------------------------------
ccbc8038
VZ
88
89void wxSizerItem::Init(const wxSizerFlags& flags)
90{
91 Init();
92
93 m_proportion = flags.GetProportion();
94 m_flag = flags.GetFlags();
95 m_border = flags.GetBorderInPixels();
96}
97
50c06297
VZ
98wxSizerItem::wxSizerItem()
99{
100 Init();
101
102 m_proportion = 0;
103 m_border = 0;
104 m_flag = 0;
105
106 m_kind = Item_None;
107}
108
109// window item
110void wxSizerItem::SetWindow(wxWindow *window)
111{
112 wxCHECK_RET( window, _T("NULL window in wxSizerItem::SetWindow()") );
113
114 m_kind = Item_Window;
115 m_window = window;
116
117 // window doesn't become smaller than its initial size, whatever happens
ba763a45 118 m_minSize = window->GetSize();
8b2bac62 119
50c06297
VZ
120 if ( m_flag & wxFIXED_MINSIZE )
121 window->SetMinSize(m_minSize);
122
36461f58 123 // aspect ratio calculated from initial size
50c06297
VZ
124 SetRatio(m_minSize);
125}
36461f58 126
50c06297
VZ
127wxSizerItem::wxSizerItem(wxWindow *window,
128 int proportion,
129 int flag,
130 int border,
131 wxObject* userData)
132 : m_proportion(proportion),
133 m_border(border),
134 m_flag(flag),
135 m_userData(userData)
136{
137 SetWindow(window);
5279a24d
RR
138}
139
50c06297
VZ
140// sizer item
141void wxSizerItem::SetSizer(wxSizer *sizer)
5279a24d 142{
50c06297
VZ
143 m_kind = Item_Sizer;
144 m_sizer = sizer;
5279a24d
RR
145}
146
50c06297
VZ
147wxSizerItem::wxSizerItem(wxSizer *sizer,
148 int proportion,
149 int flag,
150 int border,
151 wxObject* userData)
152 : m_proportion(proportion),
153 m_border(border),
154 m_flag(flag),
155 m_ratio(0.0),
156 m_userData(userData)
20b35a69 157{
50c06297 158 SetSizer(sizer);
ccbc8038 159
50c06297
VZ
160 // m_minSize is set later
161}
162
163// spacer item
164void wxSizerItem::SetSpacer(const wxSize& size)
165{
166 m_kind = Item_Spacer;
167 m_spacer = new wxSizerSpacer(size);
168 m_minSize = size;
169 SetRatio(size);
170}
171
172wxSizerItem::wxSizerItem(int width,
173 int height,
174 int proportion,
175 int flag,
176 int border,
177 wxObject* userData)
178 : m_minSize(width, height), // minimal size is the initial size
179 m_proportion(proportion),
180 m_border(border),
181 m_flag(flag),
182 m_userData(userData)
183{
184 SetSpacer(width, height);
20b35a69
RD
185}
186
0c0d686f
RD
187wxSizerItem::~wxSizerItem()
188{
f91e8382
VZ
189 delete m_userData;
190
50c06297 191 switch ( m_kind )
f91e8382 192 {
50c06297
VZ
193 case Item_None:
194 break;
195
196 case Item_Window:
197 m_window->SetContainingSizer(NULL);
198 break;
199
200 case Item_Sizer:
201 delete m_sizer;
202 break;
203
204 case Item_Spacer:
205 delete m_spacer;
206 break;
207
208 case Item_Max:
209 default:
210 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
f91e8382 211 }
0c0d686f
RD
212}
213
50c06297
VZ
214wxSize wxSizerItem::GetSpacer() const
215{
216 wxSize size;
217 if ( m_kind == Item_Spacer )
218 size = m_spacer->GetSize();
219
220 return size;
221}
222
0c0d686f 223
9cbee2ce 224wxSize wxSizerItem::GetSize() const
5279a24d 225{
d597fcb7 226 wxSize ret;
50c06297
VZ
227 switch ( m_kind )
228 {
229 case Item_None:
230 break;
231
232 case Item_Window:
233 ret = m_window->GetSize();
234 break;
235
236 case Item_Sizer:
237 ret = m_sizer->GetSize();
238 break;
239
240 case Item_Spacer:
241 ret = m_spacer->GetSize();
242 break;
243
244 case Item_Max:
245 default:
246 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
247 }
0c0d686f 248
d597fcb7
RR
249 if (m_flag & wxWEST)
250 ret.x += m_border;
251 if (m_flag & wxEAST)
252 ret.x += m_border;
253 if (m_flag & wxNORTH)
254 ret.y += m_border;
255 if (m_flag & wxSOUTH)
256 ret.y += m_border;
0c0d686f 257
d597fcb7 258 return ret;
5279a24d
RR
259}
260
3417c2cd 261wxSize wxSizerItem::CalcMin()
c62ac5b6 262{
3417c2cd 263 if (IsSizer())
be2577e4 264 {
ba763a45 265 m_minSize = m_sizer->GetMinSize();
d13d8d4e 266
be2577e4
RD
267 // if we have to preserve aspect ratio _AND_ this is
268 // the first-time calculation, consider ret to be initial size
2fb18bf4 269 if ( (m_flag & wxSHAPED) && wxIsNullDouble(m_ratio) )
36461f58 270 SetRatio(m_minSize);
be2577e4 271 }
ba763a45 272 else if ( IsWindow() )
d13d8d4e 273 {
ba763a45
RD
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();
d13d8d4e 277 }
0c0d686f 278
ba763a45
RD
279 return GetMinSizeWithBorder();
280}
281
282wxSize wxSizerItem::GetMinSizeWithBorder() const
283{
284 wxSize ret = m_minSize;
285
d597fcb7
RR
286 if (m_flag & wxWEST)
287 ret.x += m_border;
288 if (m_flag & wxEAST)
289 ret.x += m_border;
290 if (m_flag & wxNORTH)
291 ret.y += m_border;
292 if (m_flag & wxSOUTH)
293 ret.y += m_border;
8b2bac62 294
d597fcb7 295 return ret;
c62ac5b6
RR
296}
297
ba763a45 298
fbfb8bcc 299void wxSizerItem::SetDimension( const wxPoint& pos_, const wxSize& size_ )
c62ac5b6 300{
fbfb8bcc
VZ
301 wxPoint pos = pos_;
302 wxSize size = size_;
cdddaeea 303 if (m_flag & wxSHAPED)
d597fcb7 304 {
be2577e4
RD
305 // adjust aspect ratio
306 int rwidth = (int) (size.y * m_ratio);
cdddaeea
VZ
307 if (rwidth > size.x)
308 {
be2577e4
RD
309 // fit horizontally
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
317 size.y =rheight;
cdddaeea
VZ
318 }
319 else if (rwidth < size.x)
320 {
be2577e4
RD
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);
326 size.x = rwidth;
327 }
328 }
33ac7e6f 329
cdddaeea
VZ
330 // This is what GetPosition() returns. Since we calculate
331 // borders afterwards, GetPosition() will be the left/top
332 // corner of the surrounding border.
333 m_pos = pos;
334
335 if (m_flag & wxWEST)
336 {
337 pos.x += m_border;
338 size.x -= m_border;
339 }
340 if (m_flag & wxEAST)
341 {
342 size.x -= m_border;
343 }
344 if (m_flag & wxNORTH)
345 {
346 pos.y += m_border;
347 size.y -= m_border;
348 }
349 if (m_flag & wxSOUTH)
350 {
351 size.y -= m_border;
352 }
0c0d686f 353
50c06297 354 m_rect = wxRect(pos, size);
0c0d686f 355
50c06297
VZ
356 switch ( m_kind )
357 {
358 case Item_None:
359 wxFAIL_MSG( _T("can't set size of uninitialized sizer item") );
360 break;
361
362 case Item_Window:
363 m_window->SetSize(pos.x, pos.y, size.x, size.y,
364 wxSIZE_ALLOW_MINUS_ONE);
365 break;
366
367 case Item_Sizer:
368 m_sizer->SetDimension(pos.x, pos.y, size.x, size.y);
369 break;
370
371 case Item_Spacer:
372 m_spacer->SetSize(size);
373 break;
374
375 case Item_Max:
376 default:
377 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
378 }
c62ac5b6
RR
379}
380
84f7908b
RR
381void wxSizerItem::DeleteWindows()
382{
50c06297 383 switch ( m_kind )
77aa9abd 384 {
50c06297
VZ
385 case Item_None:
386 case Item_Spacer:
387 break;
388
389 case Item_Window:
caa2490c
RN
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);
50c06297 394 m_window->Destroy();
caa2490c
RN
395 //Putting this after the switch will result in a spacer
396 //not being deleted properly on destruction
902725ee 397 m_kind = Item_None;
50c06297
VZ
398 break;
399
400 case Item_Sizer:
401 m_sizer->DeleteWindows();
402 break;
403
404 case Item_Max:
405 default:
406 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
77aa9abd 407 }
be90c029 408
84f7908b
RR
409}
410
50c06297 411void wxSizerItem::Show( bool show )
5279a24d 412{
50c06297
VZ
413 switch ( m_kind )
414 {
415 case Item_None:
416 wxFAIL_MSG( _T("can't show uninitialized sizer item") );
417 break;
5279a24d 418
50c06297
VZ
419 case Item_Window:
420 m_window->Show(show);
421 break;
5279a24d 422
50c06297 423 case Item_Sizer:
3a5910cb 424 m_sizer->Show(show);
50c06297
VZ
425 break;
426
427 case Item_Spacer:
428 m_spacer->Show(show);
429 break;
430
431 case Item_Max:
432 default:
433 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
434 }
5279a24d
RR
435}
436
50c06297 437bool wxSizerItem::IsShown() const
12a3f227 438{
50c06297
VZ
439 switch ( m_kind )
440 {
441 case Item_None:
f1662177
VZ
442 // we may be called from CalcMin(), just return false so that we're
443 // not used
50c06297
VZ
444 break;
445
446 case Item_Window:
447 return m_window->IsShown();
12a3f227 448
50c06297 449 case Item_Sizer:
f303d69f
VZ
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)
453 {
454 for ( wxSizerItemList::compatibility_iterator
455 node = m_sizer->GetChildren().GetFirst();
456 node;
457 node = node->GetNext() )
458 {
459 if ( node->GetData()->IsShown() )
460 return true;
461 }
462 }
463 return false;
12a3f227 464
50c06297
VZ
465 case Item_Spacer:
466 return m_spacer->IsShown();
467
468 case Item_Max:
469 default:
470 wxFAIL_MSG( _T("unexpected wxSizerItem::m_kind") );
471 }
472
473 return false;
12a3f227
RL
474}
475
ca3e85cf 476#if WXWIN_COMPATIBILITY_2_6
12a3f227
RL
477void wxSizerItem::SetOption( int option )
478{
479 SetProportion( option );
480}
481
482int wxSizerItem::GetOption() const
483{
484 return GetProportion();
485}
ca3e85cf 486#endif // WXWIN_COMPATIBILITY_2_6
12a3f227
RL
487
488
5279a24d 489//---------------------------------------------------------------------------
3417c2cd 490// wxSizer
5279a24d
RR
491//---------------------------------------------------------------------------
492
3417c2cd 493wxSizer::~wxSizer()
5279a24d 494{
222ed1d6 495 WX_CLEAR_LIST(wxSizerItemList, m_children);
5279a24d 496}
0c0d686f 497
56eee37f 498wxSizerItem* wxSizer::Insert( size_t index, wxSizerItem *item )
12a3f227
RL
499{
500 m_children.Insert( index, item );
0c0d686f 501
50c06297 502 if ( item->GetWindow() )
12a3f227 503 item->GetWindow()->SetContainingSizer( this );
56eee37f
WS
504
505 return item;
12a3f227
RL
506}
507
ca3e85cf 508#if WXWIN_COMPATIBILITY_2_6
12a3f227
RL
509bool wxSizer::Remove( wxWindow *window )
510{
511 return Detach( window );
42b4e99e 512}
ca3e85cf 513#endif // WXWIN_COMPATIBILITY_2_6
42b4e99e
RR
514
515bool wxSizer::Remove( wxSizer *sizer )
516{
12a3f227 517 wxASSERT_MSG( sizer, _T("Removing NULL sizer") );
0c0d686f 518
222ed1d6 519 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
42b4e99e
RR
520 while (node)
521 {
12a3f227
RL
522 wxSizerItem *item = node->GetData();
523
3ca6a5f0 524 if (item->GetSizer() == sizer)
222ed1d6
MB
525 {
526 delete item;
527 m_children.Erase( node );
528 return true;
529 }
12a3f227
RL
530
531 node = node->GetNext();
42b4e99e 532 }
0c0d686f 533
e0d8fb45 534 return false;
42b4e99e
RR
535}
536
e0d8fb45 537bool wxSizer::Remove( int index )
42b4e99e 538{
e0d8fb45
VZ
539 wxCHECK_MSG( index >= 0 && (size_t)index < m_children.GetCount(),
540 false,
12a3f227 541 _T("Remove index is out of range") );
0c0d686f 542
222ed1d6 543 wxSizerItemList::compatibility_iterator node = m_children.Item( index );
0c0d686f 544
e0d8fb45 545 wxCHECK_MSG( node, false, _T("Failed to find child node") );
12a3f227 546
e0d8fb45 547 wxSizerItem *item = node->GetData();
9cbee2ce 548
50c06297 549 if ( item->IsWindow() )
9cbee2ce
RL
550 item->GetWindow()->SetContainingSizer( NULL );
551
222ed1d6
MB
552 delete item;
553 m_children.Erase( node );
554 return true;
42b4e99e 555}
0c0d686f 556
00976fe5
RL
557bool wxSizer::Detach( wxSizer *sizer )
558{
12a3f227 559 wxASSERT_MSG( sizer, _T("Detaching NULL sizer") );
00976fe5 560
222ed1d6 561 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
00976fe5
RL
562 while (node)
563 {
12a3f227
RL
564 wxSizerItem *item = node->GetData();
565
00976fe5
RL
566 if (item->GetSizer() == sizer)
567 {
96fdbb60 568 item->DetachSizer();
89c20ac1 569 delete item;
222ed1d6
MB
570 m_children.Erase( node );
571 return true;
12a3f227
RL
572 }
573 node = node->GetNext();
574 }
575
e0d8fb45 576 return false;
12a3f227
RL
577}
578
579bool wxSizer::Detach( wxWindow *window )
580{
581 wxASSERT_MSG( window, _T("Detaching NULL window") );
582
222ed1d6 583 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
12a3f227
RL
584 while (node)
585 {
586 wxSizerItem *item = node->GetData();
587
588 if (item->GetWindow() == window)
589 {
590 item->GetWindow()->SetContainingSizer( NULL );
89c20ac1 591 delete item;
222ed1d6
MB
592 m_children.Erase( node );
593 return true;
00976fe5 594 }
12a3f227 595 node = node->GetNext();
00976fe5
RL
596 }
597
e0d8fb45 598 return false;
00976fe5
RL
599}
600
e0d8fb45 601bool wxSizer::Detach( int index )
00976fe5 602{
e0d8fb45
VZ
603 wxCHECK_MSG( index >= 0 && (size_t)index < m_children.GetCount(),
604 false,
12a3f227
RL
605 _T("Detach index is out of range") );
606
222ed1d6 607 wxSizerItemList::compatibility_iterator node = m_children.Item( index );
00976fe5 608
e0d8fb45 609 wxCHECK_MSG( node, false, _T("Failed to find child node") );
00976fe5 610
e0d8fb45 611 wxSizerItem *item = node->GetData();
9cbee2ce 612
50c06297 613 if ( item->IsSizer() )
9cbee2ce 614 item->DetachSizer();
50c06297 615 else if ( item->IsWindow() )
9cbee2ce 616 item->GetWindow()->SetContainingSizer( NULL );
12a3f227 617
89c20ac1 618 delete item;
222ed1d6
MB
619 m_children.Erase( node );
620 return true;
00976fe5
RL
621}
622
84f7908b
RR
623void wxSizer::Clear( bool delete_windows )
624{
be90c029 625 // First clear the ContainingSizer pointers
222ed1d6 626 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
be90c029
RD
627 while (node)
628 {
12a3f227
RL
629 wxSizerItem *item = node->GetData();
630
be90c029 631 if (item->IsWindow())
12a3f227
RL
632 item->GetWindow()->SetContainingSizer( NULL );
633 node = node->GetNext();
be90c029
RD
634 }
635
636 // Destroy the windows if needed
84f7908b
RR
637 if (delete_windows)
638 DeleteWindows();
be90c029
RD
639
640 // Now empty the list
222ed1d6 641 WX_CLEAR_LIST(wxSizerItemList, m_children);
84f7908b
RR
642}
643
644void wxSizer::DeleteWindows()
645{
222ed1d6 646 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
84f7908b
RR
647 while (node)
648 {
12a3f227
RL
649 wxSizerItem *item = node->GetData();
650
84f7908b 651 item->DeleteWindows();
12a3f227 652 node = node->GetNext();
84f7908b
RR
653 }
654}
655
e5251d4f 656wxSize wxSizer::Fit( wxWindow *window )
5279a24d 657{
f43c7771
VZ
658 wxSize size(window->IsTopLevel() ? FitSize(window)
659 : GetMinWindowSize(window));
9ef2e675 660
ccf5c8a8 661 window->SetSize( size );
e5251d4f
VZ
662
663 return size;
5279a24d
RR
664}
665
566d84a7
RL
666void wxSizer::FitInside( wxWindow *window )
667{
668 wxSize size;
669 if (window->IsTopLevel())
670 size = VirtualFitSize( window );
671 else
672 size = GetMinClientSize( window );
673
674 window->SetVirtualSize( size );
675}
676
3417c2cd 677void wxSizer::Layout()
c62ac5b6 678{
ba763a45
RD
679 // (re)calculates minimums needed for each item and other preparations
680 // for layout
42b4e99e 681 CalcMin();
ba763a45
RD
682
683 // Applies the layout and repositions/resizes the items
c62ac5b6
RR
684 RecalcSizes();
685}
686
3417c2cd 687void wxSizer::SetSizeHints( wxWindow *window )
5279a24d 688{
34c3ffca
RL
689 // Preserve the window's max size hints, but set the
690 // lower bound according to the sizer calculations.
691
e5251d4f
VZ
692 wxSize size = Fit( window );
693
34c3ffca
RL
694 window->SetSizeHints( size.x,
695 size.y,
696 window->GetMaxWidth(),
697 window->GetMaxHeight() );
5279a24d
RR
698}
699
566d84a7
RL
700void wxSizer::SetVirtualSizeHints( wxWindow *window )
701{
702 // Preserve the window's max size hints, but set the
703 // lower bound according to the sizer calculations.
704
705 FitInside( window );
706 wxSize size( window->GetVirtualSize() );
707 window->SetVirtualSizeHints( size.x,
708 size.y,
709 window->GetMaxWidth(),
710 window->GetMaxHeight() );
711}
712
9cbee2ce 713wxSize wxSizer::GetMaxWindowSize( wxWindow *window ) const
65ba4113 714{
34c3ffca 715 return window->GetMaxSize();
65ba4113
GT
716}
717
3417c2cd 718wxSize wxSizer::GetMinWindowSize( wxWindow *window )
5279a24d 719{
12a3f227
RL
720 wxSize minSize( GetMinSize() );
721 wxSize size( window->GetSize() );
722 wxSize client_size( window->GetClientSize() );
723
77671fd2 724 return wxSize( minSize.x+size.x-client_size.x,
0c0d686f 725 minSize.y+size.y-client_size.y );
5279a24d
RR
726}
727
e11d436b
SC
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
731
65ba4113
GT
732// Return a window size that will fit within the screens dimensions
733wxSize wxSizer::FitSize( wxWindow *window )
734{
f43255e8
WS
735 if ( window->IsTopLevel() )
736 {
737 wxTopLevelWindow *tlw = wxDynamicCast(window, wxTopLevelWindow);
738 if ( tlw && tlw->IsAlwaysMaximized() )
739 {
740 return tlw->GetClientSize();
741 }
742 }
743
65ba4113
GT
744 wxSize size = GetMinWindowSize( window );
745 wxSize sizeMax = GetMaxWindowSize( window );
746
34c3ffca
RL
747 // Limit the size if sizeMax != wxDefaultSize
748
d775fa82 749 if ( size.x > sizeMax.x && sizeMax.x != wxDefaultCoord )
65ba4113 750 size.x = sizeMax.x;
d775fa82 751 if ( size.y > sizeMax.y && sizeMax.y != wxDefaultCoord )
65ba4113
GT
752 size.y = sizeMax.y;
753
754 return size;
755}
756
9cbee2ce 757wxSize wxSizer::GetMaxClientSize( wxWindow *window ) const
566d84a7
RL
758{
759 wxSize maxSize( window->GetMaxSize() );
760
50c06297 761 if ( maxSize != wxDefaultSize )
566d84a7
RL
762 {
763 wxSize size( window->GetSize() );
764 wxSize client_size( window->GetClientSize() );
765
766 return wxSize( maxSize.x + client_size.x - size.x,
767 maxSize.y + client_size.y - size.y );
768 }
769 else
770 return wxDefaultSize;
771}
772
1b0674f7 773wxSize wxSizer::GetMinClientSize( wxWindow *WXUNUSED(window) )
566d84a7
RL
774{
775 return GetMinSize(); // Already returns client size.
776}
777
778wxSize wxSizer::VirtualFitSize( wxWindow *window )
779{
780 wxSize size = GetMinClientSize( window );
781 wxSize sizeMax = GetMaxClientSize( window );
782
783 // Limit the size if sizeMax != wxDefaultSize
784
d775fa82 785 if ( size.x > sizeMax.x && sizeMax.x != wxDefaultCoord )
566d84a7 786 size.x = sizeMax.x;
d775fa82 787 if ( size.y > sizeMax.y && sizeMax.y != wxDefaultCoord )
566d84a7
RL
788 size.y = sizeMax.y;
789
790 return size;
791}
792
3417c2cd 793void wxSizer::SetDimension( int x, int y, int width, int height )
5279a24d
RR
794{
795 m_position.x = x;
796 m_position.y = y;
797 m_size.x = width;
798 m_size.y = height;
2b5f62a0 799 Layout();
5279a24d
RR
800}
801
f6bcfd97 802wxSize wxSizer::GetMinSize()
3ca6a5f0 803{
f6bcfd97
BP
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;
3ca6a5f0 807 return ret;
f6bcfd97
BP
808}
809
810void wxSizer::DoSetMinSize( int width, int height )
811{
812 m_minSize.x = width;
813 m_minSize.y = height;
814}
815
816bool wxSizer::DoSetItemMinSize( wxWindow *window, int width, int height )
817{
12a3f227
RL
818 wxASSERT_MSG( window, _T("SetMinSize for NULL window") );
819
820 // Is it our immediate child?
f6bcfd97 821
222ed1d6 822 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
f6bcfd97
BP
823 while (node)
824 {
12a3f227
RL
825 wxSizerItem *item = node->GetData();
826
3ca6a5f0
BP
827 if (item->GetWindow() == window)
828 {
1eba2193 829 item->SetMinSize( width, height );
e0d8fb45 830 return true;
3ca6a5f0 831 }
12a3f227 832 node = node->GetNext();
f6bcfd97
BP
833 }
834
12a3f227
RL
835 // No? Search any subsizers we own then
836
837 node = m_children.GetFirst();
f6bcfd97
BP
838 while (node)
839 {
12a3f227
RL
840 wxSizerItem *item = node->GetData();
841
842 if ( item->GetSizer() &&
843 item->GetSizer()->DoSetItemMinSize( window, width, height ) )
3ca6a5f0 844 {
12a3f227 845 // A child sizer found the requested windw, exit.
e0d8fb45 846 return true;
3ca6a5f0 847 }
12a3f227 848 node = node->GetNext();
f6bcfd97
BP
849 }
850
e0d8fb45 851 return false;
f6bcfd97
BP
852}
853
854bool wxSizer::DoSetItemMinSize( wxSizer *sizer, int width, int height )
855{
12a3f227 856 wxASSERT_MSG( sizer, _T("SetMinSize for NULL sizer") );
f6bcfd97 857
12a3f227
RL
858 // Is it our immediate child?
859
222ed1d6 860 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
f6bcfd97
BP
861 while (node)
862 {
12a3f227
RL
863 wxSizerItem *item = node->GetData();
864
3ca6a5f0
BP
865 if (item->GetSizer() == sizer)
866 {
f6bcfd97 867 item->GetSizer()->DoSetMinSize( width, height );
e0d8fb45 868 return true;
3ca6a5f0 869 }
12a3f227 870 node = node->GetNext();
f6bcfd97
BP
871 }
872
12a3f227
RL
873 // No? Search any subsizers we own then
874
875 node = m_children.GetFirst();
f6bcfd97
BP
876 while (node)
877 {
12a3f227
RL
878 wxSizerItem *item = node->GetData();
879
880 if ( item->GetSizer() &&
881 item->GetSizer()->DoSetItemMinSize( sizer, width, height ) )
3ca6a5f0 882 {
12a3f227 883 // A child found the requested sizer, exit.
e0d8fb45 884 return true;
3ca6a5f0 885 }
12a3f227 886 node = node->GetNext();
f6bcfd97
BP
887 }
888
e0d8fb45 889 return false;
f6bcfd97
BP
890}
891
12a3f227 892bool wxSizer::DoSetItemMinSize( size_t index, int width, int height )
f6bcfd97 893{
222ed1d6 894 wxSizerItemList::compatibility_iterator node = m_children.Item( index );
12a3f227 895
e0d8fb45 896 wxCHECK_MSG( node, false, _T("Failed to find child node") );
12a3f227
RL
897
898 wxSizerItem *item = node->GetData();
f6bcfd97 899
f6bcfd97
BP
900 if (item->GetSizer())
901 {
0ca5105b 902 // Sizers contains the minimal size in them, if not calculated ...
f6bcfd97
BP
903 item->GetSizer()->DoSetMinSize( width, height );
904 }
905 else
906 {
ba763a45 907 // ... but the minimal size of spacers and windows is stored via the item
1eba2193 908 item->SetMinSize( width, height );
f6bcfd97
BP
909 }
910
e0d8fb45 911 return true;
f6bcfd97
BP
912}
913
9f13661f 914wxSizerItem* wxSizer::GetItem( wxWindow *window, bool recursive )
2b5f62a0 915{
9f13661f 916 wxASSERT_MSG( window, _T("GetItem for NULL window") );
12a3f227 917
222ed1d6 918 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
2b5f62a0
VZ
919 while (node)
920 {
12a3f227 921 wxSizerItem *item = node->GetData();
2b5f62a0 922
12a3f227 923 if (item->GetWindow() == window)
2b5f62a0 924 {
9f13661f 925 return item;
2b5f62a0 926 }
8b2bac62
WS
927 else if (recursive && item->IsSizer())
928 {
9f13661f
WS
929 wxSizerItem *subitem = item->GetSizer()->GetItem( window, true );
930 if (subitem)
931 return subitem;
8b2bac62
WS
932 }
933
12a3f227 934 node = node->GetNext();
2b5f62a0 935 }
8b2bac62 936
9f13661f 937 return NULL;
2b5f62a0
VZ
938}
939
9f13661f 940wxSizerItem* wxSizer::GetItem( wxSizer *sizer, bool recursive )
2b5f62a0 941{
9f13661f 942 wxASSERT_MSG( sizer, _T("GetItem for NULL sizer") );
12a3f227 943
222ed1d6 944 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
2b5f62a0
VZ
945 while (node)
946 {
9f13661f 947 wxSizerItem *item = node->GetData();
2b5f62a0 948
12a3f227 949 if (item->GetSizer() == sizer)
2b5f62a0 950 {
9f13661f 951 return item;
2b5f62a0 952 }
8b2bac62
WS
953 else if (recursive && item->IsSizer())
954 {
9f13661f
WS
955 wxSizerItem *subitem = item->GetSizer()->GetItem( sizer, true );
956 if (subitem)
957 return subitem;
8b2bac62
WS
958 }
959
12a3f227 960 node = node->GetNext();
2b5f62a0 961 }
8b2bac62 962
9f13661f
WS
963 return NULL;
964}
965
966wxSizerItem* wxSizer::GetItem( size_t index )
967{
968 wxCHECK_MSG( index < m_children.GetCount(),
969 NULL,
970 _T("GetItem index is out of range") );
971
972 return m_children.Item( index )->GetData();
973}
974
975bool wxSizer::Show( wxWindow *window, bool show, bool recursive )
976{
977 wxSizerItem *item = GetItem( window, recursive );
978
979 if ( item )
980 {
981 item->Show( show );
982 return true;
983 }
984
985 return false;
986}
987
988bool wxSizer::Show( wxSizer *sizer, bool show, bool recursive )
989{
990 wxSizerItem *item = GetItem( sizer, recursive );
991
992 if ( item )
993 {
994 item->Show( show );
995 return true;
996 }
997
8b2bac62 998 return false;
2b5f62a0
VZ
999}
1000
8b2bac62 1001bool wxSizer::Show( size_t index, bool show)
2b5f62a0 1002{
9f13661f 1003 wxSizerItem *item = GetItem( index );
2b5f62a0 1004
9f13661f
WS
1005 if ( item )
1006 {
1007 item->Show( show );
1008 return true;
1009 }
8b2bac62 1010
9f13661f 1011 return false;
12a3f227 1012}
2b5f62a0 1013
12a3f227
RL
1014void wxSizer::ShowItems( bool show )
1015{
222ed1d6 1016 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
12a3f227
RL
1017 while (node)
1018 {
1019 node->GetData()->Show( show );
1020 node = node->GetNext();
2b5f62a0
VZ
1021 }
1022}
1023
9cbee2ce 1024bool wxSizer::IsShown( wxWindow *window ) const
2b5f62a0 1025{
222ed1d6 1026 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
2b5f62a0
VZ
1027 while (node)
1028 {
12a3f227 1029 wxSizerItem *item = node->GetData();
dc259b79 1030
12a3f227 1031 if (item->GetWindow() == window)
2b5f62a0
VZ
1032 {
1033 return item->IsShown();
1034 }
12a3f227 1035 node = node->GetNext();
2b5f62a0
VZ
1036 }
1037
12a3f227
RL
1038 wxFAIL_MSG( _T("IsShown failed to find sizer item") );
1039
e0d8fb45 1040 return false;
2b5f62a0
VZ
1041}
1042
9cbee2ce 1043bool wxSizer::IsShown( wxSizer *sizer ) const
2b5f62a0 1044{
222ed1d6 1045 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
2b5f62a0
VZ
1046 while (node)
1047 {
12a3f227 1048 wxSizerItem *item = node->GetData();
2b5f62a0 1049
12a3f227 1050 if (item->GetSizer() == sizer)
2b5f62a0
VZ
1051 {
1052 return item->IsShown();
1053 }
12a3f227 1054 node = node->GetNext();
2b5f62a0
VZ
1055 }
1056
12a3f227
RL
1057 wxFAIL_MSG( _T("IsShown failed to find sizer item") );
1058
e0d8fb45 1059 return false;
2b5f62a0
VZ
1060}
1061
9cbee2ce 1062bool wxSizer::IsShown( size_t index ) const
12a3f227
RL
1063{
1064 wxCHECK_MSG( index < m_children.GetCount(),
e0d8fb45 1065 false,
12a3f227
RL
1066 _T("IsShown index is out of range") );
1067
1068 return m_children.Item( index )->GetData()->IsShown();
1069}
1070
1071
f6bcfd97
BP
1072//---------------------------------------------------------------------------
1073// wxGridSizer
1074//---------------------------------------------------------------------------
1075
1076wxGridSizer::wxGridSizer( int rows, int cols, int vgap, int hgap )
902725ee 1077 : m_rows( ( cols == 0 && rows == 0 ) ? 1 : rows )
12a3f227
RL
1078 , m_cols( cols )
1079 , m_vgap( vgap )
1080 , m_hgap( hgap )
f6bcfd97 1081{
f6bcfd97
BP
1082}
1083
1084wxGridSizer::wxGridSizer( int cols, int vgap, int hgap )
902725ee 1085 : m_rows( cols == 0 ? 1 : 0 )
12a3f227
RL
1086 , m_cols( cols )
1087 , m_vgap( vgap )
1088 , m_hgap( hgap )
f6bcfd97 1089{
f6bcfd97
BP
1090}
1091
0ca5105b 1092int wxGridSizer::CalcRowsCols(int& nrows, int& ncols) const
f6bcfd97 1093{
f6bcfd97 1094 int nitems = m_children.GetCount();
2b5f62a0 1095 if ( nitems)
0ca5105b
VZ
1096 {
1097 if ( m_cols )
1098 {
1099 ncols = m_cols;
1100 nrows = (nitems + m_cols - 1) / m_cols;
1101 }
1102 else if ( m_rows )
1103 {
1104 ncols = (nitems + m_rows - 1) / m_rows;
1105 nrows = m_rows;
1106 }
1107 else // 0 columns, 0 rows?
1108 {
1109 wxFAIL_MSG( _T("grid sizer must have either rows or columns fixed") );
f6bcfd97 1110
0ca5105b
VZ
1111 nrows = ncols = 0;
1112 }
1113 }
1114
1115 return nitems;
1116}
1117
1118void wxGridSizer::RecalcSizes()
1119{
1120 int nitems, nrows, ncols;
1121 if ( (nitems = CalcRowsCols(nrows, ncols)) == 0 )
1122 return;
f6bcfd97
BP
1123
1124 wxSize sz( GetSize() );
1125 wxPoint pt( GetPosition() );
3ca6a5f0
BP
1126
1127 int w = (sz.x - (ncols - 1) * m_hgap) / ncols;
1128 int h = (sz.y - (nrows - 1) * m_vgap) / nrows;
f6bcfd97
BP
1129
1130 int x = pt.x;
1131 for (int c = 0; c < ncols; c++)
1132 {
1133 int y = pt.y;
1134 for (int r = 0; r < nrows; r++)
1135 {
1136 int i = r * ncols + c;
1137 if (i < nitems)
1138 {
222ed1d6 1139 wxSizerItemList::compatibility_iterator node = m_children.Item( i );
12a3f227
RL
1140
1141 wxASSERT_MSG( node, _T("Failed to find SizerItemList node") );
3ca6a5f0 1142
12a3f227 1143 SetItemBounds( node->GetData(), x, y, w, h);
f6bcfd97
BP
1144 }
1145 y = y + h + m_vgap;
1146 }
1147 x = x + w + m_hgap;
1148 }
1149}
1150
1151wxSize wxGridSizer::CalcMin()
1152{
196be0f1
JS
1153 int nrows, ncols;
1154 if ( CalcRowsCols(nrows, ncols) == 0 )
0ca5105b 1155 return wxSize(10, 10);
f6bcfd97 1156
4f469fb5 1157 // Find the max width and height for any component
f6bcfd97
BP
1158 int w = 0;
1159 int h = 0;
3ca6a5f0 1160
222ed1d6 1161 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
f6bcfd97
BP
1162 while (node)
1163 {
12a3f227
RL
1164 wxSizerItem *item = node->GetData();
1165 wxSize sz( item->CalcMin() );
1166
f6bcfd97
BP
1167 w = wxMax( w, sz.x );
1168 h = wxMax( h, sz.y );
3ca6a5f0 1169
12a3f227 1170 node = node->GetNext();
f6bcfd97 1171 }
3ca6a5f0 1172
12a3f227
RL
1173 return wxSize( ncols * w + (ncols-1) * m_hgap,
1174 nrows * h + (nrows-1) * m_vgap );
f6bcfd97
BP
1175}
1176
1177void wxGridSizer::SetItemBounds( wxSizerItem *item, int x, int y, int w, int h )
1178{
1179 wxPoint pt( x,y );
8b2bac62 1180 wxSize sz( item->GetMinSizeWithBorder() );
f6bcfd97
BP
1181 int flag = item->GetFlag();
1182
1183 if ((flag & wxEXPAND) || (flag & wxSHAPED))
1184 {
1185 sz = wxSize(w, h);
1186 }
1187 else
1188 {
1189 if (flag & wxALIGN_CENTER_HORIZONTAL)
1190 {
559b747d 1191 pt.x = x + (w - sz.x) / 2;
f6bcfd97
BP
1192 }
1193 else if (flag & wxALIGN_RIGHT)
1194 {
559b747d 1195 pt.x = x + (w - sz.x);
f6bcfd97 1196 }
3ca6a5f0 1197
f6bcfd97
BP
1198 if (flag & wxALIGN_CENTER_VERTICAL)
1199 {
559b747d 1200 pt.y = y + (h - sz.y) / 2;
f6bcfd97
BP
1201 }
1202 else if (flag & wxALIGN_BOTTOM)
1203 {
559b747d 1204 pt.y = y + (h - sz.y);
f6bcfd97
BP
1205 }
1206 }
3ca6a5f0 1207
f6bcfd97
BP
1208 item->SetDimension(pt, sz);
1209}
1210
1211//---------------------------------------------------------------------------
1212// wxFlexGridSizer
1213//---------------------------------------------------------------------------
1214
1215wxFlexGridSizer::wxFlexGridSizer( int rows, int cols, int vgap, int hgap )
5d76f462
VZ
1216 : wxGridSizer( rows, cols, vgap, hgap ),
1217 m_flexDirection(wxBOTH),
1218 m_growMode(wxFLEX_GROWMODE_SPECIFIED)
3ca6a5f0 1219{
f6bcfd97
BP
1220}
1221
1222wxFlexGridSizer::wxFlexGridSizer( int cols, int vgap, int hgap )
5d76f462
VZ
1223 : wxGridSizer( cols, vgap, hgap ),
1224 m_flexDirection(wxBOTH),
1225 m_growMode(wxFLEX_GROWMODE_SPECIFIED)
3ca6a5f0 1226{
f6bcfd97 1227}
3ca6a5f0 1228
f6bcfd97
BP
1229wxFlexGridSizer::~wxFlexGridSizer()
1230{
f6bcfd97
BP
1231}
1232
1233void wxFlexGridSizer::RecalcSizes()
1234{
0ca5105b
VZ
1235 int nitems, nrows, ncols;
1236 if ( (nitems = CalcRowsCols(nrows, ncols)) == 0 )
f6bcfd97
BP
1237 return;
1238
20b35a69 1239 wxPoint pt( GetPosition() );
f6bcfd97 1240 wxSize sz( GetSize() );
5d76f462 1241
ba763a45 1242 AdjustForGrowables(sz, m_calculatedMinSize, nrows, ncols);
3ca6a5f0 1243
f6bcfd97
BP
1244 sz = wxSize( pt.x + sz.x, pt.y + sz.y );
1245
1246 int x = pt.x;
1247 for (int c = 0; c < ncols; c++)
1248 {
1249 int y = pt.y;
1250 for (int r = 0; r < nrows; r++)
1251 {
1252 int i = r * ncols + c;
1253 if (i < nitems)
1254 {
222ed1d6 1255 wxSizerItemList::compatibility_iterator node = m_children.Item( i );
12a3f227
RL
1256
1257 wxASSERT_MSG( node, _T("Failed to find node") );
3ca6a5f0 1258
f6bcfd97
BP
1259 int w = wxMax( 0, wxMin( m_colWidths[c], sz.x - x ) );
1260 int h = wxMax( 0, wxMin( m_rowHeights[r], sz.y - y ) );
3ca6a5f0 1261
12a3f227 1262 SetItemBounds( node->GetData(), x, y, w, h);
f6bcfd97 1263 }
53701799
RD
1264 if (m_rowHeights[r] != -1)
1265 y = y + m_rowHeights[r] + m_vgap;
f6bcfd97 1266 }
53701799
RD
1267 if (m_colWidths[c] != -1)
1268 x = x + m_colWidths[c] + m_hgap;
f6bcfd97
BP
1269 }
1270}
1271
1272wxSize wxFlexGridSizer::CalcMin()
1273{
150c8d89
RL
1274 int nrows,
1275 ncols;
1276 size_t i, s;
1277
55f9f0cb 1278 // Number of rows/columns can change as items are added or removed.
5d76f462
VZ
1279 if ( !CalcRowsCols(nrows, ncols) )
1280 return wxSize(10, 10);
f6bcfd97 1281
5d76f462
VZ
1282 m_rowHeights.SetCount(nrows);
1283 m_colWidths.SetCount(ncols);
3ca6a5f0 1284
395a82b1
VZ
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.
55f9f0cb
VZ
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;
1294
222ed1d6 1295 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
12a3f227 1296
150c8d89 1297 i = 0;
f6bcfd97
BP
1298 while (node)
1299 {
12a3f227 1300 wxSizerItem *item = node->GetData();
55f9f0cb
VZ
1301 if ( item->IsShown() )
1302 {
1303 wxSize sz( item->CalcMin() );
1304 int row = i / ncols;
1305 int col = i % ncols;
12a3f227 1306
55f9f0cb
VZ
1307 m_rowHeights[ row ] = wxMax( wxMax( 0, sz.y ), m_rowHeights[ row ] );
1308 m_colWidths[ col ] = wxMax( wxMax( 0, sz.x ), m_colWidths[ col ] );
1309 }
3ca6a5f0 1310
12a3f227 1311 node = node->GetNext();
f6bcfd97
BP
1312 i++;
1313 }
3ca6a5f0 1314
20b35a69 1315 AdjustForFlexDirection();
8b2bac62 1316
20b35a69
RD
1317 // Sum total minimum size, including gaps between rows/columns.
1318 // -1 is used as a magic number meaning empty column.
1319 int width = 0;
1320 for (int col = 0; col < ncols; col++)
1321 if ( m_colWidths[ col ] != -1 )
53701799
RD
1322 width += m_colWidths[ col ] + m_hgap;
1323 if (width > 0)
1324 width -= m_hgap;
20b35a69
RD
1325
1326 int height = 0;
1327 for (int row = 0; row < nrows; row++)
1328 if ( m_rowHeights[ row ] != -1 )
53701799
RD
1329 height += m_rowHeights[ row ] + m_vgap;
1330 if (height > 0)
1331 height -= m_vgap;
20b35a69 1332
ba763a45
RD
1333 m_calculatedMinSize = wxSize( width, height );
1334 return m_calculatedMinSize;
20b35a69
RD
1335}
1336
1337void wxFlexGridSizer::AdjustForFlexDirection()
1338{
1339 // the logic in CalcMin works when we resize flexibly in both directions
1340 // but maybe this is not the case
5d76f462
VZ
1341 if ( m_flexDirection != wxBOTH )
1342 {
1343 // select the array corresponding to the direction in which we do *not*
1344 // resize flexibly
1345 wxArrayInt& array = m_flexDirection == wxVERTICAL ? m_colWidths
1346 : m_rowHeights;
1347
4a10ea8b 1348 const size_t count = array.GetCount();
5d76f462
VZ
1349
1350 // find the largest value in this array
4a10ea8b
MW
1351 size_t n;
1352 int largest = 0;
1353
5d76f462
VZ
1354 for ( n = 0; n < count; ++n )
1355 {
1356 if ( array[n] > largest )
1357 largest = array[n];
1358 }
1359
1360 // and now fill it with the largest value
1361 for ( n = 0; n < count; ++n )
1362 {
1363 array[n] = largest;
1364 }
1365 }
8b2bac62 1366}
5d76f462 1367
3ca6a5f0 1368
20b35a69
RD
1369void wxFlexGridSizer::AdjustForGrowables(const wxSize& sz, const wxSize& minsz,
1370 int nrows, int ncols)
1371{
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) ) )
1374 {
1375 int sum_proportions = 0;
1376 int growable_space = 0;
1377 int num = 0;
1378 size_t idx;
1379 for (idx = 0; idx < m_growableRows.GetCount(); idx++)
1380 {
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)
1385 continue;
8b2bac62 1386
20b35a69
RD
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)
1391 continue;
1392 sum_proportions += m_growableRowsProportions[idx];
1393 growable_space += m_rowHeights[ m_growableRows[idx] ];
1394 num++;
1395 }
3ca6a5f0 1396
20b35a69
RD
1397 if (num > 0)
1398 {
1399 for (idx = 0; idx < m_growableRows.GetCount(); idx++)
1400 {
1401 if (m_growableRows[idx] >= nrows )
1402 continue;
1403 if (m_rowHeights[ m_growableRows[idx] ] == -1)
1404 m_rowHeights[ m_growableRows[idx] ] = 0;
1405 else
1406 {
1407 int delta = (sz.y - minsz.y);
1408 if (sum_proportions == 0)
1409 delta = (delta/num) + m_rowHeights[ m_growableRows[idx] ];
1410 else
1411 delta = ((delta+growable_space)*m_growableRowsProportions[idx]) / sum_proportions;
1412 m_rowHeights[ m_growableRows[idx] ] = delta;
1413 }
1414 }
1415 }
1416 }
1417 else if ( (m_growMode == wxFLEX_GROWMODE_ALL) && (sz.y > minsz.y) )
1418 {
1419 // rounding problem?
1420 for ( int row = 0; row < nrows; ++row )
1421 m_rowHeights[ row ] = sz.y / nrows;
1422 }
1423
1424 // the same logic as above but for the columns
1425 if ( sz.x > minsz.x && ( (m_flexDirection & wxHORIZONTAL) || (m_growMode == wxFLEX_GROWMODE_SPECIFIED) ) )
1426 {
1427 int sum_proportions = 0;
1428 int growable_space = 0;
1429 int num = 0;
1430 size_t idx;
1431 for (idx = 0; idx < m_growableCols.GetCount(); idx++)
1432 {
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)
1437 continue;
8b2bac62 1438
20b35a69
RD
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
1441 // completely.
1442 if (m_colWidths[ m_growableCols[idx] ] == -1)
1443 continue;
1444 sum_proportions += m_growableColsProportions[idx];
1445 growable_space += m_colWidths[ m_growableCols[idx] ];
1446 num++;
1447 }
1448
1449 if (num > 0)
1450 {
1451 for (idx = 0; idx < m_growableCols.GetCount(); idx++)
1452 {
1453 if (m_growableCols[idx] >= ncols )
1454 continue;
1455 if (m_colWidths[ m_growableCols[idx] ] == -1)
1456 m_colWidths[ m_growableCols[idx] ] = 0;
1457 else
1458 {
1459 int delta = (sz.x - minsz.x);
1460 if (sum_proportions == 0)
1461 delta = (delta/num) + m_colWidths[ m_growableCols[idx] ];
1462 else
1463 delta = ((delta+growable_space)*m_growableColsProportions[idx])/sum_proportions;
1464 m_colWidths[ m_growableCols[idx] ] = delta;
1465 }
1466 }
1467 }
1468 }
1469 else if ( (m_growMode == wxFLEX_GROWMODE_ALL) && (sz.x > minsz.x) )
1470 {
1471 for ( int col=0; col < ncols; ++col )
1472 m_colWidths[ col ] = sz.x / ncols;
1473 }
f6bcfd97
BP
1474}
1475
20b35a69 1476
e8800dcf 1477void wxFlexGridSizer::AddGrowableRow( size_t idx, int proportion )
f6bcfd97
BP
1478{
1479 m_growableRows.Add( idx );
e8800dcf 1480 m_growableRowsProportions.Add( proportion );
f6bcfd97
BP
1481}
1482
8d2474f4 1483void wxFlexGridSizer::RemoveGrowableRow( size_t idx )
f6bcfd97 1484{
8d2474f4 1485 m_growableRows.Remove( idx );
f6bcfd97
BP
1486}
1487
e8800dcf 1488void wxFlexGridSizer::AddGrowableCol( size_t idx, int proportion )
f6bcfd97
BP
1489{
1490 m_growableCols.Add( idx );
e8800dcf 1491 m_growableColsProportions.Add( proportion );
f6bcfd97
BP
1492}
1493
8d2474f4 1494void wxFlexGridSizer::RemoveGrowableCol( size_t idx )
f6bcfd97 1495{
8d2474f4 1496 m_growableCols.Remove( idx );
f6bcfd97
BP
1497}
1498
c62ac5b6 1499//---------------------------------------------------------------------------
92afa2b1 1500// wxBoxSizer
61d514bb
RR
1501//---------------------------------------------------------------------------
1502
92afa2b1 1503wxBoxSizer::wxBoxSizer( int orient )
12a3f227 1504 : m_orient( orient )
61d514bb 1505{
61d514bb
RR
1506}
1507
92afa2b1 1508void wxBoxSizer::RecalcSizes()
61d514bb
RR
1509{
1510 if (m_children.GetCount() == 0)
61d514bb 1511 return;
0c0d686f 1512
61d514bb 1513 int delta = 0;
61d514bb
RR
1514 if (m_stretchable)
1515 {
1516 if (m_orient == wxHORIZONTAL)
85e5cfc9 1517 delta = m_size.x - m_fixedWidth;
3ca6a5f0 1518 else
85e5cfc9 1519 delta = m_size.y - m_fixedHeight;
61d514bb 1520 }
0c0d686f 1521
61d514bb 1522 wxPoint pt( m_position );
0c0d686f 1523
222ed1d6 1524 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
61d514bb
RR
1525 while (node)
1526 {
12a3f227
RL
1527 wxSizerItem *item = node->GetData();
1528
2b5f62a0 1529 if (item->IsShown())
3ca6a5f0 1530 {
ba763a45 1531 wxSize size( item->GetMinSizeWithBorder() );
3ca6a5f0 1532
2b5f62a0 1533 if (m_orient == wxVERTICAL)
3ca6a5f0 1534 {
2b5f62a0 1535 wxCoord height = size.y;
12a3f227 1536 if (item->GetProportion())
2b5f62a0 1537 {
85e5cfc9
VZ
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;
2b5f62a0
VZ
1541 }
1542
1543 wxPoint child_pos( pt );
42841dfc 1544 wxSize child_size( size.x, height );
2b5f62a0
VZ
1545
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;
1554
1555 item->SetDimension( child_pos, child_size );
1556
1557 pt.y += height;
1558 }
1559 else
1560 {
1561 wxCoord width = size.x;
12a3f227 1562 if (item->GetProportion())
2b5f62a0 1563 {
85e5cfc9
VZ
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;
2b5f62a0
VZ
1567 }
1568
1569 wxPoint child_pos( pt );
42841dfc 1570 wxSize child_size( width, size.y );
2b5f62a0
VZ
1571
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;
1580
1581 item->SetDimension( child_pos, child_size );
1582
1583 pt.x += width;
3ca6a5f0 1584 }
3ca6a5f0
BP
1585 }
1586
12a3f227 1587 node = node->GetNext();
61d514bb
RR
1588 }
1589}
1590
92afa2b1 1591wxSize wxBoxSizer::CalcMin()
61d514bb
RR
1592{
1593 if (m_children.GetCount() == 0)
c7a9fa36 1594 return wxSize(10,10);
0c0d686f 1595
61d514bb
RR
1596 m_stretchable = 0;
1597 m_minWidth = 0;
1598 m_minHeight = 0;
1599 m_fixedWidth = 0;
1600 m_fixedHeight = 0;
0c0d686f 1601
ba763a45 1602 // precalc item minsizes and count proportions
222ed1d6 1603 wxSizerItemList::compatibility_iterator node = m_children.GetFirst();
85e5cfc9
VZ
1604 while (node)
1605 {
1606 wxSizerItem *item = node->GetData();
12a3f227 1607
d4444410
VZ
1608 if ( item->IsShown() )
1609 {
ba763a45 1610 item->CalcMin(); // result is stored in the item
8b2bac62 1611
85e5cfc9 1612 m_stretchable += item->GetProportion();
d4444410 1613 }
85e5cfc9
VZ
1614
1615 node = node->GetNext();
1616 }
1617
1618 // Total minimum size (width or height) of sizer
1619 int maxMinSize = 0;
1620
1621 node = m_children.GetFirst();
61d514bb 1622 while (node)
f98de448 1623 {
85e5cfc9 1624 wxSizerItem *item = node->GetData();
12a3f227
RL
1625
1626 if (item->IsShown() && item->GetProportion() != 0)
f98de448 1627 {
12a3f227 1628 int stretch = item->GetProportion();
ba763a45 1629 wxSize size( item->GetMinSizeWithBorder() );
85e5cfc9 1630 int minSize;
8b2bac62 1631
f98de448 1632 // Integer division rounded up is (a + b - 1) / b
85e5cfc9
VZ
1633 // Round up needed in order to guarantee that all
1634 // all items will have size not less then their min size
f98de448 1635 if (m_orient == wxHORIZONTAL)
85e5cfc9 1636 minSize = ( size.x*m_stretchable + stretch - 1)/stretch;
f98de448 1637 else
85e5cfc9 1638 minSize = ( size.y*m_stretchable + stretch - 1)/stretch;
8b2bac62 1639
85e5cfc9
VZ
1640 if (minSize > maxMinSize)
1641 maxMinSize = minSize;
f98de448 1642 }
12a3f227 1643 node = node->GetNext();
f98de448 1644 }
12a3f227 1645
4f469fb5
RR
1646 // Calculate overall minimum size
1647 node = m_children.GetFirst();
f98de448 1648 while (node)
61d514bb 1649 {
85e5cfc9 1650 wxSizerItem *item = node->GetData();
12a3f227 1651
2b5f62a0 1652 if (item->IsShown())
f98de448 1653 {
ba763a45 1654 wxSize size( item->GetMinSizeWithBorder() );
12a3f227 1655 if (item->GetProportion() != 0)
2b5f62a0
VZ
1656 {
1657 if (m_orient == wxHORIZONTAL)
85e5cfc9 1658 size.x = (maxMinSize*item->GetProportion())/m_stretchable;
2b5f62a0 1659 else
85e5cfc9 1660 size.y = (maxMinSize*item->GetProportion())/m_stretchable;
3ca6a5f0
BP
1661 }
1662 else
2b5f62a0
VZ
1663 {
1664 if (m_orient == wxVERTICAL)
1665 {
1666 m_fixedHeight += size.y;
1667 m_fixedWidth = wxMax( m_fixedWidth, size.x );
1668 }
1669 else
1670 {
1671 m_fixedWidth += size.x;
1672 m_fixedHeight = wxMax( m_fixedHeight, size.y );
1673 }
1674 }
85e5cfc9
VZ
1675
1676 if (m_orient == wxHORIZONTAL)
1677 {
1678 m_minWidth += size.x;
1679 m_minHeight = wxMax( m_minHeight, size.y );
1680 }
1681 else
1682 {
1683 m_minHeight += size.y;
1684 m_minWidth = wxMax( m_minWidth, size.x );
1685 }
2b5f62a0 1686 }
12a3f227 1687 node = node->GetNext();
61d514bb 1688 }
0c0d686f 1689
61d514bb
RR
1690 return wxSize( m_minWidth, m_minHeight );
1691}
27ea1d8a
RR
1692
1693//---------------------------------------------------------------------------
1694// wxStaticBoxSizer
1695//---------------------------------------------------------------------------
1696
1e6feb95
VZ
1697#if wxUSE_STATBOX
1698
27ea1d8a 1699wxStaticBoxSizer::wxStaticBoxSizer( wxStaticBox *box, int orient )
e978011a
VZ
1700 : wxBoxSizer( orient ),
1701 m_staticBox( box )
27ea1d8a 1702{
223d09f6 1703 wxASSERT_MSG( box, wxT("wxStaticBoxSizer needs a static box") );
e978011a
VZ
1704
1705 // do this so that our Detach() is called if the static box is destroyed
1706 // before we are
1707 m_staticBox->SetContainingSizer(this);
27ea1d8a 1708}
0c0d686f 1709
6c1635b5
VZ
1710wxStaticBoxSizer::wxStaticBoxSizer(int orient, wxWindow *win, const wxString& s)
1711 : wxBoxSizer(orient),
1712 m_staticBox(new wxStaticBox(win, wxID_ANY, s))
1713{
e978011a
VZ
1714 // same as above
1715 m_staticBox->SetContainingSizer(this);
6c1635b5
VZ
1716}
1717
12a3f227
RL
1718static void GetStaticBoxBorders( wxStaticBox *box,
1719 int *borderTop,
1720 int *borderOther)
84028727
VZ
1721{
1722 // this has to be done platform by platform as there is no way to
1723 // guess the thickness of a wxStaticBox border
5dd070c2 1724 box->GetBordersForSizer(borderTop, borderOther);
84028727
VZ
1725}
1726
27ea1d8a
RR
1727void wxStaticBoxSizer::RecalcSizes()
1728{
84028727
VZ
1729 int top_border, other_border;
1730 GetStaticBoxBorders(m_staticBox, &top_border, &other_border);
27ea1d8a
RR
1731
1732 m_staticBox->SetSize( m_position.x, m_position.y, m_size.x, m_size.y );
0c0d686f 1733
27ea1d8a
RR
1734 wxPoint old_pos( m_position );
1735 m_position.x += other_border;
1736 m_position.y += top_border;
1737 wxSize old_size( m_size );
1738 m_size.x -= 2*other_border;
1739 m_size.y -= top_border + other_border;
0c0d686f 1740
27ea1d8a 1741 wxBoxSizer::RecalcSizes();
0c0d686f 1742
27ea1d8a
RR
1743 m_position = old_pos;
1744 m_size = old_size;
1745}
1746
1747wxSize wxStaticBoxSizer::CalcMin()
1748{
84028727
VZ
1749 int top_border, other_border;
1750 GetStaticBoxBorders(m_staticBox, &top_border, &other_border);
0c0d686f 1751
27ea1d8a 1752 wxSize ret( wxBoxSizer::CalcMin() );
cae31b8b 1753 ret.x += 2*other_border;
27ea1d8a 1754 ret.y += other_border + top_border;
0c0d686f 1755
27ea1d8a
RR
1756 return ret;
1757}
83edc0a5 1758
eb2a7883
VZ
1759void wxStaticBoxSizer::ShowItems( bool show )
1760{
1761 m_staticBox->Show( show );
1762 wxBoxSizer::ShowItems( show );
1763}
1764
e978011a
VZ
1765bool wxStaticBoxSizer::Detach( wxWindow *window )
1766{
1767 // avoid deleting m_staticBox in our dtor if it's being detached from the
1768 // sizer (which can happen because it's being already destroyed for
1769 // example)
1770 if ( window == m_staticBox )
1771 {
1772 m_staticBox = NULL;
1773 return true;
1774 }
1775
1776 return wxSizer::Detach( window );
1777}
1778
1e6feb95
VZ
1779#endif // wxUSE_STATBOX
1780
974c2a59
WS
1781#if wxUSE_BUTTON
1782
acf2ac37
RR
1783wxStdDialogButtonSizer::wxStdDialogButtonSizer()
1784 : wxBoxSizer(wxHORIZONTAL)
1785{
94f53923
JS
1786 // Vertical buttons with lots of space on either side
1787 // looks rubbish on WinCE, so let's not do this for now.
1788 // If we are going to use vertical buttons, we should
1789 // put the sizer to the right of other controls in the dialog,
1790 // and that's beyond the scope of this sizer.
1791#ifndef __WXWINCE__
acf2ac37 1792 bool is_pda = (wxSystemSettings::GetScreenType() <= wxSYS_SCREEN_PDA);
974c2a59 1793 // If we have a PDA screen, put yes/no button over
acf2ac37
RR
1794 // all other buttons, otherwise on the left side.
1795 if (is_pda)
1796 m_orient = wxVERTICAL;
94f53923 1797#endif
974c2a59 1798
acf2ac37
RR
1799 m_buttonAffirmative = NULL;
1800 m_buttonApply = NULL;
1801 m_buttonNegative = NULL;
1802 m_buttonCancel = NULL;
1803 m_buttonHelp = NULL;
1804}
1805
1806void wxStdDialogButtonSizer::AddButton(wxButton *mybutton)
1807{
1808 switch (mybutton->GetId())
1809 {
1810 case wxID_OK:
1811 case wxID_YES:
1812 case wxID_SAVE:
1813 m_buttonAffirmative = mybutton;
1814 break;
1815 case wxID_APPLY:
1816 m_buttonApply = mybutton;
1817 break;
1818 case wxID_NO:
1819 m_buttonNegative = mybutton;
1820 break;
1821 case wxID_CANCEL:
1822 m_buttonCancel = mybutton;
1823 break;
1824 case wxID_HELP:
2997ca30 1825 case wxID_CONTEXT_HELP:
acf2ac37
RR
1826 m_buttonHelp = mybutton;
1827 break;
1828 default:
1829 break;
1830 }
1831}
1832
b181a505
RR
1833void wxStdDialogButtonSizer::SetAffirmativeButton( wxButton *button )
1834{
1835 m_buttonAffirmative = button;
1836}
1837
1838void wxStdDialogButtonSizer::SetNegativeButton( wxButton *button )
1839{
1840 m_buttonNegative = button;
1841}
1842
1843void wxStdDialogButtonSizer::SetCancelButton( wxButton *button )
1844{
1845 m_buttonCancel = button;
1846}
1847
718903fe 1848void wxStdDialogButtonSizer::Realize()
acf2ac37
RR
1849{
1850#ifdef __WXMAC__
1851 Add(0, 0, 0, wxLEFT, 6);
1852 if (m_buttonHelp)
974c2a59
WS
1853 Add((wxWindow*)m_buttonHelp, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, 6);
1854
acf2ac37
RR
1855 if (m_buttonNegative){
1856 // HIG POLICE BULLETIN - destructive buttons need extra padding
1857 // 24 pixels on either side
1858 Add((wxWindow*)m_buttonNegative, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, 12);
1859 }
974c2a59 1860
acf2ac37 1861 // extra whitespace between help/negative and cancel/ok buttons
974c2a59
WS
1862 Add(0, 0, 1, wxEXPAND, 0);
1863
acf2ac37
RR
1864 if (m_buttonCancel){
1865 Add((wxWindow*)m_buttonCancel, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, 6);
1866 // Cancel or help should be default
1867 // m_buttonCancel->SetDefaultButton();
1868 }
974c2a59
WS
1869
1870 // Ugh, Mac doesn't really have apply dialogs, so I'll just
acf2ac37
RR
1871 // figure the best place is between Cancel and OK
1872 if (m_buttonApply)
1873 Add((wxWindow*)m_buttonApply, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, 6);
974c2a59 1874
acf2ac37
RR
1875 if (m_buttonAffirmative){
1876 Add((wxWindow*)m_buttonAffirmative, 0, wxALIGN_CENTRE | wxLEFT, 6);
974c2a59 1877
acf2ac37
RR
1878 if (m_buttonAffirmative->GetId() == wxID_SAVE){
1879 // these buttons have set labels under Mac so we should use them
1880 m_buttonAffirmative->SetLabel(_("Save"));
1881 m_buttonNegative->SetLabel(_("Don't Save"));
1882 }
1883 }
974c2a59 1884
acf2ac37
RR
1885 // Extra space around and at the right
1886 Add(12, 24);
1887#elif defined(__WXGTK20__)
1888 Add(0, 0, 0, wxLEFT, 9);
1889 if (m_buttonHelp)
974c2a59
WS
1890 Add((wxWindow*)m_buttonHelp, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, 3);
1891
acf2ac37 1892 // extra whitespace between help and cancel/ok buttons
974c2a59
WS
1893 Add(0, 0, 1, wxEXPAND, 0);
1894
acf2ac37
RR
1895 if (m_buttonNegative){
1896 Add((wxWindow*)m_buttonNegative, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, 3);
1897 }
974c2a59 1898
acf2ac37
RR
1899 if (m_buttonCancel){
1900 Add((wxWindow*)m_buttonCancel, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, 3);
1901 // Cancel or help should be default
1902 // m_buttonCancel->SetDefaultButton();
1903 }
974c2a59 1904
acf2ac37
RR
1905 if (m_buttonApply)
1906 Add((wxWindow*)m_buttonApply, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, 3);
974c2a59 1907
acf2ac37
RR
1908 if (m_buttonAffirmative)
1909 Add((wxWindow*)m_buttonAffirmative, 0, wxALIGN_CENTRE | wxLEFT, 6);
0f884515
JS
1910#elif defined(__WXMSW__)
1911 // Windows
1912
1913 // right-justify buttons
1914 Add(0, 0, 1, wxEXPAND, 0);
1915
1916 if (m_buttonAffirmative){
1917 Add((wxWindow*)m_buttonAffirmative, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, m_buttonAffirmative->ConvertDialogToPixels(wxSize(2, 0)).x);
1918 }
1919
1920 if (m_buttonNegative){
1921 Add((wxWindow*)m_buttonNegative, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, m_buttonNegative->ConvertDialogToPixels(wxSize(2, 0)).x);
1922 }
1923
1924 if (m_buttonCancel){
1925 Add((wxWindow*)m_buttonCancel, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, m_buttonCancel->ConvertDialogToPixels(wxSize(2, 0)).x);
1926 }
1927 if (m_buttonApply)
1928 Add((wxWindow*)m_buttonApply, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, m_buttonApply->ConvertDialogToPixels(wxSize(2, 0)).x);
1929
1930 if (m_buttonHelp)
1931 Add((wxWindow*)m_buttonHelp, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, m_buttonHelp->ConvertDialogToPixels(wxSize(2, 0)).x);
acf2ac37 1932#else
0f884515 1933 // GTK+1 and any other platform
902725ee 1934
23b1018f 1935 // Add(0, 0, 0, wxLEFT, 5); // Not sure what this was for but it unbalances the dialog
acf2ac37 1936 if (m_buttonHelp)
974c2a59
WS
1937 Add((wxWindow*)m_buttonHelp, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, m_buttonHelp->ConvertDialogToPixels(wxSize(4, 0)).x);
1938
acf2ac37 1939 // extra whitespace between help and cancel/ok buttons
974c2a59 1940 Add(0, 0, 1, wxEXPAND, 0);
acf2ac37
RR
1941
1942 if (m_buttonApply)
1943 Add((wxWindow*)m_buttonApply, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, m_buttonApply->ConvertDialogToPixels(wxSize(4, 0)).x);
974c2a59 1944
acf2ac37
RR
1945 if (m_buttonAffirmative){
1946 Add((wxWindow*)m_buttonAffirmative, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, m_buttonAffirmative->ConvertDialogToPixels(wxSize(4, 0)).x);
1947 }
974c2a59 1948
acf2ac37
RR
1949 if (m_buttonNegative){
1950 Add((wxWindow*)m_buttonNegative, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, m_buttonNegative->ConvertDialogToPixels(wxSize(4, 0)).x);
1951 }
974c2a59 1952
acf2ac37 1953 if (m_buttonCancel){
23b1018f 1954 Add((wxWindow*)m_buttonCancel, 0, wxALIGN_CENTRE | wxLEFT | wxRIGHT, m_buttonCancel->ConvertDialogToPixels(wxSize(4, 0)).x);
acf2ac37
RR
1955 // Cancel or help should be default
1956 // m_buttonCancel->SetDefaultButton();
1957 }
974c2a59 1958
acf2ac37
RR
1959#endif
1960}
adbf2d73 1961
974c2a59
WS
1962#endif // wxUSE_BUTTON
1963
adbf2d73
VS
1964#if WXWIN_COMPATIBILITY_2_4
1965
ade4eb65 1966// ----------------------------------------------------------------------------
83edc0a5 1967// wxNotebookSizer
ade4eb65 1968// ----------------------------------------------------------------------------
83edc0a5 1969
adbf2d73
VS
1970#if wxUSE_BOOKCTRL
1971IMPLEMENT_CLASS(wxBookCtrlSizer, wxSizer)
1972#if wxUSE_NOTEBOOK
1973IMPLEMENT_CLASS(wxNotebookSizer, wxBookCtrlSizer)
1974#endif // wxUSE_NOTEBOOK
1975#endif // wxUSE_BOOKCTRL
1976
ade4eb65 1977#if wxUSE_BOOKCTRL
60be2f47 1978
3abb199d
VZ
1979#if WXWIN_COMPATIBILITY_2_6
1980
61c083e7 1981wxBookCtrlSizer::wxBookCtrlSizer(wxBookCtrlBase *bookctrl)
ade4eb65 1982 : m_bookctrl(bookctrl)
83edc0a5 1983{
ade4eb65 1984 wxASSERT_MSG( bookctrl, wxT("wxBookCtrlSizer needs a control") );
83edc0a5
RR
1985}
1986
3abb199d
VZ
1987#endif // WXWIN_COMPATIBILITY_2_6
1988
ade4eb65 1989void wxBookCtrlSizer::RecalcSizes()
83edc0a5 1990{
ade4eb65 1991 m_bookctrl->SetSize( m_position.x, m_position.y, m_size.x, m_size.y );
83edc0a5
RR
1992}
1993
ade4eb65 1994wxSize wxBookCtrlSizer::CalcMin()
83edc0a5 1995{
c47addef 1996 wxSize sizeBorder = m_bookctrl->CalcSizeFromPage(wxSize(0,0));
1e6feb95
VZ
1997
1998 sizeBorder.x += 5;
1999 sizeBorder.y += 5;
3ca6a5f0 2000
ade4eb65 2001 if ( m_bookctrl->GetPageCount() == 0 )
1e6feb95
VZ
2002 {
2003 return wxSize(sizeBorder.x + 10, sizeBorder.y + 10);
2004 }
83edc0a5
RR
2005
2006 int maxX = 0;
2007 int maxY = 0;
2008
ade4eb65
VZ
2009 wxWindowList::compatibility_iterator
2010 node = m_bookctrl->GetChildren().GetFirst();
83edc0a5
RR
2011 while (node)
2012 {
2013 wxWindow *item = node->GetData();
3ca6a5f0
BP
2014 wxSizer *itemsizer = item->GetSizer();
2015
2016 if (itemsizer)
2017 {
83edc0a5 2018 wxSize subsize( itemsizer->CalcMin() );
83edc0a5 2019
1e6feb95
VZ
2020 if (subsize.x > maxX)
2021 maxX = subsize.x;
2022 if (subsize.y > maxY)
2023 maxY = subsize.y;
3ca6a5f0
BP
2024 }
2025
2026 node = node->GetNext();
83edc0a5
RR
2027 }
2028
1e6feb95 2029 return wxSize( maxX, maxY ) + sizeBorder;
83edc0a5
RR
2030}
2031
2d480e40
RD
2032#if wxUSE_NOTEBOOK
2033
f3190d1c
VZ
2034#if WXWIN_COMPATIBILITY_2_6
2035
2d480e40 2036wxNotebookSizer::wxNotebookSizer(wxNotebook *nb)
2d480e40 2037{
adbf2d73
VS
2038 wxASSERT_MSG( nb, wxT("wxNotebookSizer needs a control") );
2039 m_bookctrl = nb;
2d480e40
RD
2040}
2041
f3190d1c
VZ
2042#endif // WXWIN_COMPATIBILITY_2_6
2043
2d480e40 2044#endif // wxUSE_NOTEBOOOK
ade4eb65 2045#endif // wxUSE_BOOKCTRL
34c3ffca 2046
adbf2d73 2047#endif // WXWIN_COMPATIBILITY_2_4