]> git.saurik.com Git - wxWidgets.git/blame - src/common/wincmn.cpp
Preserve -1 for dilog unit conversions
[wxWidgets.git] / src / common / wincmn.cpp
CommitLineData
7ec1983b 1/////////////////////////////////////////////////////////////////////////////
f03fc89f 2// Name: common/window.cpp
7ec1983b
VZ
3// Purpose: common (to all ports) wxWindow functions
4// Author: Julian Smart, Vadim Zeitlin
5// Modified by:
6// Created: 13/07/98
7// RCS-ID: $Id$
f03fc89f 8// Copyright: (c) wxWindows team
7ec1983b
VZ
9// Licence: wxWindows license
10/////////////////////////////////////////////////////////////////////////////
11
f03fc89f
VZ
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
f701d7ab 19
58654ed0
VZ
20#ifdef __GNUG__
21 #pragma implementation "windowbase.h"
22#endif
23
341287bf
JS
24// For compilers that support precompilation, includes "wx.h".
25#include "wx/wxprec.h"
26
f701d7ab 27#ifdef __BORLANDC__
f03fc89f 28 #pragma hdrstop
f701d7ab
JS
29#endif
30
f03fc89f
VZ
31#ifndef WX_PRECOMP
32 #include "wx/string.h"
33 #include "wx/log.h"
34 #include "wx/intl.h"
35 #include "wx/frame.h"
36 #include "wx/defs.h"
37 #include "wx/window.h"
38 #include "wx/checkbox.h"
39 #include "wx/radiobut.h"
40 #include "wx/settings.h"
41 #include "wx/dialog.h"
42#endif //WX_PRECOMP
43
44#if wxUSE_CONSTRAINTS
45 #include "wx/layout.h"
46#endif // wxUSE_CONSTRAINTS
47
48#if wxUSE_DRAG_AND_DROP
49 #include "wx/dnd.h"
50#endif // wxUSE_DRAG_AND_DROP
51
52#if wxUSE_TOOLTIPS
53 #include "wx/tooltip.h"
54#endif // wxUSE_TOOLTIPS
55
789295bf
VZ
56#if wxUSE_CARET
57 #include "wx/caret.h"
58#endif // wxUSE_CARET
59
f03fc89f
VZ
60// ----------------------------------------------------------------------------
61// static data
62// ----------------------------------------------------------------------------
63
42e69d6b 64int wxWindowBase::ms_lastControlId = -200;
f03fc89f
VZ
65
66IMPLEMENT_ABSTRACT_CLASS(wxWindowBase, wxEvtHandler)
67
68// ----------------------------------------------------------------------------
69// event table
70// ----------------------------------------------------------------------------
71
72BEGIN_EVENT_TABLE(wxWindowBase, wxEvtHandler)
73 EVT_SYS_COLOUR_CHANGED(wxWindowBase::OnSysColourChanged)
74 EVT_INIT_DIALOG(wxWindowBase::OnInitDialog)
75END_EVENT_TABLE()
76
77// ============================================================================
78// implementation of the common functionality of the wxWindow class
79// ============================================================================
80
81// ----------------------------------------------------------------------------
82// initialization
83// ----------------------------------------------------------------------------
84
85// the default initialization
86void wxWindowBase::InitBase()
87{
88 // no window yet, no parent nor children
f03fc89f
VZ
89 m_parent = (wxWindow *)NULL;
90 m_windowId = -1;
91 m_children.DeleteContents( FALSE ); // don't auto delete node data
92
93 // no constraints on the minimal window size
94 m_minWidth =
95 m_minHeight =
96 m_maxWidth =
97 m_maxHeight = -1;
98
99 // window is created enabled but it's not visible yet
100 m_isShown = FALSE;
101 m_isEnabled = TRUE;
102
103 // no client data
104 m_clientObject = (wxClientData *)NULL;
105 m_clientData = NULL;
106
107 // the default event handler is just this window
108 m_eventHandler = this;
109
88ac883a 110#if wxUSE_VALIDATORS
f03fc89f
VZ
111 // no validator
112 m_windowValidator = (wxValidator *) NULL;
88ac883a 113#endif // wxUSE_VALIDATORS
f03fc89f
VZ
114
115 // use the system default colours
116 wxSystemSettings settings;
117
118 m_backgroundColour = settings.GetSystemColour(wxSYS_COLOUR_BTNFACE);
119 m_foregroundColour = *wxBLACK; // TODO take this from sys settings too?
120 m_font = *wxSWISS_FONT; // and this?
121
122 // no style bits
123 m_windowStyle = 0;
124
125 // an optimization for the event processing: checking this flag is much
126 // faster than using IsKindOf(CLASSINFO(wxWindow))
127 m_isWindow = TRUE;
128
129#if wxUSE_CONSTRAINTS
130 // no constraints whatsoever
131 m_constraints = (wxLayoutConstraints *) NULL;
132 m_constraintsInvolvedIn = (wxWindowList *) NULL;
133 m_windowSizer = (wxSizer *) NULL;
134 m_sizerParent = (wxWindowBase *) NULL;
135 m_autoLayout = FALSE;
136#endif // wxUSE_CONSTRAINTS
137
138#if wxUSE_DRAG_AND_DROP
139 m_dropTarget = (wxDropTarget *)NULL;
140#endif // wxUSE_DRAG_AND_DROP
141
142#if wxUSE_TOOLTIPS
143 m_tooltip = (wxToolTip *)NULL;
144#endif // wxUSE_TOOLTIPS
789295bf
VZ
145
146#if wxUSE_CARET
147 m_caret = (wxCaret *)NULL;
148#endif // wxUSE_CARET
f03fc89f
VZ
149}
150
151// common part of window creation process
152bool wxWindowBase::CreateBase(wxWindowBase *parent,
153 wxWindowID id,
74e3313b
VZ
154 const wxPoint& WXUNUSED(pos),
155 const wxSize& WXUNUSED(size),
f03fc89f
VZ
156 long style,
157 const wxString& name)
158{
159 // m_isWindow is set to TRUE in wxWindowBase::Init() as well as many other
160 // member variables - check that it has been called (will catch the case
161 // when a new ctor is added which doesn't call InitWindow)
162 wxASSERT_MSG( m_isWindow, _T("Init() must have been called before!") );
163
164 // generate a new id if the user doesn't care about it
69418a8e 165 m_windowId = id == -1 ? NewControlId() : id;
f03fc89f
VZ
166
167 SetName(name);
168 SetWindowStyleFlag(style);
169 SetParent(parent);
170
171 return TRUE;
172}
173
174// ----------------------------------------------------------------------------
175// destruction
176// ----------------------------------------------------------------------------
177
178// common clean up
179wxWindowBase::~wxWindowBase()
180{
181 // FIXME if these 2 cases result from programming errors in the user code
182 // we should probably assert here instead of silently fixing them
183
184 // Just in case the window has been Closed, but we're then deleting
185 // immediately: don't leave dangling pointers.
186 wxPendingDelete.DeleteObject(this);
187
188 // Just in case we've loaded a top-level window via LoadNativeDialog but
189 // we weren't a dialog class
190 wxTopLevelWindows.DeleteObject(this);
a23fd0e1 191
9111db68 192 wxASSERT_MSG( GetChildren().GetCount() == 0, _T("children not destroyed") );
f03fc89f 193
319fefa9
VZ
194 // make sure that there are no dangling pointers left pointing to us
195 wxPanel *panel = wxDynamicCast(GetParent(), wxPanel);
196 if ( panel )
197 {
198 if ( panel->GetLastFocus() == this )
199 {
200 panel->SetLastFocus((wxWindow *)NULL);
201 }
202 }
203
789295bf
VZ
204#if wxUSE_CARET
205 if ( m_caret )
206 delete m_caret;
207#endif // wxUSE_CARET
208
88ac883a 209#if wxUSE_VALIDATORS
f03fc89f
VZ
210 if ( m_windowValidator )
211 delete m_windowValidator;
88ac883a 212#endif // wxUSE_VALIDATORS
f03fc89f
VZ
213
214 if ( m_clientObject )
215 delete m_clientObject;
216
217#if wxUSE_CONSTRAINTS
218 // Have to delete constraints/sizer FIRST otherwise sizers may try to look
219 // at deleted windows as they delete themselves.
220 DeleteRelatedConstraints();
221
222 if ( m_constraints )
223 {
224 // This removes any dangling pointers to this window in other windows'
225 // constraintsInvolvedIn lists.
226 UnsetConstraints(m_constraints);
227 delete m_constraints;
228 m_constraints = NULL;
229 }
230
231 if ( m_windowSizer )
232 delete m_windowSizer;
233
234 // If this is a child of a sizer, remove self from parent
235 if ( m_sizerParent )
236 m_sizerParent->RemoveChild(this);
237#endif // wxUSE_CONSTRAINTS
238
239#if wxUSE_DRAG_AND_DROP
240 if ( m_dropTarget )
241 delete m_dropTarget;
242#endif // wxUSE_DRAG_AND_DROP
243
244#if wxUSE_TOOLTIPS
245 if ( m_tooltip )
246 delete m_tooltip;
247#endif // wxUSE_TOOLTIPS
248}
249
250bool wxWindowBase::Destroy()
251{
252 delete this;
253
254 return TRUE;
255}
256
257bool wxWindowBase::Close(bool force)
258{
259 wxCloseEvent event(wxEVT_CLOSE_WINDOW, m_windowId);
260 event.SetEventObject(this);
261#if WXWIN_COMPATIBILITY
262 event.SetForce(force);
263#endif // WXWIN_COMPATIBILITY
264 event.SetCanVeto(!force);
265
266 // return FALSE if window wasn't closed because the application vetoed the
267 // close event
268 return GetEventHandler()->ProcessEvent(event) && !event.GetVeto();
269}
270
271bool wxWindowBase::DestroyChildren()
272{
273 wxWindowList::Node *node;
a23fd0e1 274 for ( ;; )
f03fc89f 275 {
a23fd0e1
VZ
276 // we iterate until the list becomes empty
277 node = GetChildren().GetFirst();
278 if ( !node )
279 break;
280
f03fc89f 281 wxWindow *child = node->GetData();
a23fd0e1 282
9111db68 283 wxASSERT_MSG( child, _T("children list contains empty nodes") );
a23fd0e1 284
eb082a08 285 delete child;
a23fd0e1
VZ
286
287 wxASSERT_MSG( !GetChildren().Find(child),
9111db68 288 _T("child didn't remove itself using RemoveChild()") );
f03fc89f
VZ
289 }
290
291 return TRUE;
292}
293
294// ----------------------------------------------------------------------------
295// centre/fit the window
296// ----------------------------------------------------------------------------
297
298// centre the window with respect to its parent in either (or both) directions
299void wxWindowBase::Centre(int direction)
300{
301 int widthParent, heightParent;
302
303 wxWindow *parent = GetParent();
304 if ( parent )
305 {
306 parent->GetClientSize(&widthParent, &heightParent);
307 }
308 else
309 {
310 // centre with respect to the whole screen
311 wxDisplaySize(&widthParent, &heightParent);
312 }
313
314 int width, height;
315 GetSize(&width, &height);
316
317 int new_x = -1,
318 new_y = -1;
319
320 if ( direction & wxHORIZONTAL )
321 new_x = (widthParent - width)/2;
322
323 if ( direction & wxVERTICAL )
324 new_y = (heightParent - height)/2;
325
326 Move(new_x, new_y);
327}
328
7631a292
RD
329// Center TopLevel windows over thier parent instead of the whole screen
330void wxWindowBase::CentreOnParent(int direction)
331{
332 wxPoint ppos;
333 wxSize psze;
334 wxSize wsze;
335 wxWindow* parent = GetParent();
336 int x, y;
337
338 if (!parent || !IsTopLevel()) {
339 Centre(direction);
340 return;
341 }
342
343 psze = parent->GetSize();
344 ppos = parent->ClientToScreen(wxPoint(0,0));
345 wsze = GetSize();
346
347 x = y = -1;
348
349 if (direction == wxBOTH || direction == wxHORIZONTAL)
350 x = ppos.x + (psze.x - wsze.x)/2;
351 if (direction == wxBOTH || direction == wxVERTICAL)
352 y = ppos.y + (psze.y - wsze.y)/2;
353
354 Move(x, y);
355}
356
f03fc89f
VZ
357// fits the window around the children
358void wxWindowBase::Fit()
359{
360 int maxX = 0,
361 maxY = 0;
362
bfac8499
VZ
363 for ( wxWindowList::Node *node = GetChildren().GetFirst();
364 node;
365 node = node->GetNext() )
f03fc89f
VZ
366 {
367 wxWindow *win = node->GetData();
34636400 368 if ( win->IsTopLevel() )
42e69d6b 369 {
34636400 370 // dialogs and frames lie in different top level windows - don't
42e69d6b
VZ
371 // deal with them here
372 continue;
373 }
374
f03fc89f
VZ
375 int wx, wy, ww, wh;
376 win->GetPosition(&wx, &wy);
377 win->GetSize(&ww, &wh);
378 if ( wx + ww > maxX )
379 maxX = wx + ww;
380 if ( wy + wh > maxY )
381 maxY = wy + wh;
f03fc89f
VZ
382 }
383
384 // leave a margin
385 SetClientSize(maxX + 7, maxY + 14);
386}
387
388// set the min/max size of the window
389
390void wxWindowBase::SetSizeHints(int minW, int minH,
391 int maxW, int maxH,
392 int WXUNUSED(incW), int WXUNUSED(incH))
393{
394 m_minWidth = minW;
395 m_maxWidth = maxW;
396 m_minHeight = minH;
397 m_maxHeight = maxH;
398}
399
400// ----------------------------------------------------------------------------
401// show/hide/enable/disable the window
402// ----------------------------------------------------------------------------
403
404bool wxWindowBase::Show(bool show)
405{
406 if ( show != m_isShown )
407 {
408 m_isShown = show;
409
410 return TRUE;
411 }
412 else
413 {
414 return FALSE;
415 }
416}
417
418bool wxWindowBase::Enable(bool enable)
419{
420 if ( enable != m_isEnabled )
421 {
422 m_isEnabled = enable;
423
424 return TRUE;
425 }
426 else
427 {
428 return FALSE;
429 }
430}
34636400
VZ
431// ----------------------------------------------------------------------------
432// RTTI
433// ----------------------------------------------------------------------------
434
435bool wxWindowBase::IsTopLevel() const
436{
437 return wxDynamicCast(this, wxFrame) || wxDynamicCast(this, wxDialog);
438}
f03fc89f
VZ
439
440// ----------------------------------------------------------------------------
441// reparenting the window
442// ----------------------------------------------------------------------------
443
444void wxWindowBase::AddChild(wxWindowBase *child)
445{
446 wxCHECK_RET( child, _T("can't add a NULL child") );
447
448 GetChildren().Append(child);
449 child->SetParent(this);
450}
451
452void wxWindowBase::RemoveChild(wxWindowBase *child)
453{
454 wxCHECK_RET( child, _T("can't remove a NULL child") );
455
456 GetChildren().DeleteObject(child);
457 child->SetParent((wxWindow *)NULL);
458}
439b3bf1 459
f03fc89f
VZ
460bool wxWindowBase::Reparent(wxWindowBase *newParent)
461{
462 wxWindow *oldParent = GetParent();
463 if ( newParent == oldParent )
464 {
465 // nothing done
466 return FALSE;
467 }
468
469 // unlink this window from the existing parent.
470 if ( oldParent )
471 {
472 oldParent->RemoveChild(this);
473 }
474 else
475 {
476 wxTopLevelWindows.DeleteObject(this);
477 }
478
479 // add it to the new one
480 if ( newParent )
481 {
482 newParent->AddChild(this);
483 }
484 else
485 {
486 wxTopLevelWindows.Append(this);
487 }
488
489 return TRUE;
490}
491
492// ----------------------------------------------------------------------------
493// event handler stuff
494// ----------------------------------------------------------------------------
495
496void wxWindowBase::PushEventHandler(wxEvtHandler *handler)
497{
498 handler->SetNextHandler(GetEventHandler());
499 SetEventHandler(handler);
500}
501
502wxEvtHandler *wxWindowBase::PopEventHandler(bool deleteHandler)
503{
504 wxEvtHandler *handlerA = GetEventHandler();
505 if ( handlerA )
506 {
507 wxEvtHandler *handlerB = handlerA->GetNextHandler();
508 handlerA->SetNextHandler((wxEvtHandler *)NULL);
509 SetEventHandler(handlerB);
510 if ( deleteHandler )
511 {
512 delete handlerA;
513 handlerA = (wxEvtHandler *)NULL;
514 }
515 }
516
517 return handlerA;
518}
519
520// ----------------------------------------------------------------------------
521// cursors, fonts &c
522// ----------------------------------------------------------------------------
523
524bool wxWindowBase::SetBackgroundColour( const wxColour &colour )
525{
526 if ( !colour.Ok() || (colour == m_backgroundColour) )
527 return FALSE;
528
529 m_backgroundColour = colour;
530
531 return TRUE;
532}
533
534bool wxWindowBase::SetForegroundColour( const wxColour &colour )
535{
536 if ( !colour.Ok() || (colour == m_foregroundColour) )
537 return FALSE;
538
539 m_foregroundColour = colour;
540
541 return TRUE;
542}
543
544bool wxWindowBase::SetCursor(const wxCursor& cursor)
545{
546 // don't try to set invalid cursor, always fall back to the default
547 const wxCursor& cursorOk = cursor.Ok() ? cursor : *wxSTANDARD_CURSOR;
548
549 if ( cursorOk == m_cursor )
550 {
551 // no change
552 return FALSE;
553 }
554
555 m_cursor = cursorOk;
556
557 return TRUE;
558}
559
560bool wxWindowBase::SetFont(const wxFont& font)
561{
562 // don't try to set invalid font, always fall back to the default
563 const wxFont& fontOk = font.Ok() ? font : *wxSWISS_FONT;
564
565 if ( fontOk == m_font )
566 {
567 // no change
568 return FALSE;
569 }
570
571 m_font = fontOk;
572
573 return TRUE;
574}
575
789295bf
VZ
576#if wxUSE_CARET
577void wxWindowBase::SetCaret(wxCaret *caret)
578{
579 if ( m_caret )
580 {
581 delete m_caret;
582 }
583
584 m_caret = caret;
585
586 if ( m_caret )
587 {
588 wxASSERT_MSG( m_caret->GetWindow() == this,
bdb9dffb 589 _T("caret should be created associated to this window") );
789295bf
VZ
590 }
591}
592#endif // wxUSE_CARET
593
88ac883a 594#if wxUSE_VALIDATORS
f03fc89f
VZ
595// ----------------------------------------------------------------------------
596// validators
597// ----------------------------------------------------------------------------
598
599void wxWindowBase::SetValidator(const wxValidator& validator)
600{
601 if ( m_windowValidator )
602 delete m_windowValidator;
603
604 m_windowValidator = (wxValidator *)validator.Clone();
605
606 if ( m_windowValidator )
607 m_windowValidator->SetWindow(this) ;
608}
88ac883a 609#endif // wxUSE_VALIDATORS
f03fc89f
VZ
610
611// ----------------------------------------------------------------------------
612// update region testing
613// ----------------------------------------------------------------------------
614
615bool wxWindowBase::IsExposed(int x, int y) const
616{
617 return m_updateRegion.Contains(x, y) != wxOutRegion;
618}
619
620bool wxWindowBase::IsExposed(int x, int y, int w, int h) const
621{
622 return m_updateRegion.Contains(x, y, w, h) != wxOutRegion;
623}
624
625// ----------------------------------------------------------------------------
626// find window by id or name
627// ----------------------------------------------------------------------------
628
629wxWindow *wxWindowBase::FindWindow( long id )
630{
631 if ( id == m_windowId )
632 return (wxWindow *)this;
633
634 wxWindowBase *res = (wxWindow *)NULL;
635 wxWindowList::Node *node;
636 for ( node = m_children.GetFirst(); node && !res; node = node->GetNext() )
637 {
638 wxWindowBase *child = node->GetData();
639 res = child->FindWindow( id );
640 }
641
642 return (wxWindow *)res;
643}
644
645wxWindow *wxWindowBase::FindWindow( const wxString& name )
646{
647 if ( name == m_windowName )
648 return (wxWindow *)this;
649
650 wxWindowBase *res = (wxWindow *)NULL;
651 wxWindowList::Node *node;
652 for ( node = m_children.GetFirst(); node && !res; node = node->GetNext() )
653 {
654 wxWindow *child = node->GetData();
655 res = child->FindWindow(name);
656 }
657
658 return (wxWindow *)res;
659}
660
661// ----------------------------------------------------------------------------
662// dialog oriented functions
663// ----------------------------------------------------------------------------
664
34636400 665void wxWindowBase::MakeModal(bool modal)
f03fc89f 666{
34636400
VZ
667 // Disable all other windows
668 if ( IsTopLevel() )
669 {
670 wxWindowList::Node *node = wxTopLevelWindows.GetFirst();
671 while (node)
672 {
673 wxWindow *win = node->GetData();
674 if (win != this)
675 win->Enable(!modal);
676
677 node = node->GetNext();
678 }
679 }
f03fc89f
VZ
680}
681
682bool wxWindowBase::Validate()
683{
88ac883a 684#if wxUSE_VALIDATORS
f03fc89f
VZ
685 wxWindowList::Node *node;
686 for ( node = m_children.GetFirst(); node; node = node->GetNext() )
687 {
688 wxWindowBase *child = node->GetData();
689 wxValidator *validator = child->GetValidator();
dcd6b914 690 if ( validator && !validator->Validate((wxWindow *)this) )
f03fc89f
VZ
691 {
692 return FALSE;
693 }
694 }
88ac883a 695#endif // wxUSE_VALIDATORS
f03fc89f
VZ
696
697 return TRUE;
698}
699
700bool wxWindowBase::TransferDataToWindow()
701{
88ac883a 702#if wxUSE_VALIDATORS
f03fc89f
VZ
703 wxWindowList::Node *node;
704 for ( node = m_children.GetFirst(); node; node = node->GetNext() )
705 {
706 wxWindowBase *child = node->GetData();
707 wxValidator *validator = child->GetValidator();
708 if ( validator && !validator->TransferToWindow() )
709 {
710 wxLog *log = wxLog::GetActiveTarget();
711 if ( log )
712 {
713 wxLogWarning(_("Could not transfer data to window"));
714 log->Flush();
715 }
716
717 return FALSE;
718 }
719 }
88ac883a 720#endif // wxUSE_VALIDATORS
f03fc89f
VZ
721
722 return TRUE;
723}
724
725bool wxWindowBase::TransferDataFromWindow()
726{
88ac883a 727#if wxUSE_VALIDATORS
f03fc89f
VZ
728 wxWindowList::Node *node;
729 for ( node = m_children.GetFirst(); node; node = node->GetNext() )
730 {
731 wxWindow *child = node->GetData();
732 if ( child->GetValidator() &&
733 !child->GetValidator()->TransferFromWindow() )
734 {
735 return FALSE;
736 }
737 }
88ac883a 738#endif // wxUSE_VALIDATORS
f03fc89f
VZ
739
740 return TRUE;
741}
742
743void wxWindowBase::InitDialog()
744{
745 wxInitDialogEvent event(GetId());
746 event.SetEventObject( this );
747 GetEventHandler()->ProcessEvent(event);
748}
749
750// ----------------------------------------------------------------------------
751// tooltips
752// ----------------------------------------------------------------------------
753
754#if wxUSE_TOOLTIPS
755
756void wxWindowBase::SetToolTip( const wxString &tip )
757{
758 // don't create the new tooltip if we already have one
759 if ( m_tooltip )
760 {
761 m_tooltip->SetTip( tip );
762 }
763 else
764 {
765 SetToolTip( new wxToolTip( tip ) );
766 }
767
768 // setting empty tooltip text does not remove the tooltip any more - use
769 // SetToolTip((wxToolTip *)NULL) for this
770}
771
772void wxWindowBase::DoSetToolTip(wxToolTip *tooltip)
773{
774 if ( m_tooltip )
775 delete m_tooltip;
776
777 m_tooltip = tooltip;
778}
779
780#endif // wxUSE_TOOLTIPS
781
782// ----------------------------------------------------------------------------
783// constraints and sizers
784// ----------------------------------------------------------------------------
785
786#if wxUSE_CONSTRAINTS
787
788void wxWindowBase::SetConstraints( wxLayoutConstraints *constraints )
789{
790 if ( m_constraints )
791 {
792 UnsetConstraints(m_constraints);
793 delete m_constraints;
794 }
795 m_constraints = constraints;
796 if ( m_constraints )
797 {
798 // Make sure other windows know they're part of a 'meaningful relationship'
799 if ( m_constraints->left.GetOtherWindow() && (m_constraints->left.GetOtherWindow() != this) )
800 m_constraints->left.GetOtherWindow()->AddConstraintReference(this);
801 if ( m_constraints->top.GetOtherWindow() && (m_constraints->top.GetOtherWindow() != this) )
802 m_constraints->top.GetOtherWindow()->AddConstraintReference(this);
803 if ( m_constraints->right.GetOtherWindow() && (m_constraints->right.GetOtherWindow() != this) )
804 m_constraints->right.GetOtherWindow()->AddConstraintReference(this);
805 if ( m_constraints->bottom.GetOtherWindow() && (m_constraints->bottom.GetOtherWindow() != this) )
806 m_constraints->bottom.GetOtherWindow()->AddConstraintReference(this);
807 if ( m_constraints->width.GetOtherWindow() && (m_constraints->width.GetOtherWindow() != this) )
808 m_constraints->width.GetOtherWindow()->AddConstraintReference(this);
809 if ( m_constraints->height.GetOtherWindow() && (m_constraints->height.GetOtherWindow() != this) )
810 m_constraints->height.GetOtherWindow()->AddConstraintReference(this);
811 if ( m_constraints->centreX.GetOtherWindow() && (m_constraints->centreX.GetOtherWindow() != this) )
812 m_constraints->centreX.GetOtherWindow()->AddConstraintReference(this);
813 if ( m_constraints->centreY.GetOtherWindow() && (m_constraints->centreY.GetOtherWindow() != this) )
814 m_constraints->centreY.GetOtherWindow()->AddConstraintReference(this);
815 }
816}
817
818// This removes any dangling pointers to this window in other windows'
819// constraintsInvolvedIn lists.
820void wxWindowBase::UnsetConstraints(wxLayoutConstraints *c)
821{
822 if ( c )
823 {
824 if ( c->left.GetOtherWindow() && (c->top.GetOtherWindow() != this) )
825 c->left.GetOtherWindow()->RemoveConstraintReference(this);
826 if ( c->top.GetOtherWindow() && (c->top.GetOtherWindow() != this) )
827 c->top.GetOtherWindow()->RemoveConstraintReference(this);
828 if ( c->right.GetOtherWindow() && (c->right.GetOtherWindow() != this) )
829 c->right.GetOtherWindow()->RemoveConstraintReference(this);
830 if ( c->bottom.GetOtherWindow() && (c->bottom.GetOtherWindow() != this) )
831 c->bottom.GetOtherWindow()->RemoveConstraintReference(this);
832 if ( c->width.GetOtherWindow() && (c->width.GetOtherWindow() != this) )
833 c->width.GetOtherWindow()->RemoveConstraintReference(this);
834 if ( c->height.GetOtherWindow() && (c->height.GetOtherWindow() != this) )
835 c->height.GetOtherWindow()->RemoveConstraintReference(this);
836 if ( c->centreX.GetOtherWindow() && (c->centreX.GetOtherWindow() != this) )
837 c->centreX.GetOtherWindow()->RemoveConstraintReference(this);
838 if ( c->centreY.GetOtherWindow() && (c->centreY.GetOtherWindow() != this) )
839 c->centreY.GetOtherWindow()->RemoveConstraintReference(this);
840 }
841}
842
843// Back-pointer to other windows we're involved with, so if we delete this
844// window, we must delete any constraints we're involved with.
845void wxWindowBase::AddConstraintReference(wxWindowBase *otherWin)
846{
847 if ( !m_constraintsInvolvedIn )
848 m_constraintsInvolvedIn = new wxWindowList;
849 if ( !m_constraintsInvolvedIn->Find(otherWin) )
850 m_constraintsInvolvedIn->Append(otherWin);
851}
852
853// REMOVE back-pointer to other windows we're involved with.
854void wxWindowBase::RemoveConstraintReference(wxWindowBase *otherWin)
855{
856 if ( m_constraintsInvolvedIn )
857 m_constraintsInvolvedIn->DeleteObject(otherWin);
858}
859
860// Reset any constraints that mention this window
861void wxWindowBase::DeleteRelatedConstraints()
862{
863 if ( m_constraintsInvolvedIn )
864 {
865 wxWindowList::Node *node = m_constraintsInvolvedIn->GetFirst();
866 while (node)
867 {
868 wxWindow *win = node->GetData();
869 wxLayoutConstraints *constr = win->GetConstraints();
870
871 // Reset any constraints involving this window
872 if ( constr )
873 {
874 constr->left.ResetIfWin(this);
875 constr->top.ResetIfWin(this);
876 constr->right.ResetIfWin(this);
877 constr->bottom.ResetIfWin(this);
878 constr->width.ResetIfWin(this);
879 constr->height.ResetIfWin(this);
880 constr->centreX.ResetIfWin(this);
881 constr->centreY.ResetIfWin(this);
882 }
883
884 wxWindowList::Node *next = node->GetNext();
885 delete node;
886 node = next;
887 }
888
889 delete m_constraintsInvolvedIn;
890 m_constraintsInvolvedIn = (wxWindowList *) NULL;
891 }
892}
893
894void wxWindowBase::SetSizer(wxSizer *sizer)
895{
896 m_windowSizer = sizer;
897 if ( sizer )
898 sizer->SetSizerParent(this);
899}
900
901bool wxWindowBase::Layout()
902{
903 if ( GetConstraints() )
904 {
905 int w, h;
906 GetClientSize(&w, &h);
907 GetConstraints()->width.SetValue(w);
908 GetConstraints()->height.SetValue(h);
909 }
910
911 // If top level (one sizer), evaluate the sizer's constraints.
912 if ( GetSizer() )
913 {
914 int noChanges;
915 GetSizer()->ResetConstraints(); // Mark all constraints as unevaluated
916 GetSizer()->LayoutPhase1(&noChanges);
917 GetSizer()->LayoutPhase2(&noChanges);
918 GetSizer()->SetConstraintSizes(); // Recursively set the real window sizes
919 return TRUE;
920 }
921 else
922 {
923 // Otherwise, evaluate child constraints
924 ResetConstraints(); // Mark all constraints as unevaluated
925 DoPhase(1); // Just one phase need if no sizers involved
926 DoPhase(2);
927 SetConstraintSizes(); // Recursively set the real window sizes
928 }
929 return TRUE;
930}
931
932
933// Do a phase of evaluating constraints: the default behaviour. wxSizers may
934// do a similar thing, but also impose their own 'constraints' and order the
935// evaluation differently.
936bool wxWindowBase::LayoutPhase1(int *noChanges)
937{
938 wxLayoutConstraints *constr = GetConstraints();
939 if ( constr )
940 {
941 return constr->SatisfyConstraints(this, noChanges);
942 }
943 else
944 return TRUE;
945}
946
947bool wxWindowBase::LayoutPhase2(int *noChanges)
948{
949 *noChanges = 0;
950
951 // Layout children
952 DoPhase(1);
953 DoPhase(2);
954 return TRUE;
955}
956
957// Do a phase of evaluating child constraints
958bool wxWindowBase::DoPhase(int phase)
959{
960 int noIterations = 0;
961 int maxIterations = 500;
962 int noChanges = 1;
963 int noFailures = 0;
964 wxWindowList succeeded;
965 while ((noChanges > 0) && (noIterations < maxIterations))
966 {
967 noChanges = 0;
968 noFailures = 0;
969 wxWindowList::Node *node = GetChildren().GetFirst();
970 while (node)
971 {
972 wxWindow *child = node->GetData();
34636400 973 if ( !child->IsTopLevel() )
f03fc89f
VZ
974 {
975 wxLayoutConstraints *constr = child->GetConstraints();
976 if ( constr )
977 {
978 if ( !succeeded.Find(child) )
979 {
980 int tempNoChanges = 0;
981 bool success = ( (phase == 1) ? child->LayoutPhase1(&tempNoChanges) : child->LayoutPhase2(&tempNoChanges) ) ;
982 noChanges += tempNoChanges;
983 if ( success )
984 {
985 succeeded.Append(child);
986 }
987 }
988 }
989 }
990 node = node->GetNext();
991 }
992
993 noIterations++;
994 }
995
996 return TRUE;
997}
998
999void wxWindowBase::ResetConstraints()
1000{
1001 wxLayoutConstraints *constr = GetConstraints();
1002 if ( constr )
1003 {
1004 constr->left.SetDone(FALSE);
1005 constr->top.SetDone(FALSE);
1006 constr->right.SetDone(FALSE);
1007 constr->bottom.SetDone(FALSE);
1008 constr->width.SetDone(FALSE);
1009 constr->height.SetDone(FALSE);
1010 constr->centreX.SetDone(FALSE);
1011 constr->centreY.SetDone(FALSE);
1012 }
1013 wxWindowList::Node *node = GetChildren().GetFirst();
1014 while (node)
1015 {
1016 wxWindow *win = node->GetData();
34636400 1017 if ( !win->IsTopLevel() )
f03fc89f
VZ
1018 win->ResetConstraints();
1019 node = node->GetNext();
1020 }
1021}
1022
1023// Need to distinguish between setting the 'fake' size for windows and sizers,
1024// and setting the real values.
1025void wxWindowBase::SetConstraintSizes(bool recurse)
1026{
1027 wxLayoutConstraints *constr = GetConstraints();
1028 if ( constr && constr->left.GetDone() && constr->right.GetDone( ) &&
1029 constr->width.GetDone() && constr->height.GetDone())
1030 {
1031 int x = constr->left.GetValue();
1032 int y = constr->top.GetValue();
1033 int w = constr->width.GetValue();
1034 int h = constr->height.GetValue();
1035
1036 // If we don't want to resize this window, just move it...
1037 if ( (constr->width.GetRelationship() != wxAsIs ) ||
1038 (constr->height.GetRelationship() != wxAsIs))
1039 {
1040 // Calls Layout() recursively. AAAGH. How can we stop that.
1041 // Simply take Layout() out of non-top level OnSizes.
1042 SizerSetSize(x, y, w, h);
1043 }
1044 else
1045 {
1046 SizerMove(x, y);
1047 }
1048 }
1049 else if ( constr )
1050 {
1051 wxChar *windowClass = GetClassInfo()->GetClassName();
1052
1053 wxString winName;
1054 if ( GetName() == _T("") )
1055 winName = _T("unnamed");
1056 else
1057 winName = GetName();
1058 wxLogDebug( _T("Constraint(s) not satisfied for window of type %s, name %s:\n"),
1059 (const wxChar *)windowClass,
1060 (const wxChar *)winName);
1061 if ( !constr->left.GetDone()) wxLogDebug( _T(" unsatisfied 'left' constraint.\n") );
1062 if ( !constr->right.GetDone()) wxLogDebug( _T(" unsatisfied 'right' constraint.\n") );
1063 if ( !constr->width.GetDone()) wxLogDebug( _T(" unsatisfied 'width' constraint.\n") );
1064 if ( !constr->height.GetDone()) wxLogDebug( _T(" unsatisfied 'height' constraint.\n") );
1065 wxLogDebug( _T("Please check constraints: try adding AsIs() constraints.\n") );
1066 }
1067
1068 if ( recurse )
1069 {
1070 wxWindowList::Node *node = GetChildren().GetFirst();
1071 while (node)
1072 {
1073 wxWindow *win = node->GetData();
34636400 1074 if ( !win->IsTopLevel() )
f03fc89f
VZ
1075 win->SetConstraintSizes();
1076 node = node->GetNext();
1077 }
1078 }
1079}
1080
1081// This assumes that all sizers are 'on' the same window, i.e. the parent of
1082// this window.
1083void wxWindowBase::TransformSizerToActual(int *x, int *y) const
1084{
34636400 1085 if ( !m_sizerParent || m_sizerParent->IsTopLevel() )
f03fc89f
VZ
1086 return;
1087
1088 int xp, yp;
1089 m_sizerParent->GetPosition(&xp, &yp);
1090 m_sizerParent->TransformSizerToActual(&xp, &yp);
1091 *x += xp;
1092 *y += yp;
1093}
1094
1095void wxWindowBase::SizerSetSize(int x, int y, int w, int h)
1096{
1097 int xx = x;
1098 int yy = y;
1099 TransformSizerToActual(&xx, &yy);
1100 SetSize(xx, yy, w, h);
1101}
1102
1103void wxWindowBase::SizerMove(int x, int y)
1104{
1105 int xx = x;
1106 int yy = y;
1107 TransformSizerToActual(&xx, &yy);
1108 Move(xx, yy);
1109}
1110
1111// Only set the size/position of the constraint (if any)
1112void wxWindowBase::SetSizeConstraint(int x, int y, int w, int h)
1113{
1114 wxLayoutConstraints *constr = GetConstraints();
1115 if ( constr )
1116 {
1117 if ( x != -1 )
1118 {
1119 constr->left.SetValue(x);
1120 constr->left.SetDone(TRUE);
1121 }
1122 if ( y != -1 )
1123 {
1124 constr->top.SetValue(y);
1125 constr->top.SetDone(TRUE);
1126 }
1127 if ( w != -1 )
1128 {
1129 constr->width.SetValue(w);
1130 constr->width.SetDone(TRUE);
1131 }
1132 if ( h != -1 )
1133 {
1134 constr->height.SetValue(h);
1135 constr->height.SetDone(TRUE);
1136 }
1137 }
1138}
1139
1140void wxWindowBase::MoveConstraint(int x, int y)
1141{
1142 wxLayoutConstraints *constr = GetConstraints();
1143 if ( constr )
1144 {
1145 if ( x != -1 )
1146 {
1147 constr->left.SetValue(x);
1148 constr->left.SetDone(TRUE);
1149 }
1150 if ( y != -1 )
1151 {
1152 constr->top.SetValue(y);
1153 constr->top.SetDone(TRUE);
1154 }
1155 }
1156}
1157
1158void wxWindowBase::GetSizeConstraint(int *w, int *h) const
1159{
1160 wxLayoutConstraints *constr = GetConstraints();
1161 if ( constr )
1162 {
1163 *w = constr->width.GetValue();
1164 *h = constr->height.GetValue();
1165 }
1166 else
1167 GetSize(w, h);
1168}
1169
1170void wxWindowBase::GetClientSizeConstraint(int *w, int *h) const
1171{
1172 wxLayoutConstraints *constr = GetConstraints();
1173 if ( constr )
1174 {
1175 *w = constr->width.GetValue();
1176 *h = constr->height.GetValue();
1177 }
1178 else
1179 GetClientSize(w, h);
1180}
1181
1182void wxWindowBase::GetPositionConstraint(int *x, int *y) const
1183{
1184 wxLayoutConstraints *constr = GetConstraints();
1185 if ( constr )
1186 {
1187 *x = constr->left.GetValue();
1188 *y = constr->top.GetValue();
1189 }
1190 else
1191 GetPosition(x, y);
1192}
1193
1194#endif // wxUSE_CONSTRAINTS
1195
1196// ----------------------------------------------------------------------------
1197// do Update UI processing for child controls
1198// ----------------------------------------------------------------------------
7ec1983b
VZ
1199
1200// TODO: should this be implemented for the child window rather
1201// than the parent? Then you can override it e.g. for wxCheckBox
1202// to do the Right Thing rather than having to assume a fixed number
1203// of control classes.
f03fc89f 1204void wxWindowBase::UpdateWindowUI()
7ec1983b 1205{
f03fc89f
VZ
1206 wxWindowID id = GetId();
1207 if ( id > 0 )
7ec1983b 1208 {
f03fc89f
VZ
1209 wxUpdateUIEvent event(id);
1210 event.m_eventObject = this;
7ec1983b 1211
f03fc89f
VZ
1212 if ( GetEventHandler()->ProcessEvent(event) )
1213 {
1214 if ( event.GetSetEnabled() )
1215 Enable(event.GetEnabled());
7ec1983b 1216
34636400
VZ
1217 if ( event.GetSetText() )
1218 {
1219 wxControl *control = wxDynamicCast(this, wxControl);
1220 if ( control )
1221 control->SetLabel(event.GetText());
1222 }
f03fc89f 1223
88ac883a 1224#if wxUSE_CHECKBOX
34636400
VZ
1225 wxCheckBox *checkbox = wxDynamicCast(this, wxCheckBox);
1226 if ( checkbox )
f03fc89f
VZ
1227 {
1228 if ( event.GetSetChecked() )
34636400 1229 checkbox->SetValue(event.GetChecked());
f03fc89f 1230 }
88ac883a
VZ
1231#endif // wxUSE_CHECKBOX
1232
1233#if wxUSE_RADIOBUTTON
34636400
VZ
1234 wxRadioButton *radiobtn = wxDynamicCast(this, wxRadioButton);
1235 if ( radiobtn )
f03fc89f
VZ
1236 {
1237 if ( event.GetSetChecked() )
34636400 1238 radiobtn->SetValue(event.GetChecked());
f03fc89f 1239 }
88ac883a 1240#endif // wxUSE_RADIOBUTTON
f03fc89f 1241 }
7ec1983b 1242 }
7ec1983b 1243}
fd71308f 1244
f03fc89f
VZ
1245// ----------------------------------------------------------------------------
1246// dialog units translations
1247// ----------------------------------------------------------------------------
1248
1249wxPoint wxWindowBase::ConvertPixelsToDialog(const wxPoint& pt)
fd71308f
JS
1250{
1251 int charWidth = GetCharWidth();
1252 int charHeight = GetCharHeight();
5d9c2818
RD
1253 wxPoint pt2(-1, -1);
1254 if (pt.x != -1)
1255 pt2.x = (int) ((pt.x * 4) / charWidth) ;
1256 if (pt.y != -1)
1257 pt2.y = (int) ((pt.y * 8) / charHeight) ;
fd71308f
JS
1258
1259 return pt2;
1260}
1261
f03fc89f 1262wxPoint wxWindowBase::ConvertDialogToPixels(const wxPoint& pt)
fd71308f
JS
1263{
1264 int charWidth = GetCharWidth();
1265 int charHeight = GetCharHeight();
5d9c2818
RD
1266 wxPoint pt2(-1, -1);
1267 if (pt.x != -1)
1268 pt2.x = (int) ((pt.x * charWidth) / 4) ;
1269 if (pt.y != -1)
1270 pt2.y = (int) ((pt.y * charHeight) / 8) ;
fd71308f
JS
1271
1272 return pt2;
1273}
1274
f03fc89f
VZ
1275// ----------------------------------------------------------------------------
1276// event handlers
1277// ----------------------------------------------------------------------------
1278
1279// propagate the colour change event to the subwindows
1280void wxWindowBase::OnSysColourChanged(wxSysColourChangedEvent& event)
1281{
1282 wxWindowList::Node *node = GetChildren().GetFirst();
1283 while ( node )
1284 {
1285 // Only propagate to non-top-level windows
1286 wxWindow *win = node->GetData();
1287 if ( !win->IsTopLevel() )
1288 {
1289 wxSysColourChangedEvent event2;
1290 event.m_eventObject = win;
1291 win->GetEventHandler()->ProcessEvent(event2);
1292 }
1293
1294 node = node->GetNext();
1295 }
1296}
1297
1298// the default action is to populate dialog with data when it's created
1299void wxWindowBase::OnInitDialog( wxInitDialogEvent &WXUNUSED(event) )
1300{
1301 TransferDataToWindow();
1302}
1303
1304// ----------------------------------------------------------------------------
1305// list classes implementation
1306// ----------------------------------------------------------------------------
1307
1308void wxWindowListNode::DeleteData()
1309{
1310 delete (wxWindow *)GetData();
1311}
1312