1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: wxControl class
4 // Author: Stefan Csomor
8 // Copyright: (c) Stefan Csomor
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
13 #pragma implementation "control.h"
18 #include "wx/control.h"
22 #include "wx/dcclient.h"
23 #include "wx/notebook.h"
24 #include "wx/tabctrl.h"
25 #include "wx/radiobox.h"
26 #include "wx/spinbutt.h"
27 #include "wx/scrolbar.h"
28 #include "wx/button.h"
29 #include "wx/dialog.h"
30 #include "wx/statbox.h"
32 #include "wx/stattext.h"
34 #if !USE_SHARED_LIBRARY
35 IMPLEMENT_ABSTRACT_CLASS(wxControl
, wxWindow
)
37 BEGIN_EVENT_TABLE(wxControl
, wxWindow
)
38 EVT_MOUSE_EVENTS( wxControl::OnMouseEvent
)
39 EVT_PAINT( wxControl::OnPaint
)
43 #include "wx/mac/uma.h"
44 #include "wx/mac/private.h"
49 #if PRAGMA_STRUCT_ALIGN
50 #pragma options align=mac68k
51 #elif PRAGMA_STRUCT_PACKPUSH
53 #elif PRAGMA_STRUCT_PACK
58 unsigned short instruction
;
60 } cdefRec
, *cdefPtr
, **cdefHandle
;
62 #if PRAGMA_STRUCT_ALIGN
63 #pragma options align=reset
64 #elif PRAGMA_STRUCT_PACKPUSH
66 #elif PRAGMA_STRUCT_PACK
70 ControlActionUPP wxMacLiveScrollbarActionUPP
= NULL
;
71 wxControl
*wxFindControlFromMacControl(ControlHandle inControl
) ;
73 pascal void wxMacLiveScrollbarActionProc( ControlHandle control
, ControlPartCode partCode
) ;
74 pascal void wxMacLiveScrollbarActionProc( ControlHandle control
, ControlPartCode partCode
)
78 wxControl
* wx
= (wxControl
*) GetControlReference( control
) ;
81 wx
->MacHandleControlClick( control
, partCode
, true /* stillDown */ ) ;
86 ControlColorUPP wxMacSetupControlBackgroundUPP
= NULL
;
87 ControlDefUPP wxMacControlActionUPP
= NULL
;
89 pascal SInt32
wxMacControlDefinition(SInt16 varCode
, ControlRef theControl
, ControlDefProcMessage message
, SInt32 param
)
92 wxControl
* wx
= (wxControl
*) wxFindControlFromMacControl( theControl
) ;
93 if ( wx
!= NULL
&& wx
->IsKindOf( CLASSINFO( wxControl
) ) )
95 if( message
== drawCntl
)
97 wxMacWindowClipper
clip( wx
) ;
98 return InvokeControlDefUPP( varCode
, theControl
, message
, param
, (ControlDefUPP
) wx
->MacGetControlAction() ) ;
101 return InvokeControlDefUPP( varCode
, theControl
, message
, param
, (ControlDefUPP
) wx
->MacGetControlAction() ) ;
106 pascal OSStatus
wxMacSetupControlBackground( ControlRef iControl
, SInt16 iMessage
, SInt16 iDepth
, Boolean iIsColor
)
108 OSStatus status
= noErr
;
111 case kControlMsgSetUpBackground
:
113 wxControl
* wx
= (wxControl
*) GetControlReference( iControl
) ;
114 if ( wx
!= NULL
&& wx
->IsKindOf( CLASSINFO( wxControl
) ) )
116 wxDC::MacSetupBackgroundForCurrentPort( wx
->MacGetBackgroundBrush() ) ;
118 // under classic this would lead to partial redraws
119 RgnHandle clip
= NewRgn() ;
122 wx
->MacWindowToRootWindow( &x
,&y
) ;
123 CopyRgn( (RgnHandle
) wx
->MacGetVisibleRegion(false).GetWXHRGN() , clip
) ;
124 OffsetRgn( clip
, x
, y
) ;
142 wxControl::wxControl()
144 m_macControl
= NULL
;
145 m_macControlAction
= NULL
;
146 m_macHorizontalBorder
= 0 ; // additional pixels around the real control
147 m_macVerticalBorder
= 0 ;
148 m_backgroundColour
= *wxWHITE
;
149 m_foregroundColour
= *wxBLACK
;
151 if ( wxMacLiveScrollbarActionUPP
== NULL
)
153 #if defined(UNIVERSAL_INTERFACES_VERSION) && (UNIVERSAL_INTERFACES_VERSION >= 0x0340)
154 wxMacLiveScrollbarActionUPP
= NewControlActionUPP( wxMacLiveScrollbarActionProc
);
156 wxMacLiveScrollbarActionUPP
= NewControlActionProc( wxMacLiveScrollbarActionProc
) ;
161 bool wxControl::Create(wxWindow
*parent
, wxWindowID id
,
163 const wxSize
& size
, long style
,
164 const wxValidator
& validator
,
165 const wxString
& name
)
167 m_macControl
= NULL
;
168 m_macHorizontalBorder
= 0 ; // additional pixels around the real control
169 m_macVerticalBorder
= 0 ;
171 bool rval
= wxWindow::Create(parent
, id
, pos
, size
, style
, name
);
174 m_backgroundColour
= parent
->GetBackgroundColour() ;
175 m_foregroundColour
= parent
->GetForegroundColour() ;
179 SetValidator(validator
);
185 wxControl::~wxControl()
187 m_isBeingDeleted
= TRUE
;
188 wxRemoveMacControlAssociation( this ) ;
189 // If we delete an item, we should initialize the parent panel,
190 // because it could now be invalid.
191 wxWindow
*parent
= GetParent() ;
194 if (parent
->GetDefaultItem() == (wxButton
*) this)
195 parent
->SetDefaultItem(NULL
);
197 if ( (ControlHandle
) m_macControl
)
199 // in case the callback might be called during destruction
200 ::SetControlColorProc( m_macControl
, NULL
) ;
201 ::DisposeControl( (ControlHandle
) m_macControl
) ;
202 m_macControl
= NULL
;
206 void wxControl::SetLabel(const wxString
& title
)
208 m_label
= wxStripMenuCodes(title
) ;
212 UMASetControlTitle( (ControlHandle
) m_macControl
, m_label
, m_font
.GetEncoding() ) ;
217 wxSize
wxControl::DoGetBestSize() const
219 if ( (ControlHandle
) m_macControl
== NULL
)
220 return wxWindow::DoGetBestSize() ;
222 Rect bestsize
= { 0 , 0 , 0 , 0 } ;
223 short baselineoffset
;
224 int bestWidth
, bestHeight
;
225 ::GetBestControlRect( (ControlHandle
) m_macControl
, &bestsize
, &baselineoffset
) ;
227 if ( EmptyRect( &bestsize
) )
230 bestsize
.left
= bestsize
.top
= 0 ;
231 bestsize
.right
= 16 ;
232 bestsize
.bottom
= 16 ;
233 if ( IsKindOf( CLASSINFO( wxScrollBar
) ) )
235 bestsize
.bottom
= 16 ;
237 else if ( IsKindOf( CLASSINFO( wxSpinButton
) ) )
239 bestsize
.bottom
= 24 ;
243 bestWidth
= bestsize
.right
- bestsize
.left
;
245 bestWidth
+= 2 * m_macHorizontalBorder
;
247 bestHeight
= bestsize
.bottom
- bestsize
.top
;
248 if ( bestHeight
< 10 )
251 bestHeight
+= 2 * m_macVerticalBorder
;
254 return wxSize(bestWidth
, bestHeight
);
257 bool wxControl::ProcessCommand (wxCommandEvent
& event
)
260 // 1) OnCommand, starting at this window and working up parent hierarchy
261 // 2) OnCommand then calls ProcessEvent to search the event tables.
262 return GetEventHandler()->ProcessEvent(event
);
265 // ------------------------
266 wxList
*wxWinMacControlList
= NULL
;
267 wxControl
*wxFindControlFromMacControl(ControlHandle inControl
)
269 wxNode
*node
= wxWinMacControlList
->Find((long)inControl
);
272 return (wxControl
*)node
->GetData();
275 void wxAssociateControlWithMacControl(ControlHandle inControl
, wxControl
*control
)
277 // adding NULL WindowRef is (first) surely a result of an error and
278 // (secondly) breaks menu command processing
279 wxCHECK_RET( inControl
!= (ControlHandle
) NULL
, wxT("attempt to add a NULL WindowRef to window list") );
281 if ( !wxWinMacControlList
->Find((long)inControl
) )
282 wxWinMacControlList
->Append((long)inControl
, control
);
285 void wxRemoveMacControlAssociation(wxControl
*control
)
287 if ( wxWinMacControlList
)
288 wxWinMacControlList
->DeleteObject(control
);
291 void wxControl::MacPreControlCreate( wxWindow
*parent
, wxWindowID id
, wxString label
,
293 const wxSize
& size
, long style
,
294 const wxValidator
& validator
,
295 const wxString
& name
, WXRECTPTR outBounds
, unsigned char* maclabel
)
299 // These sizes will be adjusted in MacPostControlCreate
305 ((Rect
*)outBounds
)->top
= -10;
306 ((Rect
*)outBounds
)->left
= -10;
307 ((Rect
*)outBounds
)->bottom
= 0;
308 ((Rect
*)outBounds
)->right
= 0;
310 wxMacStringToPascal( wxStripMenuCodes(label
) , maclabel
) ;
313 void wxControl::MacPostControlCreate()
315 wxASSERT_MSG( (ControlHandle
) m_macControl
!= NULL
, wxT("No valid mac control") ) ;
317 if ( IsKindOf( CLASSINFO( wxScrollBar
) ) )
321 else if ( !UMAHasAquaLayout() && (IsKindOf( CLASSINFO( wxStaticBox
) ) || IsKindOf( CLASSINFO( wxRadioBox
) ) || IsKindOf( CLASSINFO( wxButton
) ) ) )
323 ControlFontStyleRec controlstyle
;
324 controlstyle
.flags
= kControlUseFontMask
;
325 controlstyle
.font
= kControlFontSmallBoldSystemFont
;
327 ::SetControlFontStyle( (ControlHandle
) m_macControl
, &controlstyle
) ;
331 ControlFontStyleRec controlstyle
;
332 controlstyle
.flags
= kControlUseFontMask
;
334 if (IsKindOf( CLASSINFO( wxButton
) ) )
335 controlstyle
.font
= kControlFontBigSystemFont
; // eventually kControlFontBigSystemFont ;
337 controlstyle
.font
= kControlFontSmallSystemFont
;
339 ::SetControlFontStyle( (ControlHandle
) m_macControl
, &controlstyle
) ;
341 ControlHandle container
= (ControlHandle
) GetParent()->MacGetContainerForEmbedding() ;
342 wxASSERT_MSG( container
!= NULL
, wxT("No valid mac container control") ) ;
343 ::EmbedControl( (ControlHandle
) m_macControl
, container
) ;
344 m_macControlIsShown
= MacIsReallyShown() ;
346 wxAssociateControlWithMacControl( (ControlHandle
) m_macControl
, this ) ;
347 if ( wxMacSetupControlBackgroundUPP
== NULL
)
349 wxMacSetupControlBackgroundUPP
= NewControlColorUPP( wxMacSetupControlBackground
) ;
351 if ( wxMacControlActionUPP
== NULL
)
353 wxMacControlActionUPP
= NewControlDefUPP( wxMacControlDefinition
) ;
355 // The following block of code is responsible for crashes when switching
356 // back to windows, which can be seen in the dialogs sample.
357 // It is disabled until a proper solution can be found.
361 only working under classic carbon
362 m_macControlAction = *(**(ControlHandle)m_macControl).contrlDefProc ;
363 (**(ControlHandle)m_macControl).contrlDefProc = (Handle) &wxMacControlActionUPP ;
366 m_macControlAction
= *(**(ControlHandle
)m_macControl
).contrlDefProc
;
369 cdef
= (cdefHandle
) NewHandle( sizeof(cdefRec
) ) ;
370 if ( (**(ControlHandle
)m_macControl
).contrlDefProc
!= NULL
)
372 (**cdef
).instruction
= 0x4EF9; /* JMP instruction */
373 (**cdef
).function
= (void(*)()) wxMacControlActionUPP
;
374 (**(ControlHandle
)m_macControl
).contrlDefProc
= (Handle
) cdef
;
378 SetControlColorProc( (ControlHandle
) m_macControl
, wxMacSetupControlBackgroundUPP
) ;
380 // Adjust the controls size and position
381 wxPoint
pos(m_x
, m_y
);
382 wxSize
best_size( DoGetBestSize() );
383 wxSize
new_size( m_width
, m_height
);
385 m_x
= m_y
= m_width
= m_height
= -1; // Forces SetSize to move/size the control
387 if (new_size
.x
== -1) {
388 new_size
.x
= best_size
.x
;
390 if (new_size
.y
== -1) {
391 new_size
.y
= best_size
.y
;
394 SetSize(pos
.x
, pos
.y
, new_size
.x
, new_size
.y
);
397 UMASetControlTitle( (ControlHandle
) m_macControl
, wxStripMenuCodes(m_label
) , m_font
.GetEncoding() ) ;
400 if ( m_macControlIsShown
)
401 UMAShowControl( (ControlHandle
) m_macControl
) ;
403 SetCursor( *wxSTANDARD_CURSOR
) ;
408 void wxControl::MacAdjustControlRect()
410 wxASSERT_MSG( (ControlHandle
) m_macControl
!= NULL
, wxT("No valid mac control") ) ;
411 if ( m_width
== -1 || m_height
== -1 )
413 Rect bestsize
= { 0 , 0 , 0 , 0 } ;
414 short baselineoffset
;
416 ::GetBestControlRect( (ControlHandle
) m_macControl
, &bestsize
, &baselineoffset
) ;
418 if ( EmptyRect( &bestsize
) )
421 bestsize
.left
= bestsize
.top
= 0 ;
422 bestsize
.right
= 16 ;
423 bestsize
.bottom
= 16 ;
424 if ( IsKindOf( CLASSINFO( wxScrollBar
) ) )
426 bestsize
.bottom
= 16 ;
428 else if ( IsKindOf( CLASSINFO( wxSpinButton
) ) )
430 bestsize
.bottom
= 24 ;
436 if ( IsKindOf( CLASSINFO( wxButton
) ) )
438 m_width
= m_label
.Length() * 8 + 12 ;
442 else if ( IsKindOf( CLASSINFO( wxStaticText
) ) )
444 m_width
= m_label
.Length() * 8 ;
447 m_width
= bestsize
.right
- bestsize
.left
;
449 m_width
+= 2 * m_macHorizontalBorder
+ MacGetLeftBorderSize() + MacGetRightBorderSize() ;
451 if ( m_height
== -1 )
453 m_height
= bestsize
.bottom
- bestsize
.top
;
457 m_height
+= 2 * m_macVerticalBorder
+ MacGetTopBorderSize() + MacGetBottomBorderSize() ;
459 MacUpdateDimensions() ;
463 WXWidget
wxControl::MacGetContainerForEmbedding()
466 return m_macControl
;
468 return wxWindow::MacGetContainerForEmbedding() ;
471 void wxControl::MacUpdateDimensions()
473 // actually in the current systems this should never be possible, but later reparenting
474 // may become a reality
476 if ( (ControlHandle
) m_macControl
== NULL
)
479 if ( GetParent() == NULL
)
482 WindowRef rootwindow
= (WindowRef
) MacGetRootWindow() ;
483 if ( rootwindow
== NULL
)
487 GetControlBounds( (ControlHandle
) m_macControl
, &oldBounds
) ;
489 int new_x
= m_x
+ MacGetLeftBorderSize() + m_macHorizontalBorder
;
490 int new_y
= m_y
+ MacGetTopBorderSize() + m_macVerticalBorder
;
491 int new_width
= m_width
- MacGetLeftBorderSize() - MacGetRightBorderSize() - 2 * m_macHorizontalBorder
;
492 int new_height
= m_height
- MacGetTopBorderSize() - MacGetBottomBorderSize() - 2 * m_macVerticalBorder
;
494 GetParent()->MacWindowToRootWindow( & new_x
, & new_y
) ;
495 bool doMove
= new_x
!= oldBounds
.left
|| new_y
!= oldBounds
.top
;
496 bool doResize
= ( oldBounds
.right
- oldBounds
.left
) != new_width
|| (oldBounds
.bottom
- oldBounds
.top
) != new_height
;
497 if ( doMove
|| doResize
)
499 InvalWindowRect( rootwindow
, &oldBounds
) ;
502 UMAMoveControl( (ControlHandle
) m_macControl
, new_x
, new_y
) ;
506 UMASizeControl( (ControlHandle
) m_macControl
, new_width
, new_height
) ;
511 void wxControl::MacSuperChangedPosition()
513 MacUpdateDimensions() ;
514 wxWindow::MacSuperChangedPosition() ;
517 void wxControl::MacSuperEnabled( bool enabled
)
520 wxWindow::MacSuperEnabled( enabled
) ;
523 void wxControl::MacSuperShown( bool show
)
525 if ( (ControlHandle
) m_macControl
)
529 if ( m_macControlIsShown
)
531 ::UMAHideControl( (ControlHandle
) m_macControl
) ;
532 m_macControlIsShown
= false ;
537 if ( MacIsReallyShown() && !m_macControlIsShown
)
539 ::UMAShowControl( (ControlHandle
) m_macControl
) ;
540 m_macControlIsShown
= true ;
545 wxWindow::MacSuperShown( show
) ;
548 void wxControl::DoSetSize(int x
, int y
,
549 int width
, int height
,
552 wxWindow::DoSetSize( x
, y
,width
, height
,sizeFlags
) ;
555 Rect meta
, control
;
556 GetControlBounds( (ControlHandle
) m_macControl
, &control
) ;
557 RgnHandle rgn
= NewRgn() ;
558 GetControlRegion( (ControlHandle
) m_macControl
, kControlStructureMetaPart
, rgn
) ;
559 GetRegionBounds( rgn
, &meta
) ;
560 if ( !EmptyRect( &meta
) )
562 wxASSERT( meta
.left
>= control
.left
- m_macHorizontalBorder
) ;
563 wxASSERT( meta
.right
<= control
.right
+ m_macHorizontalBorder
) ;
564 wxASSERT( meta
.top
>= control
.top
- m_macVerticalBorder
) ;
565 wxASSERT( meta
.bottom
<= control
.bottom
+ m_macVerticalBorder
) ;
573 bool wxControl::Show(bool show
)
575 if ( !wxWindow::Show( show
) )
578 if ( (ControlHandle
) m_macControl
)
582 if ( m_macControlIsShown
)
584 ::UMAHideControl( (ControlHandle
) m_macControl
) ;
585 m_macControlIsShown
= false ;
590 if ( MacIsReallyShown() && !m_macControlIsShown
)
592 ::UMAShowControl( (ControlHandle
) m_macControl
) ;
593 m_macControlIsShown
= true ;
600 bool wxControl::Enable(bool enable
)
602 if ( !wxWindow::Enable(enable
) )
605 if ( (ControlHandle
) m_macControl
)
608 UMAActivateControl( (ControlHandle
) m_macControl
) ;
610 UMADeactivateControl( (ControlHandle
) m_macControl
) ;
615 void wxControl::Refresh(bool eraseBack
, const wxRect
*rect
)
617 wxWindow::Refresh( eraseBack
, rect
) ;
620 void wxControl::MacRedrawControl()
622 if ( (ControlHandle
) m_macControl
&& MacGetRootWindow() && m_macControlIsShown
)
624 wxClientDC
dc(this) ;
625 wxMacPortSetter
helper(&dc
) ;
626 wxMacWindowClipper
clipper(this) ;
627 wxDC::MacSetupBackgroundForCurrentPort( MacGetBackgroundBrush() ) ;
628 UMADrawControl( (ControlHandle
) m_macControl
) ;
632 void wxControl::OnPaint(wxPaintEvent
& event
)
634 if ( (ControlHandle
) m_macControl
)
637 wxMacPortSetter
helper(&dc
) ;
638 wxMacWindowClipper
clipper(this) ;
639 wxDC::MacSetupBackgroundForCurrentPort( MacGetBackgroundBrush() ) ;
640 UMADrawControl( (ControlHandle
) m_macControl
) ;
647 void wxControl::OnEraseBackground(wxEraseEvent
& event
)
649 wxWindow::OnEraseBackground( event
) ;
652 void wxControl::OnKeyDown( wxKeyEvent
&event
)
654 if ( (ControlHandle
) m_macControl
== NULL
)
663 GetEventParameter( (EventRef
) wxTheApp
->MacGetCurrentEvent(), kEventParamKeyMacCharCodes
, typeChar
, NULL
,sizeof(char), NULL
,&charCode
);
664 GetEventParameter( (EventRef
) wxTheApp
->MacGetCurrentEvent(), kEventParamKeyCode
, typeUInt32
, NULL
, sizeof(UInt32
), NULL
, &keyCode
);
665 GetEventParameter((EventRef
) wxTheApp
->MacGetCurrentEvent(), kEventParamKeyModifiers
, typeUInt32
, NULL
, sizeof(UInt32
), NULL
, &modifiers
);
667 ::HandleControlKey( (ControlHandle
) m_macControl
, keyCode
, charCode
, modifiers
) ;
670 EventRecord
*ev
= (EventRecord
*) wxTheApp
->MacGetCurrentEvent() ;
673 keychar
= short(ev
->message
& charCodeMask
);
674 keycode
= short(ev
->message
& keyCodeMask
) >> 8 ;
676 ::HandleControlKey( (ControlHandle
) m_macControl
, keycode
, keychar
, ev
->modifiers
) ;
680 void wxControl::OnMouseEvent( wxMouseEvent
&event
)
682 if ( (ControlHandle
) m_macControl
== NULL
)
688 if (event
.GetEventType() == wxEVT_LEFT_DOWN
|| event
.GetEventType() == wxEVT_LEFT_DCLICK
)
694 MacClientToRootWindow( &x
, &y
) ;
696 ControlHandle control
;
705 if ( !event
.m_leftDown
&& !event
.m_rightDown
)
706 modifiers
|= btnState
;
708 if ( event
.m_shiftDown
)
709 modifiers
|= shiftKey
;
711 if ( event
.m_controlDown
)
712 modifiers
|= controlKey
;
714 if ( event
.m_altDown
)
715 modifiers
|= optionKey
;
717 if ( event
.m_metaDown
)
718 modifiers
|= cmdKey
;
720 control
= (ControlHandle
) m_macControl
;
721 if ( control
&& ::IsControlActive( control
) )
724 controlpart
= ::HandleControlClick( control
, localwhere
, modifiers
, (ControlActionUPP
) -1 ) ;
725 wxTheApp
->s_lastMouseDown
= 0 ;
726 if ( control
&& controlpart
!= kControlNoPart
)
728 MacHandleControlClick( control
, controlpart
, false /* mouse not down anymore */ ) ;
740 bool wxControl::MacCanFocus() const
742 if ( (ControlHandle
) m_macControl
== NULL
)
748 void wxControl::MacHandleControlClick( WXWidget control
, wxInt16 controlpart
, bool WXUNUSED( mouseStillDown
) )
750 wxASSERT_MSG( (ControlHandle
) m_macControl
!= NULL
, wxT("No valid mac control") ) ;