]> git.saurik.com Git - wxWidgets.git/blob - src/common/layout.cpp
3718e78c12c83fcca1e11d1dc450f60fd2f564d1
[wxWidgets.git] / src / common / layout.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: layout.cpp
3 // Purpose: Constraint layout system classes
4 // Author: Julian Smart
5 // Modified by:
6 // Created: 04/01/98
7 // RCS-ID: $Id$
8 // Copyright: (c) Julian Smart and Markus Holzem
9 // Licence: wxWindows license
10 /////////////////////////////////////////////////////////////////////////////
11
12 // =============================================================================
13 // declarations
14 // =============================================================================
15
16 // ----------------------------------------------------------------------------
17 // headers
18 // ----------------------------------------------------------------------------
19
20 #ifdef __GNUG__
21 #pragma implementation "layout.h"
22 #endif
23
24 // For compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
26
27 #ifdef __BORLANDC__
28 #pragma hdrstop
29 #endif
30
31 #include "wx/defs.h"
32
33 #if wxUSE_CONSTRAINTS
34
35 #ifndef WX_PRECOMP
36 #include "wx/window.h"
37 #include "wx/utils.h"
38 #include "wx/dialog.h"
39 #include "wx/msgdlg.h"
40 #include "wx/intl.h"
41 #endif
42
43 #include "wx/layout.h"
44
45 #if !USE_SHARED_LIBRARY
46 IMPLEMENT_DYNAMIC_CLASS(wxIndividualLayoutConstraint, wxObject)
47 IMPLEMENT_DYNAMIC_CLASS(wxLayoutConstraints, wxObject)
48 IMPLEMENT_DYNAMIC_CLASS(wxSizer, wxObject)
49 IMPLEMENT_DYNAMIC_CLASS(wxRowColSizer, wxSizer)
50 IMPLEMENT_DYNAMIC_CLASS(wxSpacingSizer, wxSizer)
51 #endif
52
53 /*
54 TODO:
55 - Non shrink-to-fit row-col behaviour.
56 - Give justification styles, so can e.g. centre
57 the rows & cols, distribute the available space...
58 - Shrink-to-fit: should resize outer window (e.g. dialog box)
59 if directly associated with this kind of window.
60 - How to deal with a rowcol that stretches in one direction
61 but shrinks-to-fit in other. E.g. a horizontal toolbar: the width
62 stretches to fit the frame, but the height is constant
63 or wraps around contents. The algorithm currently assumes
64 both dimensions have the same behaviour. Could assume a constant
65 height (absolute value).
66 - rowcol where each row or column is aligned (length of
67 largest element determines spacing)
68 - Groupbox sizer
69 - Analyze aesthetic dialog boxes and implement using sizers.
70 - What reuseable components can we provide? E.g. Ok/Cancel/Help
71 group of buttons.
72 - use wxStaticItems for aesthetic dialogs.
73
74 */
75
76 // Find margin sizes if a sizer, or zero otherwise
77 int wxSizerMarginX(wxWindowBase *win)
78 {
79 if ( win->IsKindOf(CLASSINFO(wxSizer)) )
80 {
81 wxSizer *sizer = (wxSizer *)win;
82 return sizer->GetBorderX();
83 }
84 else
85 return 0;
86 }
87
88 int wxSizerMarginY(wxWindowBase *win)
89 {
90 if ( win->IsKindOf(CLASSINFO(wxSizer)) )
91 {
92 wxSizer *sizer = (wxSizer *)win;
93 return sizer->GetBorderY();
94 }
95 else
96 return 0;
97 }
98
99
100 wxIndividualLayoutConstraint::wxIndividualLayoutConstraint()
101 {
102 myEdge = wxTop;
103 relationship = wxUnconstrained;
104 margin = 0;
105 value = 0;
106 percent = 0;
107 otherEdge = wxTop;
108 done = FALSE;
109 otherWin = (wxWindowBase *) NULL;
110 }
111
112 wxIndividualLayoutConstraint::~wxIndividualLayoutConstraint()
113 {
114 }
115
116 void wxIndividualLayoutConstraint::Set(wxRelationship rel, wxWindowBase *otherW, wxEdge otherE, int val, int marg)
117 {
118 relationship = rel;
119 otherWin = otherW;
120 otherEdge = otherE;
121 value = val;
122 margin = marg;
123 }
124
125 void wxIndividualLayoutConstraint::LeftOf(wxWindowBase *sibling, int marg)
126 {
127 Set(wxLeftOf, sibling, wxLeft, 0, marg);
128 }
129
130 void wxIndividualLayoutConstraint::RightOf(wxWindowBase *sibling, int marg)
131 {
132 Set(wxRightOf, sibling, wxRight, 0, marg);
133 }
134
135 void wxIndividualLayoutConstraint::Above(wxWindowBase *sibling, int marg)
136 {
137 Set(wxAbove, sibling, wxTop, 0, marg);
138 }
139
140 void wxIndividualLayoutConstraint::Below(wxWindowBase *sibling, int marg)
141 {
142 Set(wxBelow, sibling, wxBottom, 0, marg);
143 }
144
145 //
146 // 'Same edge' alignment
147 //
148 void wxIndividualLayoutConstraint::SameAs(wxWindowBase *otherW, wxEdge edge, int marg)
149 {
150 Set(wxPercentOf, otherW, edge, 0, marg);
151 percent = 100;
152 }
153
154 // The edge is a percentage of the other window's edge
155 void wxIndividualLayoutConstraint::PercentOf(wxWindowBase *otherW, wxEdge wh, int per)
156 {
157 otherWin = otherW;
158 relationship = wxPercentOf;
159 percent = per;
160
161 otherEdge = wh;
162 }
163
164 //
165 // Edge has absolute value
166 //
167 void wxIndividualLayoutConstraint::Absolute(int val)
168 {
169 value = val; relationship = wxAbsolute;
170 }
171
172 // Reset constraint if it mentions otherWin
173 bool wxIndividualLayoutConstraint::ResetIfWin(wxWindowBase *otherW)
174 {
175 if (otherW == otherWin)
176 {
177 myEdge = wxTop;
178 relationship = wxAsIs;
179 margin = 0;
180 value = 0;
181 percent = 0;
182 otherEdge = wxTop;
183 otherWin = (wxWindowBase *) NULL;
184 return TRUE;
185 }
186 else
187 return FALSE;
188 }
189
190 // Try to satisfy constraint
191 bool wxIndividualLayoutConstraint::SatisfyConstraint(wxLayoutConstraints *constraints, wxWindowBase *win)
192 {
193 if (relationship == wxAbsolute)
194 {
195 done = TRUE;
196 return TRUE;
197 }
198
199 switch (myEdge)
200 {
201 case wxLeft:
202 {
203 switch (relationship)
204 {
205 case wxLeftOf:
206 {
207 // We can know this edge if: otherWin is win's
208 // parent, or otherWin has a satisfied constraint,
209 // or otherWin has no constraint.
210 int edgePos = GetEdge(otherEdge, win, otherWin);
211 if (edgePos != -1)
212 {
213 value = edgePos - margin;
214 done = TRUE;
215 return TRUE;
216 }
217 else
218 return FALSE;
219 }
220 case wxRightOf:
221 {
222 int edgePos = GetEdge(otherEdge, win, otherWin);
223 if (edgePos != -1)
224 {
225 value = edgePos + margin;
226 done = TRUE;
227 return TRUE;
228 }
229 else
230 return FALSE;
231 }
232 case wxPercentOf:
233 {
234 int edgePos = GetEdge(otherEdge, win, otherWin);
235 if (edgePos != -1)
236 {
237 value = (int)(edgePos*(((float)percent)*0.01) + margin);
238 done = TRUE;
239 return TRUE;
240 }
241 else
242 return FALSE;
243 }
244 case wxUnconstrained:
245 {
246 // We know the left-hand edge position if we know
247 // the right-hand edge and we know the width; OR if
248 // we know the centre and the width.
249 if (constraints->right.GetDone() && constraints->width.GetDone())
250 {
251 value = (constraints->right.GetValue() - constraints->width.GetValue() + margin);
252 done = TRUE;
253 return TRUE;
254 }
255 else if (constraints->centreX.GetDone() && constraints->width.GetDone())
256 {
257 value = (int)(constraints->centreX.GetValue() - (constraints->width.GetValue()/2) + margin);
258 done = TRUE;
259 return TRUE;
260 }
261 else
262 return FALSE;
263 }
264 case wxAsIs:
265 {
266 int y;
267 win->GetPosition(&value, &y);
268 done = TRUE;
269 return TRUE;
270 }
271 default:
272 break;
273 }
274 break;
275 }
276 case wxRight:
277 {
278 switch (relationship)
279 {
280 case wxLeftOf:
281 {
282 // We can know this edge if: otherWin is win's
283 // parent, or otherWin has a satisfied constraint,
284 // or otherWin has no constraint.
285 int edgePos = GetEdge(otherEdge, win, otherWin);
286 if (edgePos != -1)
287 {
288 value = edgePos - margin;
289 done = TRUE;
290 return TRUE;
291 }
292 else
293 return FALSE;
294 }
295 case wxRightOf:
296 {
297 int edgePos = GetEdge(otherEdge, win, otherWin);
298 if (edgePos != -1)
299 {
300 value = edgePos + margin;
301 done = TRUE;
302 return TRUE;
303 }
304 else
305 return FALSE;
306 }
307 case wxPercentOf:
308 {
309 int edgePos = GetEdge(otherEdge, win, otherWin);
310 if (edgePos != -1)
311 {
312 value = (int)(edgePos*(((float)percent)*0.01) - margin);
313 done = TRUE;
314 return TRUE;
315 }
316 else
317 return FALSE;
318 }
319 case wxUnconstrained:
320 {
321 // We know the right-hand edge position if we know the
322 // left-hand edge and we know the width, OR if we know the
323 // centre edge and the width.
324 if (constraints->left.GetDone() && constraints->width.GetDone())
325 {
326 value = (constraints->left.GetValue() + constraints->width.GetValue() - margin);
327 done = TRUE;
328 return TRUE;
329 }
330 else if (constraints->centreX.GetDone() && constraints->width.GetDone())
331 {
332 value = (int)(constraints->centreX.GetValue() + (constraints->width.GetValue()/2) - margin);
333 done = TRUE;
334 return TRUE;
335 }
336 else
337 return FALSE;
338 }
339 case wxAsIs:
340 {
341 int x, y;
342 int w, h;
343 win->GetSize(&w, &h);
344 win->GetPosition(&x, &y);
345 value = x + w;
346 done = TRUE;
347 return TRUE;
348 }
349 default:
350 break;
351 }
352 break;
353 }
354 case wxTop:
355 {
356 switch (relationship)
357 {
358 case wxAbove:
359 {
360 // We can know this edge if: otherWin is win's
361 // parent, or otherWin has a satisfied constraint,
362 // or otherWin has no constraint.
363 int edgePos = GetEdge(otherEdge, win, otherWin);
364 if (edgePos != -1)
365 {
366 value = edgePos - margin;
367 done = TRUE;
368 return TRUE;
369 }
370 else
371 return FALSE;
372 }
373 case wxBelow:
374 {
375 int edgePos = GetEdge(otherEdge, win, otherWin);
376 if (edgePos != -1)
377 {
378 value = edgePos + margin;
379 done = TRUE;
380 return TRUE;
381 }
382 else
383 return FALSE;
384 }
385 case wxPercentOf:
386 {
387 int edgePos = GetEdge(otherEdge, win, otherWin);
388 if (edgePos != -1)
389 {
390 value = (int)(edgePos*(((float)percent)*0.01) + margin);
391 done = TRUE;
392 return TRUE;
393 }
394 else
395 return FALSE;
396 }
397 case wxUnconstrained:
398 {
399 // We know the top edge position if we know the bottom edge
400 // and we know the height; OR if we know the centre edge and
401 // the height.
402 if (constraints->bottom.GetDone() && constraints->height.GetDone())
403 {
404 value = (constraints->bottom.GetValue() - constraints->height.GetValue() + margin);
405 done = TRUE;
406 return TRUE;
407 }
408 else if (constraints->centreY.GetDone() && constraints->height.GetDone())
409 {
410 value = (constraints->centreY.GetValue() - (constraints->height.GetValue()/2) + margin);
411 done = TRUE;
412 return TRUE;
413 }
414 else
415 return FALSE;
416 }
417 case wxAsIs:
418 {
419 int x;
420 win->GetPosition(&x, &value);
421 done = TRUE;
422 return TRUE;
423 }
424 default:
425 break;
426 }
427 break;
428 }
429 case wxBottom:
430 {
431 switch (relationship)
432 {
433 case wxAbove:
434 {
435 // We can know this edge if: otherWin is win's parent,
436 // or otherWin has a satisfied constraint, or
437 // otherWin has no constraint.
438 int edgePos = GetEdge(otherEdge, win, otherWin);
439 if (edgePos != -1)
440 {
441 value = edgePos + margin;
442 done = TRUE;
443 return TRUE;
444 }
445 else
446 return FALSE;
447 }
448 case wxBelow:
449 {
450 int edgePos = GetEdge(otherEdge, win, otherWin);
451 if (edgePos != -1)
452 {
453 value = edgePos - margin;
454 done = TRUE;
455 return TRUE;
456 }
457 else
458 return FALSE;
459 }
460 case wxPercentOf:
461 {
462 int edgePos = GetEdge(otherEdge, win, otherWin);
463 if (edgePos != -1)
464 {
465 value = (int)(edgePos*(((float)percent)*0.01) - margin);
466 done = TRUE;
467 return TRUE;
468 }
469 else
470 return FALSE;
471 }
472 case wxUnconstrained:
473 {
474 // We know the bottom edge position if we know the top edge
475 // and we know the height; OR if we know the centre edge and
476 // the height.
477 if (constraints->top.GetDone() && constraints->height.GetDone())
478 {
479 value = (constraints->top.GetValue() + constraints->height.GetValue() - margin);
480 done = TRUE;
481 return TRUE;
482 }
483 else if (constraints->centreY.GetDone() && constraints->height.GetDone())
484 {
485 value = (constraints->centreY.GetValue() + (constraints->height.GetValue()/2) - margin);
486 done = TRUE;
487 return TRUE;
488 }
489 else
490 return FALSE;
491 }
492 case wxAsIs:
493 {
494 int x, y;
495 int w, h;
496 win->GetSize(&w, &h);
497 win->GetPosition(&x, &y);
498 value = h + y;
499 done = TRUE;
500 return TRUE;
501 }
502 default:
503 break;
504 }
505 break;
506 }
507 case wxCentreX:
508 {
509 switch (relationship)
510 {
511 case wxLeftOf:
512 {
513 // We can know this edge if: otherWin is win's parent, or
514 // otherWin has a satisfied constraint, or otherWin has no
515 // constraint.
516 int edgePos = GetEdge(otherEdge, win, otherWin);
517 if (edgePos != -1)
518 {
519 value = edgePos - margin;
520 done = TRUE;
521 return TRUE;
522 }
523 else
524 return FALSE;
525 }
526 case wxRightOf:
527 {
528 int edgePos = GetEdge(otherEdge, win, otherWin);
529 if (edgePos != -1)
530 {
531 value = edgePos + margin;
532 done = TRUE;
533 return TRUE;
534 }
535 else
536 return FALSE;
537 }
538 case wxPercentOf:
539 {
540 int edgePos = GetEdge(otherEdge, win, otherWin);
541 if (edgePos != -1)
542 {
543 value = (int)(edgePos*(((float)percent)*0.01) + margin);
544 done = TRUE;
545 return TRUE;
546 }
547 else
548 return FALSE;
549 }
550 case wxUnconstrained:
551 {
552 // We know the centre position if we know
553 // the left-hand edge and we know the width, OR
554 // the right-hand edge and the width
555 if (constraints->left.GetDone() && constraints->width.GetDone())
556 {
557 value = (int)(constraints->left.GetValue() + (constraints->width.GetValue()/2) + margin);
558 done = TRUE;
559 return TRUE;
560 }
561 else if (constraints->right.GetDone() && constraints->width.GetDone())
562 {
563 value = (int)(constraints->left.GetValue() - (constraints->width.GetValue()/2) + margin);
564 done = TRUE;
565 return TRUE;
566 }
567 else
568 return FALSE;
569 }
570 default:
571 break;
572 }
573 break;
574 }
575 case wxCentreY:
576 {
577 switch (relationship)
578 {
579 case wxAbove:
580 {
581 // We can know this edge if: otherWin is win's parent,
582 // or otherWin has a satisfied constraint, or otherWin
583 // has no constraint.
584 int edgePos = GetEdge(otherEdge, win, otherWin);
585 if (edgePos != -1)
586 {
587 value = edgePos - margin;
588 done = TRUE;
589 return TRUE;
590 }
591 else
592 return FALSE;
593 }
594 case wxBelow:
595 {
596 int edgePos = GetEdge(otherEdge, win, otherWin);
597 if (edgePos != -1)
598 {
599 value = edgePos + margin;
600 done = TRUE;
601 return TRUE;
602 }
603 else
604 return FALSE;
605 }
606 case wxPercentOf:
607 {
608 int edgePos = GetEdge(otherEdge, win, otherWin);
609 if (edgePos != -1)
610 {
611 value = (int)(edgePos*(((float)percent)*0.01) + margin);
612 done = TRUE;
613 return TRUE;
614 }
615 else
616 return FALSE;
617 }
618 case wxUnconstrained:
619 {
620 // We know the centre position if we know
621 // the top edge and we know the height, OR
622 // the bottom edge and the height.
623 if (constraints->bottom.GetDone() && constraints->height.GetDone())
624 {
625 value = (int)(constraints->bottom.GetValue() - (constraints->height.GetValue()/2) + margin);
626 done = TRUE;
627 return TRUE;
628 }
629 else if (constraints->top.GetDone() && constraints->height.GetDone())
630 {
631 value = (int)(constraints->top.GetValue() + (constraints->height.GetValue()/2) + margin);
632 done = TRUE;
633 return TRUE;
634 }
635 else
636 return FALSE;
637 }
638 default:
639 break;
640 }
641 break;
642 }
643 case wxWidth:
644 {
645 switch (relationship)
646 {
647 case wxPercentOf:
648 {
649 int edgePos = GetEdge(otherEdge, win, otherWin);
650 if (edgePos != -1)
651 {
652 value = (int)(edgePos*(((float)percent)*0.01));
653 done = TRUE;
654 return TRUE;
655 }
656 else
657 return FALSE;
658 }
659 case wxAsIs:
660 {
661 if (win)
662 {
663 int h;
664 win->GetSize(&value, &h);
665 done = TRUE;
666 return TRUE;
667 }
668 else return FALSE;
669 }
670 case wxUnconstrained:
671 {
672 // We know the width if we know the left edge and the right edge, OR
673 // if we know the left edge and the centre, OR
674 // if we know the right edge and the centre
675 if (constraints->left.GetDone() && constraints->right.GetDone())
676 {
677 value = constraints->right.GetValue() - constraints->left.GetValue();
678 done = TRUE;
679 return TRUE;
680 }
681 else if (constraints->centreX.GetDone() && constraints->left.GetDone())
682 {
683 value = (int)(2*(constraints->centreX.GetValue() - constraints->left.GetValue()));
684 done = TRUE;
685 return TRUE;
686 }
687 else if (constraints->centreX.GetDone() && constraints->right.GetDone())
688 {
689 value = (int)(2*(constraints->right.GetValue() - constraints->centreX.GetValue()));
690 done = TRUE;
691 return TRUE;
692 }
693 else
694 return FALSE;
695 }
696 default:
697 break;
698 }
699 break;
700 }
701 case wxHeight:
702 {
703 switch (relationship)
704 {
705 case wxPercentOf:
706 {
707 int edgePos = GetEdge(otherEdge, win, otherWin);
708 if (edgePos != -1)
709 {
710 value = (int)(edgePos*(((float)percent)*0.01));
711 done = TRUE;
712 return TRUE;
713 }
714 else
715 return FALSE;
716 }
717 case wxAsIs:
718 {
719 if (win)
720 {
721 int w;
722 win->GetSize(&w, &value);
723 done = TRUE;
724 return TRUE;
725 }
726 else return FALSE;
727 }
728 case wxUnconstrained:
729 {
730 // We know the height if we know the top edge and the bottom edge, OR
731 // if we know the top edge and the centre, OR
732 // if we know the bottom edge and the centre
733 if (constraints->top.GetDone() && constraints->bottom.GetDone())
734 {
735 value = constraints->bottom.GetValue() - constraints->top.GetValue();
736 done = TRUE;
737 return TRUE;
738 }
739 else if (constraints->top.GetDone() && constraints->centreY.GetDone())
740 {
741 value = (int)(2*(constraints->centreY.GetValue() - constraints->top.GetValue()));
742 done = TRUE;
743 return TRUE;
744 }
745 else if (constraints->bottom.GetDone() && constraints->centreY.GetDone())
746 {
747 value = (int)(2*(constraints->bottom.GetValue() - constraints->centreY.GetValue()));
748 done = TRUE;
749 return TRUE;
750 }
751 else
752 return FALSE;
753 }
754 default:
755 break;
756 }
757 break;
758 }
759 default:
760 break;
761 }
762 return FALSE;
763 }
764
765 // Get the value of this edge or dimension, or if this is not determinable, -1.
766 int wxIndividualLayoutConstraint::GetEdge(wxEdge which,
767 wxWindowBase *thisWin,
768 wxWindowBase *other) const
769 {
770 // If the edge or dimension belongs to the parent, then we know the
771 // dimension is obtainable immediately. E.g. a wxExpandSizer may contain a
772 // button (but the button's true parent is a panel, not the sizer)
773 if (other->GetChildren().Find(thisWin))
774 {
775 switch (which)
776 {
777 case wxLeft:
778 {
779 return wxSizerMarginX(other);
780 }
781 case wxTop:
782 {
783 return wxSizerMarginY(other);
784 }
785 case wxRight:
786 {
787 int w, h;
788 other->GetClientSizeConstraint(&w, &h);
789 return w - wxSizerMarginX(other);
790 }
791 case wxBottom:
792 {
793 int w, h;
794 other->GetClientSizeConstraint(&w, &h);
795 return h - wxSizerMarginY(other);
796 }
797 case wxWidth:
798 {
799 int w, h;
800 other->GetClientSizeConstraint(&w, &h);
801 return w - 2*wxSizerMarginX(other);
802 }
803 case wxHeight:
804 {
805 int w, h;
806 other->GetClientSizeConstraint(&w, &h);
807 return h - 2*wxSizerMarginY(other);
808 }
809 case wxCentreX:
810 case wxCentreY:
811 {
812 int w, h;
813 other->GetClientSizeConstraint(&w, &h);
814 if (which == wxCentreX)
815 return (int)(w/2);
816 else
817 return (int)(h/2);
818 }
819 default:
820 return -1;
821 }
822 }
823 switch (which)
824 {
825 case wxLeft:
826 {
827 wxLayoutConstraints *constr = other->GetConstraints();
828 // If no constraints, it means the window is not dependent
829 // on anything, and therefore we know its value immediately
830 if (constr)
831 {
832 if (constr->left.GetDone())
833 return constr->left.GetValue();
834 else
835 return -1;
836 }
837 else
838 {
839 int x, y;
840 other->GetPosition(&x, &y);
841 return x;
842 }
843 }
844 case wxTop:
845 {
846 wxLayoutConstraints *constr = other->GetConstraints();
847 // If no constraints, it means the window is not dependent
848 // on anything, and therefore we know its value immediately
849 if (constr)
850 {
851 if (constr->top.GetDone())
852 return constr->top.GetValue();
853 else
854 return -1;
855 }
856 else
857 {
858 int x, y;
859 other->GetPosition(&x, &y);
860 return y;
861 }
862 }
863 case wxRight:
864 {
865 wxLayoutConstraints *constr = other->GetConstraints();
866 // If no constraints, it means the window is not dependent
867 // on anything, and therefore we know its value immediately
868 if (constr)
869 {
870 if (constr->right.GetDone())
871 return constr->right.GetValue();
872 else
873 return -1;
874 }
875 else
876 {
877 int x, y, w, h;
878 other->GetPosition(&x, &y);
879 other->GetSize(&w, &h);
880 return (int)(x + w);
881 }
882 }
883 case wxBottom:
884 {
885 wxLayoutConstraints *constr = other->GetConstraints();
886 // If no constraints, it means the window is not dependent
887 // on anything, and therefore we know its value immediately
888 if (constr)
889 {
890 if (constr->bottom.GetDone())
891 return constr->bottom.GetValue();
892 else
893 return -1;
894 }
895 else
896 {
897 int x, y, w, h;
898 other->GetPosition(&x, &y);
899 other->GetSize(&w, &h);
900 return (int)(y + h);
901 }
902 }
903 case wxWidth:
904 {
905 wxLayoutConstraints *constr = other->GetConstraints();
906 // If no constraints, it means the window is not dependent
907 // on anything, and therefore we know its value immediately
908 if (constr)
909 {
910 if (constr->width.GetDone())
911 return constr->width.GetValue();
912 else
913 return -1;
914 }
915 else
916 {
917 int w, h;
918 other->GetSize(&w, &h);
919 return w;
920 }
921 }
922 case wxHeight:
923 {
924 wxLayoutConstraints *constr = other->GetConstraints();
925 // If no constraints, it means the window is not dependent
926 // on anything, and therefore we know its value immediately
927 if (constr)
928 {
929 if (constr->height.GetDone())
930 return constr->height.GetValue();
931 else
932 return -1;
933 }
934 else
935 {
936 int w, h;
937 other->GetSize(&w, &h);
938 return h;
939 }
940 }
941 case wxCentreX:
942 {
943 wxLayoutConstraints *constr = other->GetConstraints();
944 // If no constraints, it means the window is not dependent
945 // on anything, and therefore we know its value immediately
946 if (constr)
947 {
948 if (constr->centreX.GetDone())
949 return constr->centreX.GetValue();
950 else
951 return -1;
952 }
953 else
954 {
955 int x, y, w, h;
956 other->GetPosition(&x, &y);
957 other->GetSize(&w, &h);
958 return (int)(x + (w/2));
959 }
960 }
961 case wxCentreY:
962 {
963 wxLayoutConstraints *constr = other->GetConstraints();
964 // If no constraints, it means the window is not dependent
965 // on anything, and therefore we know its value immediately
966 if (constr)
967 {
968 if (constr->centreY.GetDone())
969 return constr->centreY.GetValue();
970 else
971 return -1;
972 }
973 else
974 {
975 int x, y, w, h;
976 other->GetPosition(&x, &y);
977 other->GetSize(&w, &h);
978 return (int)(y + (h/2));
979 }
980 }
981 default:
982 break;
983 }
984 return -1;
985 }
986
987 wxLayoutConstraints::wxLayoutConstraints()
988 {
989 left.SetEdge(wxLeft);
990 top.SetEdge(wxTop);
991 right.SetEdge(wxRight);
992 bottom.SetEdge(wxBottom);
993 centreX.SetEdge(wxCentreX);
994 centreY.SetEdge(wxCentreY);
995 width.SetEdge(wxWidth);
996 height.SetEdge(wxHeight);
997 }
998
999 wxLayoutConstraints::~wxLayoutConstraints()
1000 {
1001 }
1002
1003 bool wxLayoutConstraints::SatisfyConstraints(wxWindowBase *win, int *nChanges)
1004 {
1005 int noChanges = 0;
1006
1007 bool done = width.GetDone();
1008 bool newDone = (done ? TRUE : width.SatisfyConstraint(this, win));
1009 if (newDone != done)
1010 noChanges ++;
1011
1012 done = height.GetDone();
1013 newDone = (done ? TRUE : height.SatisfyConstraint(this, win));
1014 if (newDone != done)
1015 noChanges ++;
1016
1017 done = left.GetDone();
1018 newDone = (done ? TRUE : left.SatisfyConstraint(this, win));
1019 if (newDone != done)
1020 noChanges ++;
1021
1022 done = top.GetDone();
1023 newDone = (done ? TRUE : top.SatisfyConstraint(this, win));
1024 if (newDone != done)
1025 noChanges ++;
1026
1027 done = right.GetDone();
1028 newDone = (done ? TRUE : right.SatisfyConstraint(this, win));
1029 if (newDone != done)
1030 noChanges ++;
1031
1032 done = bottom.GetDone();
1033 newDone = (done ? TRUE : bottom.SatisfyConstraint(this, win));
1034 if (newDone != done)
1035 noChanges ++;
1036
1037 done = centreX.GetDone();
1038 newDone = (done ? TRUE : centreX.SatisfyConstraint(this, win));
1039 if (newDone != done)
1040 noChanges ++;
1041
1042 done = centreY.GetDone();
1043 newDone = (done ? TRUE : centreY.SatisfyConstraint(this, win));
1044 if (newDone != done)
1045 noChanges ++;
1046
1047 *nChanges = noChanges;
1048
1049 return AreSatisfied();
1050 }
1051
1052 /*
1053 * Main constrained layout algorithm. Look at all the child
1054 * windows, and their constraints (if any).
1055 * The idea is to keep iterating through the constraints
1056 * until all left, right, bottom and top edges, and widths and heights,
1057 * are known (or no change occurs and we've failed to resolve all
1058 * constraints).
1059 *
1060 * If the user has not specified a dimension or edge, it will be
1061 * be calculated from the other known values. E.g. If we know
1062 * the right hand edge and the left hand edge, we now know the width.
1063 * The snag here is that this means we must specify absolute dimensions
1064 * twice (in constructor and in constraint), if we wish to use the
1065 * constraint notation to just set the position, for example.
1066 * Otherwise, if we only set ONE edge and no dimension, it would never
1067 * find the other edge.
1068 *
1069 * Algorithm:
1070
1071 Mark all constraints as not done.
1072
1073 iterations = 0;
1074 until no change or iterations >= max iterations
1075 For each child:
1076 {
1077 Calculate all constraints
1078 }
1079 iterations ++;
1080
1081 For each child
1082 Set each calculated position and size
1083
1084 */
1085
1086 #if WXWIN_COMPATIBILITY
1087 bool wxOldDoLayout(wxWindowBase *win)
1088 {
1089 // Make sure this isn't called recursively from below
1090 static wxList doneSoFar;
1091
1092 if (doneSoFar.Member(win))
1093 return TRUE;
1094
1095 doneSoFar.Append(win);
1096
1097 wxNode *node = win->GetChildren().First();
1098 while (node)
1099 {
1100 wxWindowBase *child = (wxWindowBase *)node->Data();
1101 wxLayoutConstraints *constr = child->GetConstraints();
1102 if (constr)
1103 {
1104 constr->left.SetDone(FALSE);
1105 constr->top.SetDone(FALSE);
1106 constr->right.SetDone(FALSE);
1107 constr->bottom.SetDone(FALSE);
1108 constr->width.SetDone(FALSE);
1109 constr->height.SetDone(FALSE);
1110 constr->centreX.SetDone(FALSE);
1111 constr->centreY.SetDone(FALSE);
1112 }
1113 node = node->Next();
1114 }
1115 int noIterations = 0;
1116 int maxIterations = 500;
1117 int noChanges = 1;
1118
1119 while ((noChanges > 0) && (noIterations < maxIterations))
1120 {
1121 noChanges = 0;
1122 wxNode *node = win->GetChildren().First();
1123 while (node)
1124 {
1125 wxWindowBase *child = (wxWindowBase *)node->Data();
1126 wxLayoutConstraints *constr = child->GetConstraints();
1127 if (constr)
1128 {
1129 int tempNoChanges = 0;
1130 (void)constr->SatisfyConstraints(child, &tempNoChanges);
1131 noChanges += tempNoChanges;
1132 }
1133 node = node->Next();
1134 }
1135 noIterations ++;
1136 }
1137 /*
1138 // Would be nice to have a test here to see _which_ constraint(s)
1139 // failed, so we can print a specific diagnostic message.
1140 if (noFailures > 0)
1141 {
1142 wxDebugMsg(_("wxWindowBase::Layout() failed.\n"));
1143 }
1144 */
1145 // Now set the sizes and positions of the children, and
1146 // recursively call Layout().
1147 node = win->GetChildren().First();
1148 while (node)
1149 {
1150 wxWindowBase *child = (wxWindowBase *)node->Data();
1151 wxLayoutConstraints *constr = child->GetConstraints();
1152 if (constr && constr->left.GetDone() && constr->right.GetDone() &&
1153 constr->width.GetDone() && constr->height.GetDone())
1154 {
1155 int x = constr->left.GetValue();
1156 int y = constr->top.GetValue();
1157 int w = constr->width.GetValue();
1158 int h = constr->height.GetValue();
1159
1160 // If we don't want to resize this window, just move it...
1161 if ((constr->width.GetRelationship() != wxAsIs) ||
1162 (constr->height.GetRelationship() != wxAsIs))
1163 {
1164 // _Should_ call Layout() recursively.
1165 child->SetSize(x, y, w, h);
1166 }
1167 else
1168 {
1169 child->Move(x, y);
1170 }
1171 }
1172 else
1173 child->Layout();
1174 node = node->Next();
1175 }
1176 doneSoFar.DeleteObject(win);
1177
1178 return TRUE;
1179 }
1180 #endif // WXWIN_COMPATIBILITY
1181
1182 wxSizer::wxSizer()
1183 {
1184 sizerBehaviour = wxSizerNone;
1185 borderX = 2;
1186 borderY = 2;
1187 sizerX = 0;
1188 sizerY = 0;
1189 sizerWidth = 0;
1190 sizerHeight = 0;
1191 }
1192
1193 wxSizer::wxSizer(wxWindowBase *parent, wxSizerBehaviour behav)
1194 {
1195 Create(parent, behav);
1196 }
1197
1198 bool wxSizer::Create(wxWindowBase *parent, wxSizerBehaviour behav)
1199 {
1200 sizerBehaviour = behav;
1201 borderX = 2;
1202 borderY = 2;
1203 m_sizerParent = parent;
1204 sizerX = 0;
1205 sizerY = 0;
1206 sizerWidth = 0;
1207 sizerHeight = 0;
1208
1209 // A normal window can have just one top-level sizer
1210 // associated with it.
1211 if (!parent->IsKindOf(CLASSINFO(wxSizer)))
1212 {
1213 parent->SetSizer(this);
1214 }
1215 else
1216 ((wxSizer *)parent)->AddSizerChild(this);
1217
1218 switch (sizerBehaviour)
1219 {
1220 case wxSizerExpand:
1221 {
1222 // Defines a set of constraints
1223 // to expand the sizer to fit the parent window
1224 wxLayoutConstraints *c = new wxLayoutConstraints;
1225
1226 c->left.SameAs(parent, wxLeft, 0);
1227 c->top.SameAs(parent, wxTop, 0);
1228 c->right.SameAs(parent, wxRight, 0);
1229 c->bottom.SameAs(parent, wxBottom, 0);
1230
1231 SetConstraints(c);
1232 break;
1233 }
1234 case wxSizerShrink:
1235 case wxSizerNone:
1236 default:
1237 {
1238 }
1239 }
1240 return TRUE;
1241 }
1242
1243 wxSizer::~wxSizer()
1244 {
1245 // Remove all children without deleting them,
1246 // or ~wxbWindow will delete proper windows _twice_
1247 wxNode *node = GetChildren().First();
1248 while (node)
1249 {
1250 wxNode *next = node->Next();
1251 wxWindowBase *win = (wxWindowBase *)node->Data();
1252 if (!win->IsKindOf(CLASSINFO(wxSizer)))
1253 {
1254 delete node;
1255 win->SetSizerParent((wxWindowBase *) NULL);
1256 }
1257 else
1258 {
1259 RemoveSizerChild(win);
1260 delete win;
1261 }
1262 node = next;
1263 }
1264
1265 if (m_sizerParent) // && !m_sizerParent->IsKindOf(CLASSINFO(wxSizer)))
1266 {
1267 m_sizerParent->SetSizer((wxSizer *) NULL);
1268 m_sizerParent = (wxWindowBase *) NULL;
1269 }
1270
1271 }
1272
1273 void wxSizer::SetBorder(int x, int y)
1274 {
1275 borderX = x;
1276 borderY = y;
1277 /* No: the margin is for inside, not outside (expansion)
1278
1279 if ( GetConstraints() )
1280 {
1281 GetConstraints()->left.SetMargin(x);
1282 GetConstraints()->right.SetMargin(x);
1283 GetConstraints()->top.SetMargin(y);
1284 GetConstraints()->bottom.SetMargin(y);
1285 }
1286 */
1287
1288 }
1289
1290 void wxSizer::AddSizerChild(wxWindowBase *child)
1291 {
1292 child->SetSizerParent(this);
1293 GetChildren().Append(child);
1294
1295 // Add some constraints for the purpose of storing
1296 // the relative position of the window/sizer
1297 // during layout calculations.
1298 if (!child->GetConstraints())
1299 {
1300 wxLayoutConstraints *c = new wxLayoutConstraints;
1301 c->left.AsIs();
1302 c->top.AsIs();
1303 c->width.AsIs();
1304 c->height.AsIs();
1305 int w, h;
1306 child->GetSize(&w, &h);
1307 c->width.SetValue(w);
1308 c->height.SetValue(h);
1309
1310 child->SetConstraints(c);
1311 }
1312 }
1313
1314 void wxSizer::RemoveSizerChild(wxWindowBase *child)
1315 {
1316 GetChildren().DeleteObject(child);
1317 }
1318
1319 void wxSizer::DoSetSize(int x, int y, int w, int h, int WXUNUSED(flags))
1320 {
1321 wxLayoutConstraints *constr = GetConstraints();
1322 if (x != -1)
1323 {
1324 sizerX = x;
1325 if (constr)
1326 constr->left.SetValue(x);
1327 }
1328 if (y != -1)
1329 {
1330 sizerY = y;
1331 if (constr)
1332 constr->top.SetValue(y);
1333 }
1334 if (w != -1)
1335 {
1336 sizerWidth = w;
1337 if (constr)
1338 constr->width.SetValue(w);
1339 }
1340 if (h != -1)
1341 {
1342 sizerHeight = h;
1343 if (constr)
1344 constr->height.SetValue(h);
1345 }
1346 }
1347
1348 void wxSizer::DoGetSize(int *w, int *h) const
1349 {
1350 *w = sizerWidth;
1351 *h = sizerHeight;
1352 }
1353
1354 void wxSizer::DoGetPosition(int *x, int *y) const
1355 {
1356 *x = sizerX;
1357 *y = sizerY;
1358 }
1359
1360 bool wxSizer::LayoutPhase1(int *noChanges)
1361 {
1362 *noChanges = 0;
1363 switch (sizerBehaviour)
1364 {
1365 case wxSizerExpand:
1366 {
1367 if (!m_sizerParent)
1368 {
1369 wxMessageBox(_("wxExpandSizer has no parent!"), _("Sizer error"), wxOK);
1370 return TRUE;
1371 }
1372
1373 // Set the size to fill the parent client area
1374 int pw, ph;
1375 m_sizerParent->GetClientSize(&pw, &ph);
1376 SetSize(GetBorderX(), GetBorderY(), pw - 2*GetBorderX(), ph - 2*GetBorderY());
1377 wxLayoutConstraints *constr = GetConstraints();
1378
1379 // Fill in the constraints
1380 if (constr)
1381 {
1382 constr->left.SetValue(0); constr->left.SetDone(TRUE);
1383 constr->top.SetValue(0); constr->right.SetDone(TRUE);
1384 constr->width.SetValue(pw); constr->width.SetDone(TRUE);
1385 constr->height.SetValue(ph); constr->height.SetDone(TRUE);
1386 }
1387
1388 return TRUE;
1389 break;
1390 }
1391 case wxSizerShrink:
1392 {
1393 wxLayoutConstraints *constr = GetConstraints();
1394
1395 if (constr)
1396 {
1397 // Force the constraint to have as-is width and height
1398 // if we're in shrink-to-fit mode, because if left unconstrained,
1399 // SatisfyConstraints will fail. The shrink-to-fit option
1400 // essentially specifies the width and height as 'whatever I calculate'.
1401 constr->width.AsIs();
1402 constr->height.AsIs();
1403 }
1404 DoPhase(1);
1405 DoPhase(2);
1406 // Find the bounding box and set own size
1407 int maxX = 0;
1408 int maxY = 0;
1409
1410 wxNode *node = GetChildren().First();
1411 while (node)
1412 {
1413 int x, y, width, height;
1414 wxWindowBase *win = (wxWindowBase *)node->Data();
1415 win->GetSizeConstraint(&width, &height);
1416 win->GetPositionConstraint(&x, &y);
1417 if ((x+width) > maxX)
1418 maxX = (x + width);
1419 if ((y+height) > maxY)
1420 maxY = (y + height);
1421
1422 node = node->Next();
1423 }
1424 SetSize(GetBorderX(), GetBorderY(), maxX, maxY);
1425
1426 // If this is the only sizer for the parent, size the parent to this sizer.
1427 if ( m_sizerParent && (m_sizerParent->GetSizer() == this) )
1428 m_sizerParent->SetClientSize(maxX + 2*GetBorderX(), maxY + 2*GetBorderY());
1429
1430 return TRUE;
1431 break;
1432 }
1433 case wxSizerNone:
1434 {
1435 wxLayoutConstraints *constr = GetConstraints();
1436 if (constr)
1437 {
1438 bool success = constr->SatisfyConstraints(this, noChanges);
1439 if (success)
1440 {
1441 int x = constr->left.GetValue();
1442 int y = constr->top.GetValue();
1443 int w = constr->width.GetValue();
1444 int h = constr->height.GetValue();
1445 SetSize(x, y, w, h);
1446 }
1447 return success;
1448 }
1449 else
1450 return TRUE;
1451 break;
1452 }
1453 }
1454 return TRUE;
1455
1456 }
1457
1458 bool wxSizer::LayoutPhase2(int *noChanges)
1459 {
1460 *noChanges = 0;
1461
1462 switch (sizerBehaviour)
1463 {
1464 case wxSizerExpand:
1465 {
1466 // Layout children
1467 DoPhase(1);
1468 DoPhase(2);
1469 return TRUE;
1470 }
1471 case wxSizerShrink:
1472 {
1473 wxLayoutConstraints *constr = GetConstraints();
1474 if (constr)
1475 {
1476 bool success = constr->SatisfyConstraints(this, noChanges);
1477 if (success)
1478 {
1479 int x = constr->left.GetValue();
1480 int y = constr->top.GetValue();
1481 Move(x, y);
1482 }
1483 return success;
1484 }
1485 break;
1486 }
1487 case wxSizerNone:
1488 {
1489 // Layout children
1490 DoPhase(1);
1491 DoPhase(2);
1492
1493 // Is this a dumb fix for lack of constraint evaluation?
1494 wxLayoutConstraints *constr = GetConstraints();
1495 if (constr)
1496 {
1497 bool success = constr->SatisfyConstraints(this, noChanges);
1498 if (success)
1499 {
1500 int x = constr->left.GetValue();
1501 int y = constr->top.GetValue();
1502 int w = constr->width.GetValue();
1503 int h = constr->height.GetValue();
1504 SetSize(x, y, w, h);
1505 }
1506 return success;
1507 }
1508 else
1509 return TRUE;
1510 }
1511 }
1512 return TRUE;
1513 }
1514
1515 /*
1516 * wxRowColSizer
1517 */
1518
1519 wxRowColSizer::wxRowColSizer()
1520 {
1521 rowOrCol = TRUE;
1522 rowOrColSize = 20;
1523 xSpacing = 2;
1524 ySpacing = 2;
1525 }
1526
1527 wxRowColSizer::wxRowColSizer(wxWindowBase *parent, bool rc, int n, wxSizerBehaviour behav)
1528 {
1529 Create(parent, rc, n, behav);
1530 }
1531
1532 bool wxRowColSizer::Create(wxWindowBase *parent, bool rc, int n, wxSizerBehaviour behav)
1533 {
1534 wxSizer::Create(parent, behav);
1535
1536 rowOrCol = rc;
1537 rowOrColSize = n;
1538 xSpacing = 2;
1539 ySpacing = 2;
1540
1541 return TRUE;
1542 }
1543
1544 wxRowColSizer::~wxRowColSizer()
1545 {
1546 }
1547
1548 bool wxRowColSizer::LayoutPhase1(int *noChanges)
1549 {
1550 *noChanges = 0;
1551 wxLayoutConstraints *constr = GetConstraints();
1552
1553 if (constr)
1554 {
1555 // Force the constraint to have as-is width and height
1556 // if we're in shrink-to-fit mode, because if left unconstrained,
1557 // SatisfyConstraints will fail. The shrink-to-fit option
1558 // essentially specifies the width and height as 'whatever I calculate'.
1559 if (sizerBehaviour == wxSizerShrink)
1560 {
1561 constr->width.AsIs();
1562 constr->height.AsIs();
1563 }
1564
1565 // Only evaluate the constraints FIRST if we're NOT
1566 // in shrink-to-fit mode, i.e. we want to size the rowcol
1567 // first, then lay the children out in the space we've calculated.
1568 if (sizerBehaviour != wxSizerShrink)
1569 {
1570 bool success = constr->SatisfyConstraints(this, noChanges);
1571 if (success)
1572 {
1573 int x = constr->left.GetValue();
1574 int y = constr->top.GetValue();
1575 int w = constr->width.GetValue();
1576 int h = constr->height.GetValue();
1577 SetSize(x, y, w, h);
1578 }
1579 else
1580 return FALSE;
1581
1582 // Continue to do the rest of the phase when the constraints have been
1583 // satisfied, i.e. we're on the last iteration of phase 1 and
1584 // can now do the actual rowcol laying out.
1585 }
1586 }
1587
1588 // If we ARE in shrink-to-fit mode, we must now
1589 // calculate the child sizes BEFORE laying out in rows or columns.
1590 if (sizerBehaviour == wxSizerShrink)
1591 {
1592 DoPhase(1);
1593 DoPhase(2);
1594
1595 // WILL THE WINDOW BE SIZED CORRECTLY AT THIS POINT?
1596 // CHECK CONSTRAINTS IF ANY...
1597 int noRows = 0;
1598 int noCols = 0;
1599 int currentX = borderX;
1600 int currentY = borderY;
1601 int maxX = currentX;
1602 int maxY = currentY;
1603
1604 wxNode *node = GetChildren().First();
1605 while (node)
1606 {
1607 wxWindowBase *win = (wxWindowBase *)node->Data();
1608 int childWidth, childHeight;
1609 if (win->GetConstraints() &&
1610 win->GetConstraints()->width.GetDone() &&
1611 win->GetConstraints()->height.GetDone())
1612 {
1613 childWidth = win->GetConstraints()->width.GetValue();
1614 childHeight = win->GetConstraints()->height.GetValue();
1615 }
1616 else
1617 win->GetSize(&childWidth, &childHeight);
1618
1619 win->MoveConstraint(currentX, currentY);
1620
1621 if ((currentX + childWidth) > maxX)
1622 maxX = (currentX + childWidth);
1623 if ((currentY + childHeight) > maxY)
1624 maxY = (currentY + childHeight);
1625
1626 if (rowOrCol)
1627 {
1628 currentX += childWidth + xSpacing;
1629 noCols ++;
1630
1631 // Reset to start of row
1632 if (noCols == rowOrColSize)
1633 {
1634 currentX = borderX;
1635 currentY += childHeight + ySpacing;
1636 noCols = 0;
1637 }
1638 }
1639 else
1640 {
1641 currentY += childHeight + ySpacing;
1642 noRows ++;
1643
1644 // Reset to start of col
1645 if (noRows == rowOrColSize)
1646 {
1647 currentY = borderY;
1648 currentX += childWidth + xSpacing;
1649 noRows = 0;
1650 }
1651 }
1652
1653 node = node->Next();
1654 }
1655 maxX += borderX;
1656 maxY += borderY;
1657
1658 SetSize(-1, -1, maxX, maxY);
1659 }
1660 return TRUE;
1661 }
1662
1663 bool wxRowColSizer::LayoutPhase2(int *noChanges)
1664 {
1665 *noChanges = 0;
1666
1667 // If shrink-to-fit, it's only at Phase 2 that we know the size of
1668 // the wxRowColSizer, and now we can evaluate the
1669 // constraints and pass result back up to parent.
1670 // This implements a depth-first strategy
1671 if (sizerBehaviour == wxSizerShrink)
1672 {
1673 wxLayoutConstraints *constr = GetConstraints();
1674 if (constr)
1675 {
1676 bool success = constr->SatisfyConstraints(this, noChanges);
1677 if (success)
1678 {
1679 int x = constr->left.GetValue();
1680 int y = constr->top.GetValue();
1681 Move(x, y);
1682 }
1683 return success;
1684 }
1685 else return TRUE;
1686 }
1687 else
1688 {
1689 // Lay out the children: breadth-first strategy.
1690 DoPhase(1);
1691 DoPhase(2);
1692
1693 // Space them
1694 }
1695 return TRUE;
1696 }
1697
1698
1699 /*
1700 * wxSpacingSizer
1701 */
1702
1703 wxSpacingSizer::wxSpacingSizer()
1704 {
1705 }
1706
1707 wxSpacingSizer::wxSpacingSizer(wxWindowBase *parent)
1708 {
1709 Create(parent);
1710 }
1711
1712 wxSpacingSizer::wxSpacingSizer(wxWindowBase *parent, wxRelationship rel, wxWindowBase *other, int spacing)
1713 {
1714 Create(parent, rel, other, spacing);
1715 }
1716
1717 bool wxSpacingSizer::Create(wxWindowBase *parent)
1718 {
1719 wxSizer::Create(parent);
1720 return TRUE;
1721 }
1722
1723 bool wxSpacingSizer::Create(wxWindowBase *parent, wxRelationship rel, wxWindowBase *other, int spacing)
1724 {
1725 wxLayoutConstraints *c = new wxLayoutConstraints;
1726
1727 wxSizer::Create(parent);
1728
1729 switch ( rel )
1730 {
1731 case wxLeftOf :
1732 c->width.Absolute (spacing);
1733 c->top.SameAs (other, wxTop);
1734 c->bottom.SameAs (other, wxBottom);
1735 c->right.LeftOf (other);
1736 break;
1737 case wxRightOf :
1738 c->width.Absolute (spacing);
1739 c->top.SameAs (other, wxTop);
1740 c->bottom.SameAs (other, wxBottom);
1741 c->left.RightOf (other);
1742 break;
1743 case wxBelow :
1744 c->height.Absolute (spacing);
1745 c->left.SameAs (other, wxLeft);
1746 c->right.SameAs (other, wxRight);
1747 c->top.Below (other);
1748 break;
1749 case wxAbove :
1750 c->height.Absolute (spacing);
1751 c->left.SameAs (other, wxLeft);
1752 c->right.SameAs (other, wxRight);
1753 c->bottom.Above (other);
1754 break;
1755
1756 default :
1757 break;
1758 }
1759 SetConstraints(c);
1760
1761 return TRUE;
1762 }
1763
1764 wxSpacingSizer::~wxSpacingSizer()
1765 {
1766 }
1767
1768 #endif // wxUSE_CONSTRAINTS