]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/common/sizer.cpp
Fixed resize behaviour under certain circumstances.
[wxWidgets.git] / src / common / sizer.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: sizer.cpp
3// Purpose: provide new wxSizer class for layout
4// Author: Robert Roebling and Robin Dunn
5// Modified by:
6// Created:
7// RCS-ID: $Id$
8// Copyright: (c) Robin Dunn, Dirk Holtwick and Robert Roebling
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12#ifdef __GNUG__
13#pragma implementation "sizer.h"
14#endif
15
16// For compilers that support precompilation, includes "wx.h".
17#include "wx/wxprec.h"
18
19#ifdef __BORLANDC__
20 #pragma hdrstop
21#endif
22
23#include "wx/sizer.h"
24#include "wx/utils.h"
25#include "wx/statbox.h"
26#include "wx/notebook.h"
27
28//---------------------------------------------------------------------------
29
30IMPLEMENT_ABSTRACT_CLASS(wxSizerItem, wxObject);
31IMPLEMENT_ABSTRACT_CLASS(wxSizer, wxObject);
32IMPLEMENT_ABSTRACT_CLASS(wxBoxSizer, wxSizer);
33IMPLEMENT_ABSTRACT_CLASS(wxStaticBoxSizer, wxBoxSizer);
34#if wxUSE_NOTEBOOK
35IMPLEMENT_ABSTRACT_CLASS(wxNotebookSizer, wxSizer);
36#endif
37
38//---------------------------------------------------------------------------
39// wxSizerItem
40//---------------------------------------------------------------------------
41
42wxSizerItem::wxSizerItem( int width, int height, int option, int flag, int border, wxObject* userData )
43{
44 m_window = (wxWindow *) NULL;
45 m_sizer = (wxSizer *) NULL;
46 m_option = option;
47 m_border = border;
48 m_flag = flag;
49 m_userData = userData;
50
51 // minimal size is the initial size
52 m_minSize.x = width;
53 m_minSize.y = height;
54
55 SetRatio(width, height);
56
57 // size is set directly
58 m_size = m_minSize;
59}
60
61wxSizerItem::wxSizerItem( wxWindow *window, int option, int flag, int border, wxObject* userData )
62{
63 m_window = window;
64 m_sizer = (wxSizer *) NULL;
65 m_option = option;
66 m_border = border;
67 m_flag = flag;
68 m_userData = userData;
69
70 // minimal size is the initial size
71 m_minSize = window->GetSize();
72
73 // aspect ratio calculated from initial size
74 SetRatio(m_minSize);
75
76 // size is calculated later
77 // m_size = ...
78}
79
80wxSizerItem::wxSizerItem( wxSizer *sizer, int option, int flag, int border, wxObject* userData )
81{
82 m_window = (wxWindow *) NULL;
83 m_sizer = sizer;
84 m_option = option;
85 m_border = border;
86 m_flag = flag;
87 m_userData = userData;
88
89 // minimal size is calculated later
90 // m_minSize = ...
91 m_ratio = 0;
92
93 // size is calculated later
94 // m_size = ...
95}
96
97wxSizerItem::~wxSizerItem()
98{
99 if (m_userData)
100 delete m_userData;
101 if (m_sizer)
102 delete m_sizer;
103}
104
105
106wxSize wxSizerItem::GetSize()
107{
108 wxSize ret;
109 if (IsSizer())
110 ret = m_sizer->GetSize();
111 else
112 if (IsWindow())
113 ret = m_window->GetSize();
114 else ret = m_size;
115
116 if (m_flag & wxWEST)
117 ret.x += m_border;
118 if (m_flag & wxEAST)
119 ret.x += m_border;
120 if (m_flag & wxNORTH)
121 ret.y += m_border;
122 if (m_flag & wxSOUTH)
123 ret.y += m_border;
124
125 return ret;
126}
127
128wxSize wxSizerItem::CalcMin()
129{
130 wxSize ret;
131 if (IsSizer())
132 {
133 ret = m_sizer->CalcMin();
134 // if we have to preserve aspect ratio _AND_ this is
135 // the first-time calculation, consider ret to be initial size
136 if ((m_flag & wxSHAPED) && !m_ratio) SetRatio(ret);
137 }
138
139/*
140 The minimum size of a window should be the
141 initial size, as saved in m_minSize, not the
142 current size.
143
144 else
145 if (IsWindow())
146 ret = m_window->GetSize();
147*/
148 else ret = m_minSize;
149
150 if (m_flag & wxWEST)
151 ret.x += m_border;
152 if (m_flag & wxEAST)
153 ret.x += m_border;
154 if (m_flag & wxNORTH)
155 ret.y += m_border;
156 if (m_flag & wxSOUTH)
157 ret.y += m_border;
158
159 return ret;
160}
161
162void wxSizerItem::SetDimension( wxPoint pos, wxSize size )
163{
164 if (m_flag & wxWEST)
165 {
166 pos.x += m_border;
167 size.x -= m_border;
168 }
169 if (m_flag & wxEAST)
170 {
171 size.x -= m_border;
172 }
173 if (m_flag & wxNORTH)
174 {
175 pos.y += m_border;
176 size.y -= m_border;
177 }
178 if (m_flag & wxSOUTH)
179 {
180 size.y -= m_border;
181 }
182 if (m_flag & wxSHAPED) {
183 // adjust aspect ratio
184 int rwidth = (int) (size.y * m_ratio);
185 if (rwidth > size.x) {
186 // fit horizontally
187 int rheight = (int) (size.x / m_ratio);
188 // add vertical space
189 if (m_flag & wxALIGN_CENTER_VERTICAL)
190 pos.y += (size.y - rheight) / 2;
191 else if (m_flag & wxALIGN_BOTTOM)
192 pos.y += (size.y - rheight);
193 // use reduced dimensions
194 size.y =rheight;
195 } else if (rwidth < size.x) {
196 // add horizontal space
197 if (m_flag & wxALIGN_CENTER_HORIZONTAL)
198 pos.x += (size.x - rwidth) / 2;
199 else if (m_flag & wxALIGN_RIGHT)
200 pos.x += (size.x - rwidth);
201 size.x = rwidth;
202 }
203 }
204
205 if (IsSizer())
206 m_sizer->SetDimension( pos.x, pos.y, size.x, size.y );
207
208 if (IsWindow())
209 m_window->SetSize( pos.x, pos.y, size.x, size.y, wxSIZE_ALLOW_MINUS_ONE );
210
211 m_size = size;
212}
213
214bool wxSizerItem::IsWindow()
215{
216 return (m_window != NULL);
217}
218
219bool wxSizerItem::IsSizer()
220{
221 return (m_sizer != NULL);
222}
223
224bool wxSizerItem::IsSpacer()
225{
226 return (m_window == NULL) && (m_sizer == NULL);
227}
228
229//---------------------------------------------------------------------------
230// wxSizer
231//---------------------------------------------------------------------------
232
233wxSizer::wxSizer()
234{
235 m_children.DeleteContents( TRUE );
236}
237
238wxSizer::~wxSizer()
239{
240}
241
242void wxSizer::Add( wxWindow *window, int option, int flag, int border, wxObject* userData )
243{
244 m_children.Append( new wxSizerItem( window, option, flag, border, userData ) );
245}
246
247void wxSizer::Add( wxSizer *sizer, int option, int flag, int border, wxObject* userData )
248{
249 m_children.Append( new wxSizerItem( sizer, option, flag, border, userData ) );
250}
251
252void wxSizer::Add( int width, int height, int option, int flag, int border, wxObject* userData )
253{
254 m_children.Append( new wxSizerItem( width, height, option, flag, border, userData ) );
255}
256
257void wxSizer::Prepend( wxWindow *window, int option, int flag, int border, wxObject* userData )
258{
259 m_children.Insert( new wxSizerItem( window, option, flag, border, userData ) );
260}
261
262void wxSizer::Prepend( wxSizer *sizer, int option, int flag, int border, wxObject* userData )
263{
264 m_children.Insert( new wxSizerItem( sizer, option, flag, border, userData ) );
265}
266
267void wxSizer::Prepend( int width, int height, int option, int flag, int border, wxObject* userData )
268{
269 m_children.Insert( new wxSizerItem( width, height, option, flag, border, userData ) );
270}
271
272void wxSizer::Insert( int before, wxWindow *window, int option, int flag, int border, wxObject* userData )
273{
274 m_children.Insert( before, new wxSizerItem( window, option, flag, border, userData ) );
275}
276
277void wxSizer::Insert( int before, wxSizer *sizer, int option, int flag, int border, wxObject* userData )
278{
279 m_children.Insert( before, new wxSizerItem( sizer, option, flag, border, userData ) );
280}
281
282void wxSizer::Insert( int before, int width, int height, int option, int flag, int border, wxObject* userData )
283{
284 m_children.Insert( before, new wxSizerItem( width, height, option, flag, border, userData ) );
285}
286
287bool wxSizer::Remove( wxWindow *window )
288{
289 wxASSERT( window );
290
291 wxNode *node = m_children.First();
292 while (node)
293 {
294 wxSizerItem *item = (wxSizerItem*)node->Data();
295 if (item->GetWindow() == window)
296 {
297 m_children.DeleteNode( node );
298 return TRUE;
299 }
300 node = node->Next();
301 }
302
303 return FALSE;
304}
305
306bool wxSizer::Remove( wxSizer *sizer )
307{
308 wxASSERT( sizer );
309
310 wxNode *node = m_children.First();
311 while (node)
312 {
313 wxSizerItem *item = (wxSizerItem*)node->Data();
314 if (item->GetSizer() == sizer)
315 {
316 m_children.DeleteNode( node );
317 return TRUE;
318 }
319 node = node->Next();
320 }
321
322 return FALSE;
323}
324
325bool wxSizer::Remove( int pos )
326{
327 wxNode *node = m_children.Nth( pos );
328 if (!node) return FALSE;
329
330 m_children.DeleteNode( node );
331
332 return TRUE;
333}
334
335void wxSizer::Fit( wxWindow *window )
336{
337 window->SetSize( GetMinWindowSize( window ) );
338}
339
340void wxSizer::Layout()
341{
342 CalcMin();
343 RecalcSizes();
344}
345
346void wxSizer::SetSizeHints( wxWindow *window )
347{
348 wxSize size( GetMinWindowSize( window ) );
349 window->SetSizeHints( size.x, size.y );
350}
351
352wxSize wxSizer::GetMinWindowSize( wxWindow *window )
353{
354 wxSize minSize( GetMinSize() );
355 wxSize size( window->GetSize() );
356 wxSize client_size( window->GetClientSize() );
357 return wxSize( minSize.x+size.x-client_size.x,
358 minSize.y+size.y-client_size.y );
359}
360
361void wxSizer::SetDimension( int x, int y, int width, int height )
362{
363 m_position.x = x;
364 m_position.y = y;
365 m_size.x = width;
366 m_size.y = height;
367 CalcMin();
368 RecalcSizes();
369}
370
371//---------------------------------------------------------------------------
372// wxBoxSizer
373//---------------------------------------------------------------------------
374
375wxBoxSizer::wxBoxSizer( int orient )
376{
377 m_orient = orient;
378}
379
380void wxBoxSizer::RecalcSizes()
381{
382 if (m_children.GetCount() == 0)
383 return;
384
385 int delta = 0;
386 int extra = 0;
387 if (m_stretchable)
388 {
389 if (m_orient == wxHORIZONTAL)
390 {
391 delta = (m_size.x - m_fixedWidth) / m_stretchable;
392 extra = (m_size.x - m_fixedWidth) % m_stretchable;
393 }
394 else
395 {
396 delta = (m_size.y - m_fixedHeight) / m_stretchable;
397 extra = (m_size.y - m_fixedHeight) % m_stretchable;
398 }
399 }
400
401 wxPoint pt( m_position );
402
403 wxNode *node = m_children.GetFirst();
404 while (node)
405 {
406 wxSizerItem *item = (wxSizerItem*) node->Data();
407
408 int weight = 1;
409 if (item->GetOption())
410 weight = item->GetOption();
411
412 wxSize size( item->CalcMin() );
413
414 if (m_orient == wxVERTICAL)
415 {
416 wxCoord height = size.y;
417 if (item->GetOption())
418 {
419 height = (delta * weight) + extra;
420 extra = 0; // only the first item will get the remainder as extra size
421 }
422
423 wxPoint child_pos( pt );
424 wxSize child_size( wxSize( size.x, height) );
425
426 if (item->GetFlag() & (wxEXPAND | wxSHAPED))
427 child_size.x = m_size.x;
428 else if (item->GetFlag() & wxALIGN_RIGHT)
429 child_pos.x += m_size.x - size.x;
430 else if (item->GetFlag() & (wxCENTER | wxALIGN_CENTER_HORIZONTAL))
431 // XXX wxCENTER is added for backward compatibility;
432 // wxALIGN_CENTER should be used in new code
433 child_pos.x += (m_size.x - size.x) / 2;
434
435 item->SetDimension( child_pos, child_size );
436
437 pt.y += height;
438 }
439 else
440 {
441 wxCoord width = size.x;
442 if (item->GetOption())
443 {
444 width = (delta * weight) + extra;
445 extra = 0; // only the first item will get the remainder as extra size
446 }
447
448 wxPoint child_pos( pt );
449 wxSize child_size( wxSize(width, size.y) );
450
451 if (item->GetFlag() & (wxEXPAND | wxSHAPED))
452 child_size.y = m_size.y;
453 else if (item->GetFlag() & wxALIGN_BOTTOM)
454 child_pos.y += m_size.y - size.y;
455 else if (item->GetFlag() & (wxCENTER | wxALIGN_CENTER_VERTICAL))
456 // XXX wxCENTER is added for backward compatibility;
457 // wxALIGN_CENTER should be used in new code
458 child_pos.y += (m_size.y - size.y) / 2;
459
460 item->SetDimension( child_pos, child_size );
461
462 pt.x += width;
463 }
464
465 node = node->Next();
466 }
467}
468
469wxSize wxBoxSizer::CalcMin()
470{
471 if (m_children.GetCount() == 0)
472 return wxSize(10,10);
473
474 m_stretchable = 0;
475 m_minWidth = 0;
476 m_minHeight = 0;
477 m_fixedWidth = 0;
478 m_fixedHeight = 0;
479
480 wxNode *node = m_children.GetFirst();
481 while (node)
482 {
483 wxSizerItem *item = (wxSizerItem*) node->Data();
484
485 int weight = 1;
486 if (item->GetOption())
487 weight = item->GetOption();
488
489 wxSize size( item->CalcMin() );
490
491 if (m_orient == wxHORIZONTAL)
492 {
493 m_minWidth += (size.x * weight);
494 m_minHeight = wxMax( m_minHeight, size.y );
495 }
496 else
497 {
498 m_minHeight += (size.y * weight);
499 m_minWidth = wxMax( m_minWidth, size.x );
500 }
501
502 if (item->GetOption())
503 {
504 m_stretchable += weight;
505 }
506 else
507 {
508 if (m_orient == wxVERTICAL)
509 {
510 m_fixedHeight += size.y;
511 m_fixedWidth = wxMax( m_fixedWidth, size.x );
512 }
513 else
514 {
515 m_fixedWidth += size.x;
516 m_fixedHeight = wxMax( m_fixedHeight, size.y );
517 }
518 }
519
520 node = node->Next();
521 }
522
523 return wxSize( m_minWidth, m_minHeight );
524}
525
526//---------------------------------------------------------------------------
527// wxStaticBoxSizer
528//---------------------------------------------------------------------------
529
530wxStaticBoxSizer::wxStaticBoxSizer( wxStaticBox *box, int orient )
531 : wxBoxSizer( orient )
532{
533 wxASSERT_MSG( box, wxT("wxStaticBoxSizer needs a static box") );
534
535 m_staticBox = box;
536}
537
538void wxStaticBoxSizer::RecalcSizes()
539{
540 // this will have to be done platform by platform
541 // as there is no way to guess the thickness of
542 // a wxStaticBox border
543 int top_border = 15;
544 if (m_staticBox->GetLabel().IsEmpty()) top_border = 5;
545 int other_border = 5;
546
547 m_staticBox->SetSize( m_position.x, m_position.y, m_size.x, m_size.y );
548
549 wxPoint old_pos( m_position );
550 m_position.x += other_border;
551 m_position.y += top_border;
552 wxSize old_size( m_size );
553 m_size.x -= 2*other_border;
554 m_size.y -= top_border + other_border;
555
556 wxBoxSizer::RecalcSizes();
557
558 m_position = old_pos;
559 m_size = old_size;
560}
561
562wxSize wxStaticBoxSizer::CalcMin()
563{
564 // This will have to be done platform by platform
565 // as there is no way to guess the thickness of
566 // a wxStaticBox border.
567
568 int top_border = 15;
569 if (m_staticBox->GetLabel().IsEmpty()) top_border = 5;
570 int other_border = 5;
571
572 wxSize ret( wxBoxSizer::CalcMin() );
573 ret.x += 2*other_border;
574 ret.y += other_border + top_border;
575
576 return ret;
577}
578
579//---------------------------------------------------------------------------
580// wxNotebookSizer
581//---------------------------------------------------------------------------
582
583#if wxUSE_NOTEBOOK
584
585wxNotebookSizer::wxNotebookSizer( wxNotebook *nb )
586{
587 wxASSERT_MSG( nb, wxT("wxNotebookSizer needs a notebook") );
588
589 m_notebook = nb;
590}
591
592void wxNotebookSizer::RecalcSizes()
593{
594 m_notebook->SetSize( m_position.x, m_position.y, m_size.x, m_size.y );
595}
596
597wxSize wxNotebookSizer::CalcMin()
598{
599 // This will have to be done platform by platform
600 // as there is no way to guess the thickness of
601 // the wxNotebook tabs and border.
602
603 int borderX = 5;
604 int borderY = 5;
605 if ((m_notebook->HasFlag(wxNB_RIGHT)) ||
606 (m_notebook->HasFlag(wxNB_LEFT)))
607 {
608 borderX += 70; // improvements later..
609 }
610 else
611 {
612 borderY += 35; // improvements later..
613 }
614
615 if (m_notebook->GetChildren().GetCount() == 0)
616 return wxSize(borderX + 10, borderY + 10);
617
618 int maxX = 0;
619 int maxY = 0;
620
621 wxWindowList::Node *node = m_notebook->GetChildren().GetFirst();
622 while (node)
623 {
624 wxWindow *item = node->GetData();
625 wxSizer *itemsizer = item->GetSizer();
626
627 if (itemsizer)
628 {
629 wxSize subsize( itemsizer->CalcMin() );
630
631 if (subsize.x > maxX) maxX = subsize.x;
632 if (subsize.y > maxY) maxY = subsize.y;
633 }
634
635 node = node->GetNext();
636 }
637
638 return wxSize( borderX + maxX, borderY + maxY );
639}
640
641#endif // wxUSE_NOTEBOOK