]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/common/sizer.cpp
Catching up to latest thread changes
[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
27//---------------------------------------------------------------------------
28
29IMPLEMENT_ABSTRACT_CLASS(wxSizerItem, wxObject);
30IMPLEMENT_ABSTRACT_CLASS(wxSizer, wxObject);
31IMPLEMENT_ABSTRACT_CLASS(wxBoxSizer, wxSizer);
32IMPLEMENT_ABSTRACT_CLASS(wxStaticBoxSizer, wxBoxSizer);
33
34//---------------------------------------------------------------------------
35// wxSizerItem
36//---------------------------------------------------------------------------
37
38wxSizerItem::wxSizerItem( int width, int height, int option, int flag, int border, wxObject* userData )
39{
40 m_window = (wxWindow *) NULL;
41 m_sizer = (wxSizer *) NULL;
42 m_option = option;
43 m_border = border;
44 m_flag = flag;
45 m_userData = userData;
46
47 // minimal size is the initial size
48 m_minSize.x = width;
49 m_minSize.y = height;
50
51 SetRatio(width, height);
52
53 // size is set directly
54 m_size = m_minSize;
55}
56
57wxSizerItem::wxSizerItem( wxWindow *window, int option, int flag, int border, wxObject* userData )
58{
59 m_window = window;
60 m_sizer = (wxSizer *) NULL;
61 m_option = option;
62 m_border = border;
63 m_flag = flag;
64 m_userData = userData;
65
66 // minimal size is the initial size
67 m_minSize = window->GetSize();
68
69 // aspect ratio calculated from initial size
70 SetRatio(m_minSize);
71
72 // size is calculated later
73 // m_size = ...
74}
75
76wxSizerItem::wxSizerItem( wxSizer *sizer, int option, int flag, int border, wxObject* userData )
77{
78 m_window = (wxWindow *) NULL;
79 m_sizer = sizer;
80 m_option = option;
81 m_border = border;
82 m_flag = flag;
83 m_userData = userData;
84
85 // minimal size is calculated later
86 // m_minSize = ...
87 m_ratio = 0;
88
89 // size is calculated later
90 // m_size = ...
91}
92
93wxSizerItem::~wxSizerItem()
94{
95 if (m_userData)
96 delete m_userData;
97 if (m_sizer)
98 delete m_sizer;
99}
100
101
102wxSize wxSizerItem::GetSize()
103{
104 wxSize ret;
105 if (IsSizer())
106 ret = m_sizer->GetSize();
107 else
108 if (IsWindow())
109 ret = m_window->GetSize();
110 else ret = m_size;
111
112 if (m_flag & wxWEST)
113 ret.x += m_border;
114 if (m_flag & wxEAST)
115 ret.x += m_border;
116 if (m_flag & wxNORTH)
117 ret.y += m_border;
118 if (m_flag & wxSOUTH)
119 ret.y += m_border;
120
121 return ret;
122}
123
124wxSize wxSizerItem::CalcMin()
125{
126 wxSize ret;
127 if (IsSizer())
128 {
129 ret = m_sizer->CalcMin();
130 // if we have to preserve aspect ratio _AND_ this is
131 // the first-time calculation, consider ret to be initial size
132 if ((m_flag & wxSHAPED) && !m_ratio) SetRatio(ret);
133 }
134
135/*
136 The minimum size of a window should be the
137 initial size, as saved in m_minSize, not the
138 current size.
139
140 else
141 if (IsWindow())
142 ret = m_window->GetSize();
143*/
144 else ret = m_minSize;
145
146 if (m_flag & wxWEST)
147 ret.x += m_border;
148 if (m_flag & wxEAST)
149 ret.x += m_border;
150 if (m_flag & wxNORTH)
151 ret.y += m_border;
152 if (m_flag & wxSOUTH)
153 ret.y += m_border;
154
155 return ret;
156}
157
158void wxSizerItem::SetDimension( wxPoint pos, wxSize size )
159{
160 if (m_flag & wxWEST)
161 {
162 pos.x += m_border;
163 size.x -= m_border;
164 }
165 if (m_flag & wxEAST)
166 {
167 size.x -= m_border;
168 }
169 if (m_flag & wxNORTH)
170 {
171 pos.y += m_border;
172 size.y -= m_border;
173 }
174 if (m_flag & wxSOUTH)
175 {
176 size.y -= m_border;
177 }
178 if (m_flag & wxSHAPED) {
179 // adjust aspect ratio
180 int rwidth = (int) (size.y * m_ratio);
181 if (rwidth > size.x) {
182 // fit horizontally
183 int rheight = (int) (size.x / m_ratio);
184 // add vertical space
185 if (m_flag & wxALIGN_CENTER_VERTICAL)
186 pos.y += (size.y - rheight) / 2;
187 else if (m_flag & wxALIGN_BOTTOM)
188 pos.y += (size.y - rheight);
189 // use reduced dimensions
190 size.y =rheight;
191 } else if (rwidth < size.x) {
192 // add horizontal space
193 if (m_flag & wxALIGN_CENTER_HORIZONTAL)
194 pos.x += (size.x - rwidth) / 2;
195 else if (m_flag & wxALIGN_RIGHT)
196 pos.x += (size.x - rwidth);
197 size.x = rwidth;
198 }
199 }
200
201 if (IsSizer())
202 m_sizer->SetDimension( pos.x, pos.y, size.x, size.y );
203
204 if (IsWindow())
205 m_window->SetSize( pos.x, pos.y, size.x, size.y );
206
207 m_size = size;
208}
209
210bool wxSizerItem::IsWindow()
211{
212 return (m_window != NULL);
213}
214
215bool wxSizerItem::IsSizer()
216{
217 return (m_sizer != NULL);
218}
219
220bool wxSizerItem::IsSpacer()
221{
222 return (m_window == NULL) && (m_sizer == NULL);
223}
224
225//---------------------------------------------------------------------------
226// wxSizer
227//---------------------------------------------------------------------------
228
229wxSizer::wxSizer()
230{
231 m_children.DeleteContents( TRUE );
232}
233
234wxSizer::~wxSizer()
235{
236}
237
238void wxSizer::Add( wxWindow *window, int option, int flag, int border, wxObject* userData )
239{
240 m_children.Append( new wxSizerItem( window, option, flag, border, userData ) );
241}
242
243void wxSizer::Add( wxSizer *sizer, int option, int flag, int border, wxObject* userData )
244{
245 m_children.Append( new wxSizerItem( sizer, option, flag, border, userData ) );
246}
247
248void wxSizer::Add( int width, int height, int option, int flag, int border, wxObject* userData )
249{
250 m_children.Append( new wxSizerItem( width, height, option, flag, border, userData ) );
251}
252
253void wxSizer::Prepend( wxWindow *window, int option, int flag, int border, wxObject* userData )
254{
255 m_children.Insert( new wxSizerItem( window, option, flag, border, userData ) );
256}
257
258void wxSizer::Prepend( wxSizer *sizer, int option, int flag, int border, wxObject* userData )
259{
260 m_children.Insert( new wxSizerItem( sizer, option, flag, border, userData ) );
261}
262
263void wxSizer::Prepend( int width, int height, int option, int flag, int border, wxObject* userData )
264{
265 m_children.Insert( new wxSizerItem( width, height, option, flag, border, userData ) );
266}
267
268bool wxSizer::Remove( wxWindow *window )
269{
270 wxASSERT( window );
271
272 wxNode *node = m_children.First();
273 while (node)
274 {
275 wxSizerItem *item = (wxSizerItem*)node->Data();
276 if (item->GetWindow() == window)
277 {
278 m_children.DeleteNode( node );
279 return TRUE;
280 }
281 node = node->Next();
282 }
283
284 return FALSE;
285}
286
287bool wxSizer::Remove( wxSizer *sizer )
288{
289 wxASSERT( sizer );
290
291 wxNode *node = m_children.First();
292 while (node)
293 {
294 wxSizerItem *item = (wxSizerItem*)node->Data();
295 if (item->GetSizer() == sizer)
296 {
297 m_children.DeleteNode( node );
298 return TRUE;
299 }
300 node = node->Next();
301 }
302
303 return FALSE;
304}
305
306bool wxSizer::Remove( int pos )
307{
308 wxNode *node = m_children.Nth( pos );
309 if (!node) return FALSE;
310
311 m_children.DeleteNode( node );
312
313 return TRUE;
314}
315
316void wxSizer::Fit( wxWindow *window )
317{
318 window->SetSize( GetMinWindowSize( window ) );
319}
320
321void wxSizer::Layout()
322{
323 CalcMin();
324 RecalcSizes();
325}
326
327void wxSizer::SetSizeHints( wxWindow *window )
328{
329 wxSize size( GetMinWindowSize( window ) );
330 window->SetSizeHints( size.x, size.y );
331}
332
333wxSize wxSizer::GetMinWindowSize( wxWindow *window )
334{
335 wxSize minSize( GetMinSize() );
336 wxSize size( window->GetSize() );
337 wxSize client_size( window->GetClientSize() );
338 return wxSize( minSize.x+size.x-client_size.x,
339 minSize.y+size.y-client_size.y );
340}
341
342void wxSizer::SetDimension( int x, int y, int width, int height )
343{
344 m_position.x = x;
345 m_position.y = y;
346 m_size.x = width;
347 m_size.y = height;
348 CalcMin();
349 RecalcSizes();
350}
351
352//---------------------------------------------------------------------------
353// wxBoxSizer
354//---------------------------------------------------------------------------
355
356wxBoxSizer::wxBoxSizer( int orient )
357{
358 m_orient = orient;
359}
360
361void wxBoxSizer::RecalcSizes()
362{
363 if (m_children.GetCount() == 0)
364 return;
365
366 int delta = 0;
367 int extra = 0;
368 if (m_stretchable)
369 {
370 if (m_orient == wxHORIZONTAL)
371 {
372 delta = (m_size.x - m_fixedWidth) / m_stretchable;
373 extra = (m_size.x - m_fixedWidth) % m_stretchable;
374 }
375 else
376 {
377 delta = (m_size.y - m_fixedHeight) / m_stretchable;
378 extra = (m_size.y - m_fixedHeight) % m_stretchable;
379 }
380 }
381
382 wxPoint pt( m_position );
383
384 wxNode *node = m_children.GetFirst();
385 while (node)
386 {
387 wxSizerItem *item = (wxSizerItem*) node->Data();
388
389 int weight = 1;
390 if (item->GetOption())
391 weight = item->GetOption();
392
393 wxSize size( item->CalcMin() );
394
395 if (m_orient == wxVERTICAL)
396 {
397 long height = size.y;
398 if (item->GetOption())
399 {
400 height = (delta * weight) + extra;
401 extra = 0; // only the first item will get the remainder as extra size
402 }
403
404 wxPoint child_pos( pt );
405 wxSize child_size( wxSize( size.x, height) );
406
407 if (item->GetFlag() & (wxEXPAND | wxSHAPED))
408 child_size.x = m_size.x;
409 else if (item->GetFlag() & wxALIGN_RIGHT)
410 child_pos.x += m_size.x - size.x;
411 else if (item->GetFlag() & (wxCENTER | wxALIGN_CENTER_HORIZONTAL))
412 // XXX wxCENTER is added for backward compatibility;
413 // wxALIGN_CENTER should be used in new code
414 child_pos.x += (m_size.x - size.x) / 2;
415
416 item->SetDimension( child_pos, child_size );
417
418 pt.y += height;
419 }
420 else
421 {
422 long width = size.x;
423 if (item->GetOption())
424 {
425 width = (delta * weight) + extra;
426 extra = 0; // only the first item will get the remainder as extra size
427 }
428
429 wxPoint child_pos( pt );
430 wxSize child_size( wxSize(width, size.y) );
431
432 if (item->GetFlag() & (wxEXPAND | wxSHAPED))
433 child_size.y = m_size.y;
434 else if (item->GetFlag() & wxALIGN_BOTTOM)
435 child_pos.y += m_size.y - size.y;
436 else if (item->GetFlag() & (wxCENTER | wxALIGN_CENTER_VERTICAL))
437 // XXX wxCENTER is added for backward compatibility;
438 // wxALIGN_CENTER should be used in new code
439 child_pos.y += (m_size.y - size.y) / 2;
440
441 item->SetDimension( child_pos, child_size );
442
443 pt.x += width;
444 }
445
446 node = node->Next();
447 }
448}
449
450wxSize wxBoxSizer::CalcMin()
451{
452 if (m_children.GetCount() == 0)
453 return wxSize(2,2);
454
455 m_stretchable = 0;
456 m_minWidth = 0;
457 m_minHeight = 0;
458 m_fixedWidth = 0;
459 m_fixedHeight = 0;
460
461 wxNode *node = m_children.GetFirst();
462 while (node)
463 {
464 wxSizerItem *item = (wxSizerItem*) node->Data();
465
466 int weight = 1;
467 if (item->GetOption())
468 weight = item->GetOption();
469
470 wxSize size( item->CalcMin() );
471
472 if (m_orient == wxHORIZONTAL)
473 {
474 m_minWidth += (size.x * weight);
475 m_minHeight = wxMax( m_minHeight, size.y );
476 }
477 else
478 {
479 m_minHeight += (size.y * weight);
480 m_minWidth = wxMax( m_minWidth, size.x );
481 }
482
483 if (item->GetOption())
484 {
485 m_stretchable += weight;
486 }
487 else
488 {
489 if (m_orient == wxVERTICAL)
490 {
491 m_fixedHeight += size.y;
492 m_fixedWidth = wxMax( m_fixedWidth, size.x );
493 }
494 else
495 {
496 m_fixedWidth += size.x;
497 m_fixedHeight = wxMax( m_fixedHeight, size.y );
498 }
499 }
500
501 node = node->Next();
502 }
503
504 return wxSize( m_minWidth, m_minHeight );
505}
506
507//---------------------------------------------------------------------------
508// wxStaticBoxSizer
509//---------------------------------------------------------------------------
510
511wxStaticBoxSizer::wxStaticBoxSizer( wxStaticBox *box, int orient )
512 : wxBoxSizer( orient )
513{
514 wxASSERT_MSG( box, wxT("wxStaticBoxSizer needs a static box") );
515
516 m_staticBox = box;
517}
518
519void wxStaticBoxSizer::RecalcSizes()
520{
521 // this will have to be done platform by platform
522 // as there is no way to guess the thickness of
523 // a wxStaticBox border
524 int top_border = 15;
525 if (m_staticBox->GetLabel().IsEmpty()) top_border = 5;
526 int other_border = 5;
527
528 m_staticBox->SetSize( m_position.x, m_position.y, m_size.x, m_size.y );
529
530 wxPoint old_pos( m_position );
531 m_position.x += other_border;
532 m_position.y += top_border;
533 wxSize old_size( m_size );
534 m_size.x -= 2*other_border;
535 m_size.y -= top_border + other_border;
536
537 wxBoxSizer::RecalcSizes();
538
539 m_position = old_pos;
540 m_size = old_size;
541}
542
543wxSize wxStaticBoxSizer::CalcMin()
544{
545 // this will have to be done platform by platform
546 // as there is no way to guess the thickness of
547 // a wxStaticBox border
548 int top_border = 15;
549 if (m_staticBox->GetLabel().IsEmpty()) top_border = 5;
550 int other_border = 5;
551
552 wxSize ret( wxBoxSizer::CalcMin() );
553 ret.x += 2*top_border;
554 ret.y += other_border + top_border;
555
556 return ret;
557}