X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/14c9cbdb1e64d22c6ba588d2e46374be393ff7ea..37de49a3225b9e87ec950f73d361335c73ad3f7f:/src/mac/carbon/window.cpp diff --git a/src/mac/carbon/window.cpp b/src/mac/carbon/window.cpp index 868a2c03c5..9088d2f57a 100644 --- a/src/mac/carbon/window.cpp +++ b/src/mac/carbon/window.cpp @@ -1,11 +1,11 @@ ///////////////////////////////////////////////////////////////////////////// // Name: windows.cpp // Purpose: wxWindowMac -// Author: AUTHOR +// Author: Stefan Csomor // Modified by: -// Created: ??/??/98 +// Created: 1998-01-01 // RCS-ID: $Id$ -// Copyright: (c) AUTHOR +// Copyright: (c) Stefan Csomor // Licence: wxWindows licence ///////////////////////////////////////////////////////////////////////////// @@ -23,20 +23,22 @@ #include "wx/panel.h" #include "wx/layout.h" #include "wx/dialog.h" -#include "wx/listbox.h" #include "wx/scrolbar.h" #include "wx/statbox.h" #include "wx/button.h" #include "wx/settings.h" #include "wx/msgdlg.h" #include "wx/frame.h" -#include "wx/notebook.h" -#include "wx/tabctrl.h" #include "wx/tooltip.h" #include "wx/statusbr.h" #include "wx/menuitem.h" #include "wx/spinctrl.h" #include "wx/log.h" +#include "wx/geometry.h" +#include "wx/textctrl.h" + +#include "wx/toolbar.h" +#include "wx/dc.h" #if wxUSE_CARET #include "wx/caret.h" @@ -50,6 +52,14 @@ #ifndef __DARWIN__ #include #include +#include +#include +#endif + +#if TARGET_API_MAC_OSX +#ifndef __HIVIEW__ + #include +#endif #endif #if wxUSE_DRAG_AND_DROP @@ -59,7 +69,6 @@ #include extern wxList wxPendingDelete; -wxWindowMac* gFocusWindow = NULL ; #ifdef __WXUNIVERSAL__ IMPLEMENT_ABSTRACT_CLASS(wxWindowMac, wxWindowBase) @@ -70,12 +79,13 @@ wxWindowMac* gFocusWindow = NULL ; #if !USE_SHARED_LIBRARY BEGIN_EVENT_TABLE(wxWindowMac, wxWindowBase) - EVT_NC_PAINT(wxWindowMac::OnNcPaint) - EVT_ERASE_BACKGROUND(wxWindowMac::OnEraseBackground) - EVT_SYS_COLOUR_CHANGED(wxWindowMac::OnSysColourChanged) - EVT_INIT_DIALOG(wxWindowMac::OnInitDialog) - EVT_IDLE(wxWindowMac::OnIdle) - EVT_SET_FOCUS(wxWindowMac::OnSetFocus) + EVT_NC_PAINT(wxWindowMac::OnNcPaint) + EVT_ERASE_BACKGROUND(wxWindowMac::OnEraseBackground) +// TODO EVT_PAINT(wxWindowMac::OnPaint) + EVT_SYS_COLOUR_CHANGED(wxWindowMac::OnSysColourChanged) + EVT_INIT_DIALOG(wxWindowMac::OnInitDialog) +// EVT_SET_FOCUS(wxWindowMac::OnSetFocus) + EVT_MOUSE_EVENTS(wxWindowMac::OnMouseEvent) END_EVENT_TABLE() #endif @@ -87,64 +97,532 @@ END_EVENT_TABLE() #define wxMAC_USE_THEME_BORDER 0 +// --------------------------------------------------------------------------- +// Carbon Events +// --------------------------------------------------------------------------- + +extern long wxMacTranslateKey(unsigned char key, unsigned char code) ; +pascal OSStatus wxMacSetupControlBackground( ControlRef iControl , SInt16 iMessage , SInt16 iDepth , Boolean iIsColor ) ; + +#if TARGET_API_MAC_OSX + +#if MAC_OS_X_VERSION_MAX_ALLOWED < MAC_OS_X_VERSION_10_3 +enum { + kEventControlVisibilityChanged = 157 +}; +#endif + +#endif + +static const EventTypeSpec eventList[] = +{ + { kEventClassControl , kEventControlHit } , +#if TARGET_API_MAC_OSX + { kEventClassControl , kEventControlDraw } , + { kEventClassControl , kEventControlVisibilityChanged } , + { kEventClassControl , kEventControlEnabledStateChanged } , + { kEventClassControl , kEventControlHiliteChanged } , + { kEventClassControl , kEventControlSetFocusPart } , + + { kEventClassService , kEventServiceGetTypes }, + { kEventClassService , kEventServiceCopy }, + { kEventClassService , kEventServicePaste }, + + // { kEventClassControl , kEventControlInvalidateForSizeChange } , // 10.3 only +// { kEventClassControl , kEventControlBoundsChanged } , +#endif +} ; + +static pascal OSStatus wxMacWindowControlEventHandler( EventHandlerCallRef handler , EventRef event , void *data ) +{ + OSStatus result = eventNotHandledErr ; + + wxMacCarbonEvent cEvent( event ) ; + + ControlRef controlRef ; + wxWindowMac* thisWindow = (wxWindowMac*) data ; + + cEvent.GetParameter( kEventParamDirectObject , &controlRef ) ; + + switch( GetEventKind( event ) ) + { +#if TARGET_API_MAC_OSX + case kEventControlDraw : + { + RgnHandle updateRgn = NULL ; + + wxRegion visRegion = thisWindow->MacGetVisibleRegion() ; + if ( cEvent.GetParameter(kEventParamRgnHandle, &updateRgn) != noErr ) + { + updateRgn = (RgnHandle) visRegion.GetWXHRGN() ; + } + // GrafPtr myport = cEvent.GetParameter(kEventParamGrafPort,typeGrafPtr) ; + +#if 0 + // in case we would need a coregraphics compliant background erase first + // now usable to track redraws + CGContextRef cgContext = cEvent.GetParameter(kEventParamCGContextRef) ; + if ( thisWindow->MacIsUserPane() ) + { + static float color = 0.5 ; + static channel = 0 ; + HIRect bounds; + HIViewGetBounds( controlRef, &bounds ); + CGContextSetRGBFillColor( cgContext, channel == 0 ? color : 0.5 , + channel == 1 ? color : 0.5 , channel == 2 ? color : 0.5 , 1 ); + CGContextFillRect( cgContext, bounds ); + color += 0.1 ; + if ( color > 0.9 ) + { + color = 0.5 ; + channel++ ; + if ( channel == 3 ) + channel = 0 ; + } + } +#endif + if ( thisWindow->MacDoRedraw( updateRgn , cEvent.GetTicks() ) ) + result = noErr ; + } + break ; + case kEventControlVisibilityChanged : + thisWindow->MacVisibilityChanged() ; + break ; + case kEventControlEnabledStateChanged : + thisWindow->MacEnabledStateChanged() ; + break ; + case kEventControlHiliteChanged : + thisWindow->MacHiliteChanged() ; + break ; + case kEventControlSetFocusPart : + { + Boolean focusEverything = false ; + ControlPartCode controlPart = cEvent.GetParameter(kEventParamControlPart , typeControlPartCode ); + if ( cEvent.GetParameter(kEventParamControlFocusEverything , &focusEverything ) == noErr ) + { + } + if ( controlPart == kControlFocusNoPart ) + { + #if wxUSE_CARET + if ( thisWindow->GetCaret() ) + { + thisWindow->GetCaret()->OnKillFocus(); + } + #endif // wxUSE_CARET + wxFocusEvent event(wxEVT_KILL_FOCUS, thisWindow->GetId()); + event.SetEventObject(thisWindow); + thisWindow->GetEventHandler()->ProcessEvent(event) ; + } + else + { + // panel wants to track the window which was the last to have focus in it + wxChildFocusEvent eventFocus(thisWindow); + thisWindow->GetEventHandler()->ProcessEvent(eventFocus); + + #if wxUSE_CARET + if ( thisWindow->GetCaret() ) + { + thisWindow->GetCaret()->OnSetFocus(); + } + #endif // wxUSE_CARET + + wxFocusEvent event(wxEVT_SET_FOCUS, thisWindow->GetId()); + event.SetEventObject(thisWindow); + thisWindow->GetEventHandler()->ProcessEvent(event) ; + } + if ( thisWindow->MacIsUserPane() ) + result = noErr ; + } + break ; +#endif + case kEventControlHit : + { + result = thisWindow->MacControlHit( handler , event ) ; + } + break ; + default : + break ; + } + return result ; +} + +static pascal OSStatus wxMacWindowServiceEventHandler( EventHandlerCallRef handler , EventRef event , void *data ) +{ + OSStatus result = eventNotHandledErr ; + + wxMacCarbonEvent cEvent( event ) ; + + ControlRef controlRef ; + wxWindowMac* thisWindow = (wxWindowMac*) data ; + wxTextCtrl* textCtrl = wxDynamicCast( thisWindow , wxTextCtrl ) ; + cEvent.GetParameter( kEventParamDirectObject , &controlRef ) ; + + switch( GetEventKind( event ) ) + { + case kEventServiceGetTypes : + if( textCtrl ) + { + long from, to ; + textCtrl->GetSelection( &from , &to ) ; + + CFMutableArrayRef copyTypes = 0 , pasteTypes = 0; + if( from != to ) + copyTypes = cEvent.GetParameter< CFMutableArrayRef >( kEventParamServiceCopyTypes , typeCFMutableArrayRef ) ; + if ( textCtrl->IsEditable() ) + pasteTypes = cEvent.GetParameter< CFMutableArrayRef >( kEventParamServicePasteTypes , typeCFMutableArrayRef ) ; + + static const OSType textDataTypes[] = { kTXNTextData /* , 'utxt' , 'PICT', 'MooV', 'AIFF' */ }; + for ( size_t i = 0 ; i < WXSIZEOF(textDataTypes) ; ++i ) + { + CFStringRef typestring = CreateTypeStringWithOSType(textDataTypes[i]); + if ( typestring ) + { + if ( copyTypes ) + CFArrayAppendValue (copyTypes, typestring) ; + if ( pasteTypes ) + CFArrayAppendValue (pasteTypes, typestring) ; + CFRelease( typestring ) ; + } + } + result = noErr ; + } + break ; + case kEventServiceCopy : + if ( textCtrl ) + { + long from, to ; + textCtrl->GetSelection( &from , &to ) ; + wxString val = textCtrl->GetValue() ; + val = val.Mid( from , to - from ) ; + ScrapRef scrapRef = cEvent.GetParameter< ScrapRef > ( kEventParamScrapRef , typeScrapRef ) ; + verify_noerr( ClearScrap( &scrapRef ) ) ; + verify_noerr( PutScrapFlavor( scrapRef , kTXNTextData , 0 , val.Length() , val.c_str() ) ) ; + result = noErr ; + } + break ; + case kEventServicePaste : + if ( textCtrl ) + { + ScrapRef scrapRef = cEvent.GetParameter< ScrapRef > ( kEventParamScrapRef , typeScrapRef ) ; + Size textSize, pastedSize ; + verify_noerr( GetScrapFlavorSize (scrapRef, kTXNTextData, &textSize) ) ; + textSize++ ; + char *content = new char[textSize] ; + GetScrapFlavorData (scrapRef, kTXNTextData, &pastedSize, content ); + content[textSize-1] = 0 ; +#if wxUSE_UNICODE + textCtrl->WriteText( wxString( content , wxConvLocal ) ); +#else + textCtrl->WriteText( wxString( content ) ) ; +#endif + delete[] content ; + result = noErr ; + } + break ; + } + + return result ; +} + +pascal OSStatus wxMacWindowEventHandler( EventHandlerCallRef handler , EventRef event , void *data ) +{ + OSStatus result = eventNotHandledErr ; + + switch ( GetEventClass( event ) ) + { + case kEventClassControl : + result = wxMacWindowControlEventHandler( handler, event, data ) ; + break ; + case kEventClassService : + result = wxMacWindowServiceEventHandler( handler, event , data ) ; + default : + break ; + } + return result ; +} + +DEFINE_ONE_SHOT_HANDLER_GETTER( wxMacWindowEventHandler ) + +// --------------------------------------------------------------------------- +// UserPane events for non OSX builds +// --------------------------------------------------------------------------- + +static pascal void wxMacControlUserPaneDrawProc(ControlRef control, SInt16 part) +{ + wxWindow * win = wxFindControlFromMacControl(control) ; + wxCHECK_RET( win , wxT("Callback from unkown control") ) ; + win->MacControlUserPaneDrawProc(part) ; +} + +static pascal ControlPartCode wxMacControlUserPaneHitTestProc(ControlRef control, Point where) +{ + wxWindow * win = wxFindControlFromMacControl(control) ; + wxCHECK_MSG( win , kControlNoPart , wxT("Callback from unkown control") ) ; + return win->MacControlUserPaneHitTestProc(where.h , where.v) ; +} + +static pascal ControlPartCode wxMacControlUserPaneTrackingProc(ControlRef control, Point startPt, ControlActionUPP actionProc) +{ + wxWindow * win = wxFindControlFromMacControl(control) ; + wxCHECK_MSG( win , kControlNoPart , wxT("Callback from unkown control") ) ; + return win->MacControlUserPaneTrackingProc( startPt.h , startPt.v , (void*) actionProc) ; +} + +static pascal void wxMacControlUserPaneIdleProc(ControlRef control) +{ + wxWindow * win = wxFindControlFromMacControl(control) ; + wxCHECK_RET( win , wxT("Callback from unkown control") ) ; + win->MacControlUserPaneIdleProc() ; +} + +static pascal ControlPartCode wxMacControlUserPaneKeyDownProc(ControlRef control, SInt16 keyCode, SInt16 charCode, SInt16 modifiers) +{ + wxWindow * win = wxFindControlFromMacControl(control) ; + wxCHECK_MSG( win , kControlNoPart , wxT("Callback from unkown control") ) ; + return win->MacControlUserPaneKeyDownProc(keyCode,charCode,modifiers) ; +} + +static pascal void wxMacControlUserPaneActivateProc(ControlRef control, Boolean activating) +{ + wxWindow * win = wxFindControlFromMacControl(control) ; + wxCHECK_RET( win , wxT("Callback from unkown control") ) ; + win->MacControlUserPaneActivateProc(activating) ; +} + +static pascal ControlPartCode wxMacControlUserPaneFocusProc(ControlRef control, ControlFocusPart action) +{ + wxWindow * win = wxFindControlFromMacControl(control) ; + wxCHECK_MSG( win , kControlNoPart , wxT("Callback from unkown control") ) ; + return win->MacControlUserPaneFocusProc(action) ; +} + +static pascal void wxMacControlUserPaneBackgroundProc(ControlRef control, ControlBackgroundPtr info) +{ + wxWindow * win = wxFindControlFromMacControl(control) ; + wxCHECK_RET( win , wxT("Callback from unkown control") ) ; + win->MacControlUserPaneBackgroundProc(info) ; +} + +void wxWindowMac::MacControlUserPaneDrawProc(wxInt16 part) +{ + RgnHandle rgn = NewRgn() ; + GetClip( rgn ) ; + wxMacWindowStateSaver sv( this ) ; + SectRgn( rgn , (RgnHandle) MacGetVisibleRegion().GetWXHRGN() , rgn ) ; + MacDoRedraw( rgn , 0 ) ; + DisposeRgn( rgn ) ; +} + +wxInt16 wxWindowMac::MacControlUserPaneHitTestProc(wxInt16 x, wxInt16 y) +{ + return kControlNoPart ; +} + +wxInt16 wxWindowMac::MacControlUserPaneTrackingProc(wxInt16 x, wxInt16 y, void* actionProc) +{ + return kControlNoPart ; +} + +void wxWindowMac::MacControlUserPaneIdleProc() +{ +} + +wxInt16 wxWindowMac::MacControlUserPaneKeyDownProc(wxInt16 keyCode, wxInt16 charCode, wxInt16 modifiers) +{ + return kControlNoPart ; +} + +void wxWindowMac::MacControlUserPaneActivateProc(bool activating) +{ +} + +wxInt16 wxWindowMac::MacControlUserPaneFocusProc(wxInt16 action) +{ + return kControlNoPart ; +} + +void wxWindowMac::MacControlUserPaneBackgroundProc(void* info) +{ +} + +ControlUserPaneDrawUPP gControlUserPaneDrawUPP = NULL ; +ControlUserPaneHitTestUPP gControlUserPaneHitTestUPP = NULL ; +ControlUserPaneTrackingUPP gControlUserPaneTrackingUPP = NULL ; +ControlUserPaneIdleUPP gControlUserPaneIdleUPP = NULL ; +ControlUserPaneKeyDownUPP gControlUserPaneKeyDownUPP = NULL ; +ControlUserPaneActivateUPP gControlUserPaneActivateUPP = NULL ; +ControlUserPaneFocusUPP gControlUserPaneFocusUPP = NULL ; +ControlUserPaneBackgroundUPP gControlUserPaneBackgroundUPP = NULL ; // =========================================================================== // implementation // =========================================================================== +wxList wxWinMacControlList(wxKEY_INTEGER); -// ---------------------------------------------------------------------------- -// constructors and such -// ---------------------------------------------------------------------------- +wxWindow *wxFindControlFromMacControl(ControlRef inControl ) +{ + wxNode *node = wxWinMacControlList.Find((long)inControl); + if (!node) + return NULL; + return (wxControl *)node->GetData(); +} -void wxWindowMac::Init() +void wxAssociateControlWithMacControl(ControlRef inControl, wxWindow *control) +{ + // adding NULL ControlRef is (first) surely a result of an error and + // (secondly) breaks native event processing + wxCHECK_RET( inControl != (ControlRef) NULL, wxT("attempt to add a NULL WindowRef to window list") ); + + if ( !wxWinMacControlList.Find((long)inControl) ) + wxWinMacControlList.Append((long)inControl, control); +} + +void wxRemoveMacControlAssociation(wxWindow *control) +{ + wxWinMacControlList.DeleteObject(control); +} + +// UPP functions +ControlActionUPP wxMacLiveScrollbarActionUPP = NULL ; + +ControlColorUPP wxMacSetupControlBackgroundUPP = NULL ; + +// we have to setup the brush in the current port and return noErr +// or return an error code so that the control manager walks further up the +// hierarchy to find a correct background + +pascal OSStatus wxMacSetupControlBackground( ControlRef iControl , SInt16 iMessage , SInt16 iDepth , Boolean iIsColor ) +{ + OSStatus status = paramErr ; + switch( iMessage ) + { + case kControlMsgApplyTextColor : + break ; + case kControlMsgSetUpBackground : + { + wxWindow* wx = (wxWindow*) wxFindControlFromMacControl( iControl ) ; + if ( wx != NULL ) + { + /* + const wxBrush &brush = wx->MacGetBackgroundBrush() ; + if ( brush.Ok() ) + { + + wxDC::MacSetupBackgroundForCurrentPort( brush ) ; + */ + // this clipping is only needed for non HIView + + RgnHandle clip = NewRgn() ; + int x = 0 , y = 0; + + wx->MacWindowToRootWindow( &x,&y ) ; + CopyRgn( (RgnHandle) wx->MacGetVisibleRegion().GetWXHRGN() , clip ) ; + OffsetRgn( clip , x , y ) ; + SetClip( clip ) ; + DisposeRgn( clip ) ; + + status = noErr ; + /* + } + else if ( wx->MacIsUserPane() ) + { + // if we don't have a valid brush for such a control, we have to call the + // setup of our parent ourselves + status = SetUpControlBackground( (ControlRef) wx->GetParent()->GetHandle() , iDepth , iIsColor ) ; + } + */ + } + } + break ; + default : + break ; + } + return status ; +} + + +pascal void wxMacLiveScrollbarActionProc( ControlRef control , ControlPartCode partCode ) ; +pascal void wxMacLiveScrollbarActionProc( ControlRef control , ControlPartCode partCode ) { - // generic - InitBase(); + if ( partCode != 0) + { + wxWindow* wx = wxFindControlFromMacControl( control ) ; + if ( wx ) + { + wx->MacHandleControlClick( (WXWidget) control , partCode , true /* stillDown */ ) ; + } + } +} - // MSW specific - m_doubleClickAllowed = 0; - m_winCaptured = FALSE; +// ---------------------------------------------------------------------------- + // constructors and such +// ---------------------------------------------------------------------------- - m_isBeingDeleted = FALSE; +wxWindowMac::wxWindowMac() +{ + Init(); +} - m_useCtl3D = FALSE; - m_mouseInWindow = FALSE; +wxWindowMac::wxWindowMac(wxWindowMac *parent, + wxWindowID id, + const wxPoint& pos , + const wxSize& size , + long style , + const wxString& name ) +{ + Init(); + Create(parent, id, pos, size, style, name); +} - m_xThumbSize = 0; - m_yThumbSize = 0; +void wxWindowMac::Init() +{ + m_peer = NULL ; + m_frozenness = 0 ; +#if WXWIN_COMPATIBILITY_2_4 m_backgroundTransparent = FALSE; +#endif // as all windows are created with WS_VISIBLE style... m_isShown = TRUE; - m_x = 0; - m_y = 0 ; - m_width = 0 ; - m_height = 0 ; - m_hScrollBar = NULL ; m_vScrollBar = NULL ; + m_macBackgroundBrush = wxNullBrush ; + + m_macIsUserPane = TRUE; + + // make sure all proc ptrs are available + + if ( gControlUserPaneDrawUPP == NULL ) + { + gControlUserPaneDrawUPP = NewControlUserPaneDrawUPP( wxMacControlUserPaneDrawProc ) ; + gControlUserPaneHitTestUPP = NewControlUserPaneHitTestUPP( wxMacControlUserPaneHitTestProc ) ; + gControlUserPaneTrackingUPP = NewControlUserPaneTrackingUPP( wxMacControlUserPaneTrackingProc ) ; + gControlUserPaneIdleUPP = NewControlUserPaneIdleUPP( wxMacControlUserPaneIdleProc ) ; + gControlUserPaneKeyDownUPP = NewControlUserPaneKeyDownUPP( wxMacControlUserPaneKeyDownProc ) ; + gControlUserPaneActivateUPP = NewControlUserPaneActivateUPP( wxMacControlUserPaneActivateProc ) ; + gControlUserPaneFocusUPP = NewControlUserPaneFocusUPP( wxMacControlUserPaneFocusProc ) ; + gControlUserPaneBackgroundUPP = NewControlUserPaneBackgroundUPP( wxMacControlUserPaneBackgroundProc ) ; + } + if ( wxMacLiveScrollbarActionUPP == NULL ) + { + wxMacLiveScrollbarActionUPP = NewControlActionUPP( wxMacLiveScrollbarActionProc ); + } + + if ( wxMacSetupControlBackgroundUPP == NULL ) + { + wxMacSetupControlBackgroundUPP = NewControlColorUPP( wxMacSetupControlBackground ) ; + } - m_label = wxEmptyString; + // we need a valid font for the encodings + wxWindowBase::SetFont(wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT)); } // Destructor wxWindowMac::~wxWindowMac() { - // deleting a window while it is shown invalidates the region - if ( IsShown() ) { - wxWindowMac* iter = this ; - while( iter ) { - if ( iter->IsTopLevel() ) - { - Refresh() ; - break ; - } - iter = iter->GetParent() ; - - } - } + SendDestroyEvent(); m_isBeingDeleted = TRUE; @@ -163,21 +641,62 @@ wxWindowMac::~wxWindowMac() } } #endif // __WXUNIVERSAL__ - - if ( s_lastMouseWindow == this ) + + // wxRemoveMacControlAssociation( this ) ; + // If we delete an item, we should initialize the parent panel, + // because it could now be invalid. + wxWindow *parent = GetParent() ; + if ( parent ) { - s_lastMouseWindow = NULL ; + if (parent->GetDefaultItem() == (wxButton*) this) + parent->SetDefaultItem(NULL); + } + if ( m_peer && m_peer->Ok() ) + { + // in case the callback might be called during destruction + wxRemoveMacControlAssociation( this) ; + ::SetControlColorProc( *m_peer , NULL ) ; + ::DisposeControl( *m_peer ) ; } - if ( gFocusWindow == this ) + if ( g_MacLastWindow == this ) { - gFocusWindow = NULL ; + g_MacLastWindow = NULL ; } - if ( m_parent ) - m_parent->RemoveChild(this); + wxFrame* frame = wxDynamicCast( wxGetTopLevelParent( this ) , wxFrame ) ; + if ( frame ) + { + if ( frame->GetLastFocus() == this ) + frame->SetLastFocus( NULL ) ; + } DestroyChildren(); + + // delete our drop target if we've got one +#if wxUSE_DRAG_AND_DROP + if ( m_dropTarget != NULL ) + { + delete m_dropTarget; + m_dropTarget = NULL; + } +#endif // wxUSE_DRAG_AND_DROP + delete m_peer ; +} + +WXWidget wxWindowMac::GetHandle() const +{ + return (WXWidget) (ControlRef) *m_peer ; +} + + +void wxWindowMac::MacInstallEventHandler() +{ + wxAssociateControlWithMacControl( *m_peer , this ) ; + InstallControlEventHandler( *m_peer, GetwxMacWindowEventHandlerUPP(), + GetEventTypeCount(eventList), eventList, this, + (EventHandlerRef *)&m_macControlEventHandler); + } // Constructor @@ -194,11 +713,47 @@ bool wxWindowMac::Create(wxWindowMac *parent, wxWindowID id, parent->AddChild(this); - m_x = (int)pos.x; - m_y = (int)pos.y; - AdjustForParentClientOrigin(m_x, m_y, wxSIZE_USE_EXISTING); - m_width = WidthDefault( size.x ); - m_height = HeightDefault( size.y ) ; + m_windowVariant = parent->GetWindowVariant() ; + + if ( m_macIsUserPane ) + { + Rect bounds = wxMacGetBoundsForControl( this , pos , size ) ; + + UInt32 features = 0 + | kControlSupportsEmbedding +// | kControlSupportsLiveFeedback +// | kControlHasSpecialBackground +// | kControlSupportsCalcBestRect +// | kControlHandlesTracking + | kControlSupportsFocus +// | kControlWantsActivate +// | kControlWantsIdle + ; + + m_peer = new wxMacControl() ; + ::CreateUserPaneControl( MAC_WXHWND(GetParent()->MacGetTopLevelWindowRef()) , &bounds, features , *m_peer); + + + MacPostControlCreate(pos,size) ; +#if !TARGET_API_MAC_OSX + SetControlData(*m_peer,kControlEntireControl,kControlUserPaneDrawProcTag, + sizeof(gControlUserPaneDrawUPP),(Ptr) &gControlUserPaneDrawUPP); + SetControlData(*m_peer,kControlEntireControl,kControlUserPaneHitTestProcTag, + sizeof(gControlUserPaneHitTestUPP),(Ptr) &gControlUserPaneHitTestUPP); + SetControlData(*m_peer,kControlEntireControl,kControlUserPaneTrackingProcTag, + sizeof(gControlUserPaneTrackingUPP),(Ptr) &gControlUserPaneTrackingUPP); + SetControlData(*m_peer,kControlEntireControl,kControlUserPaneIdleProcTag, + sizeof(gControlUserPaneIdleUPP),(Ptr) &gControlUserPaneIdleUPP); + SetControlData(*m_peer,kControlEntireControl,kControlUserPaneKeyDownProcTag, + sizeof(gControlUserPaneKeyDownUPP),(Ptr) &gControlUserPaneKeyDownUPP); + SetControlData(*m_peer,kControlEntireControl,kControlUserPaneActivateProcTag, + sizeof(gControlUserPaneActivateUPP),(Ptr) &gControlUserPaneActivateUPP); + SetControlData(*m_peer,kControlEntireControl,kControlUserPaneFocusProcTag, + sizeof(gControlUserPaneFocusUPP),(Ptr) &gControlUserPaneFocusUPP); + SetControlData(*m_peer,kControlEntireControl,kControlUserPaneBackgroundProcTag, + sizeof(gControlUserPaneBackgroundUPP),(Ptr) &gControlUserPaneBackgroundUPP); +#endif + } #ifndef __WXUNIVERSAL__ // Don't give scrollbars to wxControls unless they ask for them if ( (! IsKindOf(CLASSINFO(wxControl)) && ! IsKindOf(CLASSINFO(wxStatusBar))) || @@ -207,74 +762,240 @@ bool wxWindowMac::Create(wxWindowMac *parent, wxWindowID id, MacCreateScrollBars( style ) ; } #endif + + wxWindowCreateEvent event(this); + GetEventHandler()->AddPendingEvent(event); + return TRUE; } -void wxWindowMac::SetFocus() +void wxWindowMac::MacPostControlCreate(const wxPoint& pos, const wxSize& size) { - if ( gFocusWindow == this ) - return ; + wxASSERT_MSG( m_peer != NULL && m_peer->Ok() , wxT("No valid mac control") ) ; - if ( AcceptsFocus() ) + ::SetControlReference( *m_peer , (long) this ) ; + + MacInstallEventHandler(); + + ControlRef container = (ControlRef) GetParent()->GetHandle() ; + wxASSERT_MSG( container != NULL , wxT("No valid mac container control") ) ; + ::EmbedControl( *m_peer , container ) ; + + // adjust font, controlsize etc + DoSetWindowVariant( m_windowVariant ) ; + +#if !TARGET_API_MAC_OSX + // eventually we can fix some clipping issues be reactivating this hook + //if ( m_macIsUserPane ) + // SetControlColorProc( *m_peer , wxMacSetupControlBackgroundUPP ) ; +#endif + + UMASetControlTitle( *m_peer , wxStripMenuCodes(m_label) , m_font.GetEncoding() ) ; + + if (!m_macIsUserPane) { - if (gFocusWindow ) - { - #if wxUSE_CARET - // Deal with caret - if ( gFocusWindow->m_caret ) - { - gFocusWindow->m_caret->OnKillFocus(); - } - #endif // wxUSE_CARET - #ifndef __WXUNIVERSAL__ - wxControl* control = wxDynamicCast( gFocusWindow , wxControl ) ; - if ( control && control->GetMacControl() ) - { - UMASetKeyboardFocus( (WindowRef) gFocusWindow->MacGetRootWindow() , (ControlHandle) control->GetMacControl() , kControlFocusNoPart ) ; - control->MacRedrawControl() ; - } - #endif - wxFocusEvent event(wxEVT_KILL_FOCUS, gFocusWindow->m_windowId); - event.SetEventObject(gFocusWindow); - gFocusWindow->GetEventHandler()->ProcessEvent(event) ; - } - gFocusWindow = this ; - { - #if wxUSE_CARET - // Deal with caret - if ( m_caret ) + SetInitialBestSize(size); + } + + SetCursor( *wxSTANDARD_CURSOR ) ; +} + +void wxWindowMac::DoSetWindowVariant( wxWindowVariant variant ) +{ + // Don't assert, in case we set the window variant before + // the window is created + // wxASSERT( m_peer->Ok() ) ; + + m_windowVariant = variant ; + + if (m_peer == NULL || !m_peer->Ok()) + return; + + ControlSize size ; + ThemeFontID themeFont = kThemeSystemFont ; + + // we will get that from the settings later + // and make this NORMAL later, but first + // we have a few calculations that we must fix + + switch ( variant ) + { + case wxWINDOW_VARIANT_NORMAL : + size = kControlSizeNormal; + themeFont = kThemeSystemFont ; + break ; + case wxWINDOW_VARIANT_SMALL : + size = kControlSizeSmall; + themeFont = kThemeSmallSystemFont ; + break ; + case wxWINDOW_VARIANT_MINI : + if (UMAGetSystemVersion() >= 0x1030 ) { - m_caret->OnSetFocus(); + // not always defined in the headers + size = 3 ; + themeFont = 109 ; } - #endif // wxUSE_CARET - // panel wants to track the window which was the last to have focus in it - wxChildFocusEvent eventFocus(this); - GetEventHandler()->ProcessEvent(eventFocus); - - #ifndef __WXUNIVERSAL__ - wxControl* control = wxDynamicCast( gFocusWindow , wxControl ) ; - if ( control && control->GetMacControl() ) + else { - UMASetKeyboardFocus( (WindowRef) gFocusWindow->MacGetRootWindow() , (ControlHandle) control->GetMacControl() , kControlFocusNextPart ) ; + size = kControlSizeSmall; + themeFont = kThemeSmallSystemFont ; } - #endif - wxFocusEvent event(wxEVT_SET_FOCUS, m_windowId); - event.SetEventObject(this); - GetEventHandler()->ProcessEvent(event) ; - } + break ; + case wxWINDOW_VARIANT_LARGE : + size = kControlSizeLarge; + themeFont = kThemeSystemFont ; + break ; + default: + wxFAIL_MSG(_T("unexpected window variant")); + break ; } + ::SetControlData( *m_peer , kControlEntireControl, kControlSizeTag, sizeof( ControlSize ), &size ); + + wxFont font ; + font.MacCreateThemeFont( themeFont ) ; + SetFont( font ) ; } -bool wxWindowMac::Enable(bool enable) +void wxWindowMac::MacUpdateControlFont() { - if ( !wxWindowBase::Enable(enable) ) - return FALSE; + ControlFontStyleRec fontStyle; + if ( m_font.MacGetThemeFontID() != kThemeCurrentPortFont ) + { + switch( m_font.MacGetThemeFontID() ) + { + case kThemeSmallSystemFont : fontStyle.font = kControlFontSmallSystemFont ; break ; + case 109 /*mini font */ : fontStyle.font = -5 ; break ; + case kThemeSystemFont : fontStyle.font = kControlFontBigSystemFont ; break ; + default : fontStyle.font = kControlFontBigSystemFont ; break ; + } + fontStyle.flags = kControlUseFontMask ; + } + else + { + fontStyle.font = m_font.MacGetFontNum() ; + fontStyle.style = m_font.MacGetFontStyle() ; + fontStyle.size = m_font.MacGetFontSize() ; + fontStyle.flags = kControlUseFontMask | kControlUseFaceMask | kControlUseSizeMask ; + } - MacSuperEnabled( enable ) ; + fontStyle.just = teJustLeft ; + fontStyle.flags |= kControlUseJustMask ; + if ( ( GetWindowStyle() & wxALIGN_MASK ) & wxALIGN_CENTER_HORIZONTAL ) + fontStyle.just = teJustCenter ; + else if ( ( GetWindowStyle() & wxALIGN_MASK ) & wxALIGN_RIGHT ) + fontStyle.just = teJustRight ; - return TRUE; + + fontStyle.foreColor = MAC_WXCOLORREF(GetForegroundColour().GetPixel() ) ; + fontStyle.flags |= kControlUseForeColorMask ; + + ::SetControlFontStyle( *m_peer , &fontStyle ); + Refresh() ; +} + +bool wxWindowMac::SetFont(const wxFont& font) +{ + bool retval = !wxWindowBase::SetFont( font ) ; + + MacUpdateControlFont() ; + + return retval; +} + +bool wxWindowMac::SetForegroundColour(const wxColour& col ) +{ + if ( !wxWindowBase::SetForegroundColour(col) ) + return false ; + + MacUpdateControlFont() ; + + return true ; +} + +bool wxWindowMac::SetBackgroundColour(const wxColour& col ) +{ + if ( !wxWindowBase::SetBackgroundColour(col) && m_hasBgCol ) + return false ; + + wxBrush brush ; + if ( col == wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE) ) + { + brush.MacSetTheme( kThemeBrushDocumentWindowBackground ) ; + } + else if ( col == wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE ) ) + { + brush.MacSetTheme( kThemeBrushDialogBackgroundActive ) ; + } + else + { + brush.SetColour( col ) ; + } + MacSetBackgroundBrush( brush ) ; + + MacUpdateControlFont() ; + + return true ; +} + + +bool wxWindowMac::MacCanFocus() const +{ + // there is currently no way to determine whether the window is running in full keyboard + // access mode, therefore we cannot rely on these features, yet the only other way would be + // to issue a SetKeyboardFocus event and verify after whether it succeeded, this would risk problems + // in event handlers... + UInt32 features = 0 ; + GetControlFeatures( *m_peer , &features ) ; + return features & ( kControlSupportsFocus | kControlGetsFocusOnClick ) ; } + +void wxWindowMac::SetFocus() +{ + if ( AcceptsFocus() ) + { +#if !TARGET_API_MAC_OSX + wxWindow* former = FindFocus() ; +#endif + OSStatus err = SetKeyboardFocus( (WindowRef) MacGetTopLevelWindowRef() , (ControlRef) GetHandle() , kControlFocusNextPart ) ; + // as we cannot rely on the control features to find out whether we are in full keyboard mode, we can only + // leave in case of an error + if ( err == errCouldntSetFocus ) + return ; + +#if !TARGET_API_MAC_OSX + // emulate carbon events when running under carbonlib where they are not natively available + if ( former ) + { + EventRef evRef = NULL ; + verify_noerr( MacCreateEvent( NULL , kEventClassControl , kEventControlSetFocusPart , TicksToEventTime( TickCount() ) , kEventAttributeUserEvent , + &evRef ) ); + + wxMacCarbonEvent cEvent( evRef ) ; + cEvent.SetParameter( kEventParamDirectObject , (ControlRef) former->GetHandle() ) ; + cEvent.SetParameter(kEventParamControlPart , typeControlPartCode , kControlFocusNoPart ) ; + + wxMacWindowEventHandler( NULL , evRef , former ) ; + ReleaseEvent(evRef) ; + } + // send new focus event + { + EventRef evRef = NULL ; + verify_noerr( MacCreateEvent( NULL , kEventClassControl , kEventControlSetFocusPart , TicksToEventTime( TickCount() ) , kEventAttributeUserEvent , + &evRef ) ); + + wxMacCarbonEvent cEvent( evRef ) ; + cEvent.SetParameter( kEventParamDirectObject , (ControlRef) GetHandle() ) ; + cEvent.SetParameter(kEventParamControlPart , typeControlPartCode , kControlFocusNextPart ) ; + + wxMacWindowEventHandler( NULL , evRef , this ) ; + ReleaseEvent(evRef) ; + } +#endif + } +} + + void wxWindowMac::DoCaptureMouse() { wxTheApp->s_captureWindow = this ; @@ -294,15 +1015,15 @@ void wxWindowMac::DoReleaseMouse() void wxWindowMac::SetDropTarget(wxDropTarget *pDropTarget) { - if ( m_dropTarget != 0 ) { - delete m_dropTarget; - } + if ( m_dropTarget != 0 ) { + delete m_dropTarget; + } - m_dropTarget = pDropTarget; - if ( m_dropTarget != 0 ) - { - // TODO - } + m_dropTarget = pDropTarget; + if ( m_dropTarget != 0 ) + { + // TODO + } } #endif @@ -313,94 +1034,148 @@ void wxWindowMac::DragAcceptFiles(bool accept) // TODO } +void wxWindowMac::MacGetPositionAndSizeFromControl(int& x, int& y, + int& w, int& h) const +{ + Rect bounds ; + GetControlBounds( *m_peer , &bounds ) ; + + + x = bounds.left ; + y = bounds.top ; + w = bounds.right - bounds.left ; + h = bounds.bottom - bounds.top ; + + wxTopLevelWindow* tlw = wxDynamicCast( this , wxTopLevelWindow ) ; + if ( tlw ) + { + Point tlworigin = { 0 , 0 } ; + QDLocalToGlobalPoint( UMAGetWindowPort( (WindowRef) tlw->MacGetWindowRef() ) , &tlworigin ) ; + x = tlworigin.h ; + y = tlworigin.v ; + } +} + +bool wxWindowMac::MacGetBoundsForControl(const wxPoint& pos, + const wxSize& size, + int& x, int& y, + int& w, int& h , bool adjustOrigin ) const +{ + x = (int)pos.x; + y = (int)pos.y; + // todo the default calls may be used as soon as PostCreateControl Is moved here + w = size.x ; // WidthDefault( size.x ); + h = size.y ; // HeightDefault( size.y ) ; +#if !TARGET_API_MAC_OSX + GetParent()->MacWindowToRootWindow( &x , &y ) ; +#endif + if ( adjustOrigin ) + AdjustForParentClientOrigin( x , y ) ; + return true ; +} + // Get total size void wxWindowMac::DoGetSize(int *x, int *y) const { - if(x) *x = m_width ; - if(y) *y = m_height ; +#if TARGET_API_MAC_OSX + int x1 , y1 , w1 ,h1 ; + MacGetPositionAndSizeFromControl( x1 , y1, w1 ,h1 ) ; + if(x) *x = w1 ; + if(y) *y = h1 ; + +#else + Rect bounds ; + GetControlBounds( *m_peer , &bounds ) ; + if(x) *x = bounds.right - bounds.left ; + if(y) *y = bounds.bottom - bounds.top ; +#endif } void wxWindowMac::DoGetPosition(int *x, int *y) const { - int xx,yy; - - xx = m_x ; - yy = m_y ; - if ( !IsTopLevel() && GetParent()) + #if TARGET_API_MAC_OSX + int x1 , y1 , w1 ,h1 ; + MacGetPositionAndSizeFromControl( x1 , y1, w1 ,h1 ) ; + if ( !IsTopLevel() ) { - wxPoint pt(GetParent()->GetClientAreaOrigin()); - xx -= pt.x; - yy -= pt.y; + wxWindow *parent = GetParent(); + if ( parent ) + { + wxPoint pt(parent->GetClientAreaOrigin()); + x1 -= pt.x ; + y1 -= pt.y ; + } } + if(x) *x = x1 ; + if(y) *y = y1 ; + #else + Rect bounds ; + GetControlBounds( *m_peer , &bounds ) ; + wxCHECK_RET( GetParent() , wxT("Missing Parent") ) ; + + int xx = bounds.left ; + int yy = bounds.top ; + + if ( !GetParent()->IsTopLevel() ) + { + GetControlBounds( (ControlRef) GetParent()->GetHandle() , &bounds ) ; + + xx -= bounds.left ; + yy -= bounds.top ; + } + + wxPoint pt(GetParent()->GetClientAreaOrigin()); + xx -= pt.x; + yy -= pt.y; + if(x) *x = xx; if(y) *y = yy; -} - -#if wxUSE_MENUS -bool wxWindowMac::DoPopupMenu(wxMenu *menu, int x, int y) -{ - menu->SetInvokingWindow(this); - menu->UpdateUI(); - ClientToScreen( &x , &y ) ; - - ::InsertMenu( (MenuHandle) menu->GetHMenu() , -1 ) ; - long menuResult = ::PopUpMenuSelect((MenuHandle) menu->GetHMenu() ,y,x, 0) ; - menu->MacMenuSelect( this , TickCount() , HiWord(menuResult) , LoWord(menuResult) ) ; - ::DeleteMenu( menu->MacGetMenuId() ) ; - menu->SetInvokingWindow(NULL); - - return TRUE; -} #endif +} void wxWindowMac::DoScreenToClient(int *x, int *y) const { - WindowRef window = (WindowRef) MacGetRootWindow() ; - - Point localwhere = {0,0} ; - - if(x) localwhere.h = * x ; - if(y) localwhere.v = * y ; - - GrafPtr port ; - ::GetPort( &port ) ; - ::SetPort( UMAGetWindowPort( window ) ) ; - ::GlobalToLocal( &localwhere ) ; - ::SetPort( port ) ; + WindowRef window = (WindowRef) MacGetTopLevelWindowRef() ; + + wxCHECK_RET( window , wxT("TopLevel Window Missing") ) ; + + { + Point localwhere = {0,0} ; - if(x) *x = localwhere.h ; - if(y) *y = localwhere.v ; + if(x) localwhere.h = * x ; + if(y) localwhere.v = * y ; + + QDGlobalToLocalPoint( GetWindowPort( window ) , &localwhere ) ; + if(x) *x = localwhere.h ; + if(y) *y = localwhere.v ; + } MacRootWindowToWindow( x , y ) ; - if ( x ) - x -= MacGetLeftBorderSize() ; - if ( y ) - y -= MacGetTopBorderSize() ; + + wxPoint origin = GetClientAreaOrigin() ; + if(x) *x -= origin.x ; + if(y) *y -= origin.y ; } void wxWindowMac::DoClientToScreen(int *x, int *y) const { - WindowRef window = (WindowRef) MacGetRootWindow() ; + WindowRef window = (WindowRef) MacGetTopLevelWindowRef() ; + wxCHECK_RET( window , wxT("TopLevel Window Missing") ) ; - if ( x ) - x += MacGetLeftBorderSize() ; - if ( y ) - y += MacGetTopBorderSize() ; + wxPoint origin = GetClientAreaOrigin() ; + if(x) *x += origin.x ; + if(y) *y += origin.y ; MacWindowToRootWindow( x , y ) ; - Point localwhere = { 0,0 }; - if(x) localwhere.h = * x ; - if(y) localwhere.v = * y ; - - GrafPtr port ; - ::GetPort( &port ) ; - ::SetPort( UMAGetWindowPort( window ) ) ; - - ::LocalToGlobal( &localwhere ) ; - ::SetPort( port ) ; - if(x) *x = localwhere.h ; - if(y) *y = localwhere.v ; + { + Point localwhere = { 0,0 }; + if(x) localwhere.h = * x ; + if(y) localwhere.v = * y ; + QDLocalToGlobalPoint( GetWindowPort( window ) , &localwhere ) ; + if(x) *x = localwhere.h ; + if(y) *y = localwhere.v ; + } } void wxWindowMac::MacClientToRootWindow( int *x , int *y ) const @@ -414,66 +1189,137 @@ void wxWindowMac::MacClientToRootWindow( int *x , int *y ) const void wxWindowMac::MacRootWindowToClient( int *x , int *y ) const { - wxPoint origin = GetClientAreaOrigin() ; MacRootWindowToWindow( x , y ) ; + + wxPoint origin = GetClientAreaOrigin() ; if(x) *x -= origin.x ; if(y) *y -= origin.y ; } void wxWindowMac::MacWindowToRootWindow( int *x , int *y ) const { + #if TARGET_API_MAC_OSX + HIPoint pt ; + if ( x ) pt.x = *x ; + if ( y ) pt.y = *y ; + + if ( !IsTopLevel() ) + { + wxTopLevelWindowMac* top = MacGetTopLevelWindow(); + if (top) + HIViewConvertPoint( &pt , *m_peer , (ControlRef) top->GetHandle() ) ; + } + + if ( x ) *x = (int) pt.x ; + if ( y ) *y = (int) pt.y ; + #else if ( !IsTopLevel() ) { - if(x) *x += m_x ; - if(y) *y += m_y ; - GetParent()->MacWindowToRootWindow( x , y ) ; + Rect bounds ; + GetControlBounds( *m_peer , &bounds ) ; + if(x) *x += bounds.left ; + if(y) *y += bounds.top ; } +#endif +} + +void wxWindowMac::MacWindowToRootWindow( short *x , short *y ) const +{ + int x1 , y1 ; + if ( x ) x1 = *x ; + if ( y ) y1 = *y ; + MacWindowToRootWindow( &x1 , &y1 ) ; + if ( x ) *x = x1 ; + if ( y ) *y = y1 ; } void wxWindowMac::MacRootWindowToWindow( int *x , int *y ) const { + #if TARGET_API_MAC_OSX + HIPoint pt ; + if ( x ) pt.x = *x ; + if ( y ) pt.y = *y ; + + if ( !IsTopLevel() ) + HIViewConvertPoint( &pt , (ControlRef) MacGetTopLevelWindow()->GetHandle() , *m_peer ) ; + + if ( x ) *x = (int) pt.x ; + if ( y ) *y = (int) pt.y ; + #else if ( !IsTopLevel() ) { - if(x) *x -= m_x ; - if(y) *y -= m_y ; - GetParent()->MacRootWindowToWindow( x , y ) ; + Rect bounds ; + GetControlBounds( *m_peer , &bounds ) ; + if(x) *x -= bounds.left ; + if(y) *y -= bounds.top ; } +#endif } -bool wxWindowMac::SetCursor(const wxCursor& cursor) +void wxWindowMac::MacRootWindowToWindow( short *x , short *y ) const { - if (m_cursor == cursor) - return FALSE; + int x1 , y1 ; + if ( x ) x1 = *x ; + if ( y ) y1 = *y ; + MacRootWindowToWindow( &x1 , &y1 ) ; + if ( x ) *x = x1 ; + if ( y ) *y = y1 ; +} - if (wxNullCursor == cursor) +void wxWindowMac::MacGetContentAreaInset( int &left , int &top , int &right , int &bottom ) +{ + RgnHandle rgn = NewRgn() ; + Rect content ; + if ( GetControlRegion( *m_peer , kControlContentMetaPart , rgn ) == noErr ) { - if ( ! wxWindowBase::SetCursor( *wxSTANDARD_CURSOR ) ) - return FALSE ; + GetRegionBounds( rgn , &content ) ; + DisposeRgn( rgn ) ; } else { - if ( ! wxWindowBase::SetCursor( cursor ) ) - return FALSE ; + GetControlBounds( *m_peer , &content ) ; } + Rect structure ; + GetControlBounds( *m_peer , &structure ) ; +#if !TARGET_API_MAC_OSX + OffsetRect( &content , -structure.left , -structure.top ) ; +#endif + left = content.left - structure.left ; + top = content.top - structure.top ; + right = structure.right - content.right ; + bottom = structure.bottom - content.bottom ; +} - wxASSERT_MSG( m_cursor.Ok(), - wxT("cursor must be valid after call to the base version")); - - Point pt ; - wxWindowMac *mouseWin ; - GetMouse( &pt ) ; +wxSize wxWindowMac::DoGetSizeFromClientSize( const wxSize & size ) const +{ + wxSize sizeTotal = size; - // Change the cursor NOW if we're within the correct window + RgnHandle rgn = NewRgn() ; - if ( MacGetWindowFromPoint( wxPoint( pt.h , pt.v ) , &mouseWin ) ) - { - if ( mouseWin == this && !wxIsBusy() ) + Rect content ; + + if ( GetControlRegion( *m_peer , kControlContentMetaPart , rgn ) == noErr ) { - m_cursor.MacInstall() ; + GetRegionBounds( rgn , &content ) ; + DisposeRgn( rgn ) ; + } + else + { + GetControlBounds( *m_peer , &content ) ; } - } + Rect structure ; + GetControlBounds( *m_peer , &structure ) ; +#if !TARGET_API_MAC_OSX + OffsetRect( &content , -structure.left , -structure.top ) ; +#endif + + sizeTotal.x += (structure.right - structure.left) - (content.right - content.left) ; + sizeTotal.y += (structure.bottom - structure.top) - (content.bottom - content.top ) ; - return TRUE ; + sizeTotal.x += MacGetLeftBorderSize( ) + MacGetRightBorderSize( ) ; + sizeTotal.y += MacGetTopBorderSize( ) + MacGetBottomBorderSize( ) ; + + return sizeTotal; } @@ -481,58 +1327,160 @@ bool wxWindowMac::SetCursor(const wxCursor& cursor) void wxWindowMac::DoGetClientSize(int *x, int *y) const { int ww, hh; - ww = m_width ; - hh = m_height ; + RgnHandle rgn = NewRgn() ; + Rect content ; + if ( GetControlRegion( *m_peer , kControlContentMetaPart , rgn ) == noErr ) + { + GetRegionBounds( rgn , &content ) ; + DisposeRgn( rgn ) ; + } + else + { + GetControlBounds( *m_peer , &content ) ; + } +#if !TARGET_API_MAC_OSX + Rect structure ; + GetControlBounds( *m_peer , &structure ) ; + OffsetRect( &content , -structure.left , -structure.top ) ; +#endif + ww = content.right - content.left ; + hh = content.bottom - content.top ; + ww -= MacGetLeftBorderSize( ) + MacGetRightBorderSize( ) ; hh -= MacGetTopBorderSize( ) + MacGetBottomBorderSize( ); - if ( (m_vScrollBar && m_vScrollBar->IsShown()) || (m_hScrollBar && m_hScrollBar->IsShown()) ) - { - int x1 = 0 ; - int y1 = 0 ; - int w = m_width ; - int h = m_height ; + if ( (m_vScrollBar && m_vScrollBar->IsShown()) || (m_hScrollBar && m_hScrollBar->IsShown()) ) + { + int x1 = 0 ; + int y1 = 0 ; + int w ; + int h ; + GetSize( &w , &h ) ; + + MacClientToRootWindow( &x1 , &y1 ) ; + MacClientToRootWindow( &w , &h ) ; - MacClientToRootWindow( &x1 , &y1 ) ; - MacClientToRootWindow( &w , &h ) ; + wxWindowMac *iter = (wxWindowMac*)this ; - wxWindowMac *iter = (wxWindowMac*)this ; + int totW = 10000 , totH = 10000; + while( iter ) + { + if ( iter->IsTopLevel() ) + { + iter->GetSize( &totW , &totH ) ; + break ; + } - int totW = 10000 , totH = 10000; - while( iter ) - { - if ( iter->IsTopLevel() ) + iter = iter->GetParent() ; + } + + if (m_hScrollBar && m_hScrollBar->IsShown() ) { - totW = iter->m_width ; - totH = iter->m_height ; - break ; + hh -= MAC_SCROLLBAR_SIZE; + if ( h-y1 >= totH ) + { + hh += 1 ; + } + } + if (m_vScrollBar && m_vScrollBar->IsShown() ) + { + ww -= MAC_SCROLLBAR_SIZE; + if ( w-x1 >= totW ) + { + ww += 1 ; + } } + } + if(x) *x = ww; + if(y) *y = hh; - iter = iter->GetParent() ; +} + +bool wxWindowMac::SetCursor(const wxCursor& cursor) +{ + if (m_cursor == cursor) + return FALSE; + + if (wxNullCursor == cursor) + { + if ( ! wxWindowBase::SetCursor( *wxSTANDARD_CURSOR ) ) + return FALSE ; + } + else + { + if ( ! wxWindowBase::SetCursor( cursor ) ) + return FALSE ; } - if (m_hScrollBar && m_hScrollBar->IsShown() ) + wxASSERT_MSG( m_cursor.Ok(), + wxT("cursor must be valid after call to the base version")); + + /* + + TODO why do we have to use current coordinates ? + + Point pt ; + wxWindowMac *mouseWin ; + GetMouse( &pt ) ; + + // Change the cursor NOW if we're within the correct window + + + if ( MacGetWindowFromPoint( wxPoint( pt.h , pt.v ) , &mouseWin ) ) { - hh -= MAC_SCROLLBAR_SIZE; - if ( h-y1 >= totH ) + if ( mouseWin == this && !wxIsBusy() ) { - hh += 1 ; + m_cursor.MacInstall() ; } } - if (m_vScrollBar && m_vScrollBar->IsShown() ) + */ + if ( !wxIsBusy() ) + { + m_cursor.MacInstall() ; + } + + return TRUE ; +} + +#if wxUSE_MENUS +bool wxWindowMac::DoPopupMenu(wxMenu *menu, int x, int y) +{ + menu->SetInvokingWindow(this); + menu->UpdateUI(); + + if ( x == -1 && y == -1 ) + { + wxPoint mouse = wxGetMousePosition(); + x = mouse.x; y = mouse.y; + } + else { - ww -= MAC_SCROLLBAR_SIZE; - if ( w-x1 >= totW ) + ClientToScreen( &x , &y ) ; + } + + menu->MacBeforeDisplay( true ) ; + long menuResult = ::PopUpMenuSelect((MenuHandle) menu->GetHMenu() ,y,x, 0) ; + if ( HiWord(menuResult) != 0 ) + { + MenuCommand id ; + GetMenuItemCommandID( GetMenuHandle(HiWord(menuResult)) , LoWord(menuResult) , &id ) ; + wxMenuItem* item = NULL ; + wxMenu* realmenu ; + item = menu->FindItem(id, &realmenu) ; + if (item->IsCheckable()) { - ww += 1 ; + item->Check( !item->IsChecked() ) ; } + menu->SendEvent( id , item->IsCheckable() ? item->IsChecked() : -1 ) ; } - } - if(x) *x = ww; - if(y) *y = hh; -} + menu->MacAfterDisplay( true ) ; + + menu->SetInvokingWindow(NULL); + return TRUE; +} +#endif // ---------------------------------------------------------------------------- // tooltips @@ -552,15 +1500,18 @@ void wxWindowMac::DoSetToolTip(wxToolTip *tooltip) void wxWindowMac::DoMoveWindow(int x, int y, int width, int height) { - int former_x = m_x ; - int former_y = m_y ; - int former_w = m_width ; - int former_h = m_height ; + int former_x , former_y , former_w, former_h ; +#if !TARGET_API_MAC_OSX + DoGetPosition( &former_x , &former_y ) ; + DoGetSize( &former_w , &former_h ) ; +#else + MacGetPositionAndSizeFromControl( former_x , former_y , former_w , former_h ) ; +#endif - int actualWidth = width; - int actualHeight = height; - int actualX = x; - int actualY = y; + int actualWidth = width; + int actualHeight = height; + int actualX = x; + int actualY = y; if ((m_minWidth != -1) && (actualWidth < m_minWidth)) actualWidth = m_minWidth; @@ -585,41 +1536,84 @@ void wxWindowMac::DoMoveWindow(int x, int y, int width, int height) if ( doMove || doResize ) { - // erase former position - - Refresh() ; - - m_x = actualX ; - m_y = actualY ; - m_width = actualWidth ; - m_height = actualHeight ; - - // erase new position - - Refresh() ; - if ( doMove ) - wxWindowMac::MacSuperChangedPosition() ; // like this only children will be notified - + // we don't adjust twice for the origin + Rect r = wxMacGetBoundsForControl(this , wxPoint( actualX,actualY), wxSize( actualWidth, actualHeight ) , false ) ; + bool vis = IsControlVisible( *m_peer ) ; +#if TARGET_API_MAC_OSX + // the HIViewSetFrame call itself should invalidate the areas, but when testing with the UnicodeTextCtrl it does not ! + if ( vis ) + SetControlVisibility( *m_peer , false , true ) ; + HIRect hir = { r.left , r.top , r.right - r.left , r.bottom - r.top } ; + HIViewSetFrame ( *m_peer , &hir ) ; + if ( vis ) + SetControlVisibility( *m_peer , true , true ) ; +#else + if ( vis ) + SetControlVisibility( *m_peer , false , true ) ; + SetControlBounds( *m_peer , &r ) ; + if ( vis ) + SetControlVisibility( *m_peer , true , true ) ; +#endif MacRepositionScrollBars() ; if ( doMove ) { - wxPoint point(m_x, m_y); + wxPoint point(actualX,actualY); wxMoveEvent event(point, m_windowId); event.SetEventObject(this); GetEventHandler()->ProcessEvent(event) ; } if ( doResize ) { - MacRepositionScrollBars() ; - wxSize size(m_width, m_height); - wxSizeEvent event(size, m_windowId); - event.SetEventObject(this); - GetEventHandler()->ProcessEvent(event); + MacRepositionScrollBars() ; + wxSize size(actualWidth, actualHeight); + wxSizeEvent event(size, m_windowId); + event.SetEventObject(this); + GetEventHandler()->ProcessEvent(event); + } + } + +} + +wxSize wxWindowMac::DoGetBestSize() const +{ + if ( m_macIsUserPane || IsTopLevel() ) + return wxWindowBase::DoGetBestSize() ; + + Rect bestsize = { 0 , 0 , 0 , 0 } ; + short baselineoffset ; + int bestWidth, bestHeight ; + ::GetBestControlRect( *m_peer , &bestsize , &baselineoffset ) ; + + if ( EmptyRect( &bestsize ) ) + { + baselineoffset = 0; + bestsize.left = bestsize.top = 0 ; + bestsize.right = 16 ; + bestsize.bottom = 16 ; + if ( IsKindOf( CLASSINFO( wxScrollBar ) ) ) + { + bestsize.bottom = 16 ; + } + else if ( IsKindOf( CLASSINFO( wxSpinButton ) ) ) + { + bestsize.bottom = 24 ; + } + else + { + // return wxWindowBase::DoGetBestSize() ; } } + bestWidth = bestsize.right - bestsize.left ; + bestHeight = bestsize.bottom - bestsize.top ; + if ( bestHeight < 10 ) + bestHeight = 13 ; + + return wxSize(bestWidth, bestHeight); +// return wxWindowBase::DoGetBestSize() ; } + // set the size of the window: if the dimensions are positive, just use them, // but if any of them is equal to -1, it means that we must find the value for // it ourselves (unless sizeFlags contains wxSIZE_ALLOW_MINUS_ONE flag, in @@ -639,8 +1633,9 @@ void wxWindowMac::DoSetSize(int x, int y, int width, int height, int sizeFlags) // ... and don't do anything (avoiding flicker) if it's already ok if ( x == currentX && y == currentY && - width == currentW && height == currentH ) + width == currentW && height == currentH && ( height != -1 && width != -1 ) ) { + // TODO REMOVE MacRepositionScrollBars() ; // we might have a real position shift return; } @@ -689,17 +1684,52 @@ void wxWindowMac::DoSetSize(int x, int y, int width, int height, int sizeFlags) DoMoveWindow(x, y, width, height); } -// For implementation purposes - sometimes decorations make the client area -// smaller wxPoint wxWindowMac::GetClientAreaOrigin() const { - return wxPoint(MacGetLeftBorderSize( ) , MacGetTopBorderSize( ) ); + RgnHandle rgn = NewRgn() ; + Rect content ; + GetControlRegion( *m_peer , kControlContentMetaPart , rgn ) ; + GetRegionBounds( rgn , &content ) ; + DisposeRgn( rgn ) ; +#if !TARGET_API_MAC_OSX + // if the content rgn is empty / not supported + // don't attempt to correct the coordinates to wxWindow relative ones + if (!::EmptyRect( &content ) ) + { + Rect structure ; + GetControlBounds( *m_peer , &structure ) ; + OffsetRect( &content , -structure.left , -structure.top ) ; + } +#endif + + return wxPoint( content.left + MacGetLeftBorderSize( ) , content.top + MacGetTopBorderSize( ) ); +} + +void wxWindowMac::DoSetClientSize(int clientwidth, int clientheight) +{ + if ( clientheight != -1 || clientheight != -1 ) + { + int currentclientwidth , currentclientheight ; + int currentwidth , currentheight ; + + GetClientSize( ¤tclientwidth , ¤tclientheight ) ; + GetSize( ¤twidth , ¤theight ) ; + + DoSetSize( -1 , -1 , currentwidth + clientwidth - currentclientwidth , + currentheight + clientheight - currentclientheight , wxSIZE_USE_EXISTING ) ; + } } void wxWindowMac::SetTitle(const wxString& title) { - m_label = title ; + m_label = wxStripMenuCodes(title) ; + + if ( m_peer && m_peer->Ok() ) + { + UMASetControlTitle( *m_peer , m_label , m_font.GetEncoding() ) ; + } + Refresh() ; } wxString wxWindowMac::GetTitle() const @@ -711,77 +1741,158 @@ bool wxWindowMac::Show(bool show) { if ( !wxWindowBase::Show(show) ) return FALSE; + + // TODO use visibilityChanged Carbon Event for OSX + bool former = MacIsReallyShown() ; + + SetControlVisibility( *m_peer , show , true ) ; + if ( former != MacIsReallyShown() ) + MacPropagateVisibilityChanged() ; + return TRUE; +} -/* - WindowRef window = (WindowRef) MacGetRootWindow() ; - wxWindowMac* win = wxFindWinFromMacWindow( window ) ; - if ( win == NULL && win->m_isBeingDeleted ) - return FALSE ; -*/ - MacSuperShown( show ) ; - Refresh() ; -/* - if ( !show ) - { - if ( win && !win->m_isBeingDeleted ) - Refresh() ; - } +bool wxWindowMac::Enable(bool enable) +{ + wxASSERT( m_peer->Ok() ) ; + if ( !wxWindowBase::Enable(enable) ) + return FALSE; + + bool former = MacIsReallyEnabled() ; +#if TARGET_API_MAC_OSX + if ( enable ) + EnableControl( *m_peer ) ; else - { - Refresh() ; - } -*/ + DisableControl( *m_peer ) ; +#else + if ( enable ) + ActivateControl( *m_peer ) ; + else + DeactivateControl( *m_peer ) ; +#endif + + if ( former != MacIsReallyEnabled() ) + MacPropagateEnabledStateChanged() ; return TRUE; } -void wxWindowMac::MacSuperShown( bool show ) +// +// status change propagations (will be not necessary for OSX later ) +// + +void wxWindowMac::MacPropagateVisibilityChanged() { - wxNode *node = GetChildren().First(); +#if !TARGET_API_MAC_OSX + MacVisibilityChanged() ; + + wxWindowListNode *node = GetChildren().GetFirst(); while ( node ) { - wxWindowMac *child = (wxWindowMac *)node->Data(); - if ( child->m_isShown ) - child->MacSuperShown( show ) ; - node = node->Next(); + wxWindowMac *child = node->GetData(); + if ( child->IsShown() ) + child->MacPropagateVisibilityChanged( ) ; + node = node->GetNext(); } +#endif } -void wxWindowMac::MacSuperEnabled( bool enabled ) +void wxWindowMac::MacPropagateEnabledStateChanged( ) { - if ( !IsTopLevel() ) - { - // to be absolutely correct we'd have to invalidate (with eraseBkground - // because unter MacOSX the frames are drawn with an addXXX mode) - // the borders area - } - wxNode *node = GetChildren().First(); +#if !TARGET_API_MAC_OSX + MacEnabledStateChanged() ; + + wxWindowListNode *node = GetChildren().GetFirst(); while ( node ) { - wxWindowMac *child = (wxWindowMac *)node->Data(); - if ( child->m_isShown ) - child->MacSuperEnabled( enabled ) ; - node = node->Next(); + wxWindowMac *child = node->GetData(); + if ( child->IsEnabled() ) + child->MacPropagateEnabledStateChanged() ; + node = node->GetNext(); } +#endif } -bool wxWindowMac::MacIsReallyShown() const +void wxWindowMac::MacPropagateHiliteChanged( ) { - if ( m_isShown && (m_parent != NULL) ) { - return m_parent->MacIsReallyShown(); - } - return m_isShown; -/* - bool status = m_isShown ; - wxWindowMac * win = this ; - while ( status && win->m_parent != NULL ) +#if !TARGET_API_MAC_OSX + MacHiliteChanged() ; + + wxWindowListNode *node = GetChildren().GetFirst(); + while ( node ) { - win = win->m_parent ; - status = win->m_isShown ; + wxWindowMac *child = node->GetData(); + // if ( child->IsEnabled() ) + child->MacPropagateHiliteChanged() ; + node = node->GetNext(); } - return status ; -*/ +#endif +} + +// +// status change notifications +// + +void wxWindowMac::MacVisibilityChanged() +{ +} + +void wxWindowMac::MacHiliteChanged() +{ +} + +void wxWindowMac::MacEnabledStateChanged() +{ +} + +// +// status queries on the inherited window's state +// + +bool wxWindowMac::MacIsReallyShown() +{ + // only under OSX the visibility of the TLW is taken into account +#if TARGET_API_MAC_OSX + return IsControlVisible( *m_peer ) ; +#else + wxWindow* win = this ; + while( win->IsShown() ) + { + if ( win->IsTopLevel() ) + return true ; + + win = win->GetParent() ; + if ( win == NULL ) + return true ; + + } ; + return false ; +#endif +} + +bool wxWindowMac::MacIsReallyEnabled() +{ +#if TARGET_API_MAC_OSX + return IsControlEnabled( *m_peer ) ; +#else + return IsControlActive( *m_peer ) ; +#endif +} + +bool wxWindowMac::MacIsReallyHilited() +{ + return IsControlActive( *m_peer ) ; +} + +void wxWindowMac::MacFlashInvalidAreas() +{ +#if TARGET_API_MAC_OSX + HIViewFlashDirtyArea( (WindowRef) MacGetTopLevelWindowRef() ) ; +#endif } +// +// +// + int wxWindowMac::GetCharHeight() const { wxClientDC dc ( (wxWindowMac*)this ) ; @@ -821,10 +1932,46 @@ void wxWindowMac::GetTextExtent(const wxString& string, int *x, int *y, void wxWindowMac::Refresh(bool eraseBack, const wxRect *rect) { +#if TARGET_API_MAC_OSX + if ( rect == NULL ) + HIViewSetNeedsDisplay( *m_peer , true ) ; + else + { + RgnHandle update = NewRgn() ; + SetRectRgn( update , rect->x , rect->y , rect->x + rect->width , rect->y + rect->height ) ; + SectRgn( (RgnHandle) MacGetVisibleRegion().GetWXHRGN() , update , update ) ; + wxPoint origin = GetClientAreaOrigin() ; + OffsetRgn( update, origin.x , origin.y ) ; + HIViewSetNeedsDisplayInRegion( *m_peer , update , true ) ; + } +#else +/* + RgnHandle updateRgn = NewRgn() ; + if ( rect == NULL ) + { + CopyRgn( (RgnHandle) MacGetVisibleRegion().GetWXHRGN() , updateRgn ) ; + } + else + { + SetRectRgn( updateRgn , rect->x , rect->y , rect->x + rect->width , rect->y + rect->height ) ; + SectRgn( (RgnHandle) MacGetVisibleRegion().GetWXHRGN() , updateRgn , updateRgn ) ; + } + InvalWindowRgn( (WindowRef) MacGetTopLevelWindowRef() , updateRgn ) ; + DisposeRgn(updateRgn) ; +*/ + if ( IsControlVisible( *m_peer ) ) + { + SetControlVisibility( *m_peer , false , false ) ; + SetControlVisibility( *m_peer , true , true ) ; + } + /* if ( MacGetTopLevelWindow() == NULL ) return ; - wxPoint client = GetClientAreaOrigin(); + if ( !IsControlVisible( *m_peer ) ) + return ; + + wxPoint client = GetClientAreaOrigin(); int x1 = -client.x; int y1 = -client.y; int x2 = m_width - client.x; @@ -856,49 +2003,79 @@ void wxWindowMac::Refresh(bool eraseBack, const wxRect *rect) MacGetTopLevelWindow()->MacInvalidate( &clientrect , eraseBack ) ; } + */ +#endif } -#if wxUSE_CARET && WXWIN_COMPATIBILITY -// --------------------------------------------------------------------------- -// Caret manipulation -// --------------------------------------------------------------------------- - -void wxWindowMac::CreateCaret(int w, int h) -{ - SetCaret(new wxCaret(this, w, h)); -} - -void wxWindowMac::CreateCaret(const wxBitmap *WXUNUSED(bitmap)) +void wxWindowMac::Freeze() { - wxFAIL_MSG("not implemented"); +#if TARGET_API_MAC_OSX + if ( !m_frozenness++ ) + { + HIViewSetDrawingEnabled( *m_peer , false ) ; + } +#endif } -void wxWindowMac::ShowCaret(bool show) +#if TARGET_API_MAC_OSX +static void InvalidateControlAndChildren( HIViewRef control ) { - wxCHECK_RET( m_caret, "no caret to show" ); + HIViewSetNeedsDisplay( control , true ) ; + UInt16 childrenCount = 0 ; + OSStatus err = CountSubControls( control , &childrenCount ) ; + if ( err == errControlIsNotEmbedder ) + return ; + wxASSERT_MSG( err == noErr , wxT("Unexpected error when accessing subcontrols") ) ; - m_caret->Show(show); + for ( UInt16 i = childrenCount ; i >=1 ; --i ) + { + HIViewRef child ; + err = GetIndexedSubControl( control , i , & child ) ; + if ( err == errControlIsNotEmbedder ) + return ; + InvalidateControlAndChildren( child ) ; + } } +#endif -void wxWindowMac::DestroyCaret() +void wxWindowMac::Thaw() { - SetCaret(NULL); +#if TARGET_API_MAC_OSX + wxASSERT_MSG( m_frozenness > 0, _T("Thaw() without matching Freeze()") ); + + if ( !--m_frozenness ) + { + HIViewSetDrawingEnabled( *m_peer , true ) ; + InvalidateControlAndChildren( *m_peer ) ; + // HIViewSetNeedsDisplay( *m_peer , true ) ; + } +#endif } -void wxWindowMac::SetCaretPos(int x, int y) +void wxWindowMac::MacRedrawControl() { - wxCHECK_RET( m_caret, "no caret to move" ); - - m_caret->Move(x, y); +/* + if ( *m_peer && MacGetTopLevelWindowRef() && IsControlVisible( *m_peer ) ) + { +#if TARGET_API_MAC_CARBON + Update() ; +#else + wxClientDC dc(this) ; + wxMacPortSetter helper(&dc) ; + wxMacWindowClipper clipper(this) ; + wxDC::MacSetupBackgroundForCurrentPort( MacGetBackgroundBrush() ) ; + UMADrawControl( *m_peer ) ; +#endif + } +*/ } - -void wxWindowMac::GetCaretPos(int *x, int *y) const + +/* TODO +void wxWindowMac::OnPaint(wxPaintEvent& event) { - wxCHECK_RET( m_caret, "no caret to get position of" ); - - m_caret->GetPosition(x, y); + // why don't we skip that here ? } -#endif // wxUSE_CARET +*/ wxWindowMac *wxGetActiveWindow() { @@ -909,77 +2086,17 @@ wxWindowMac *wxGetActiveWindow() // Coordinates relative to the window void wxWindowMac::WarpPointer (int x_pos, int y_pos) { - // We really dont move the mouse programmatically under mac + // We really don't move the mouse programmatically under Mac. } -const wxBrush& wxWindowMac::MacGetBackgroundBrush() +void wxWindowMac::OnEraseBackground(wxEraseEvent& event) { - if ( m_backgroundColour == wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE) ) - { - m_macBackgroundBrush.SetMacTheme( kThemeBrushDocumentWindowBackground ) ; - } - else if ( m_backgroundColour == wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE ) ) + if ( m_macBackgroundBrush.Ok() == false || m_macBackgroundBrush.GetStyle() == wxTRANSPARENT ) { - // on mac we have the difficult situation, that 3dface gray can be different colours, depending whether - // it is on a notebook panel or not, in order to take care of that we walk up the hierarchy until we have - // either a non gray background color or a non control window - - WindowRef window = (WindowRef) MacGetRootWindow() ; - - wxWindowMac* parent = GetParent() ; - while( parent ) - { - if ( parent->MacGetRootWindow() != window ) - { - // we are in a different window on the mac system - parent = NULL ; - break ; - } - - { - if ( parent->m_backgroundColour != wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE ) - && parent->m_backgroundColour != wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE) ) - { - // if we have any other colours in the hierarchy - m_macBackgroundBrush.SetColour( parent->m_backgroundColour ) ; - break ; - } - // if we have the normal colours in the hierarchy but another control etc. -> use it's background - if ( parent->IsKindOf( CLASSINFO( wxNotebook ) ) || parent->IsKindOf( CLASSINFO( wxTabCtrl ) )) - { - Rect extent = { 0 , 0 , 0 , 0 } ; - int x , y ; - x = y = 0 ; - wxSize size = parent->GetSize() ; - parent->MacClientToRootWindow( &x , &y ) ; - extent.left = x ; - extent.top = y ; - extent.top-- ; - extent.right = x + size.x ; - extent.bottom = y + size.y ; - m_macBackgroundBrush.SetMacThemeBackground( kThemeBackgroundTabPane , (WXRECTPTR) &extent ) ; // todo eventually change for inactive - break ; - } - } - parent = parent->GetParent() ; - } - if ( !parent ) - { - m_macBackgroundBrush.SetMacTheme( kThemeBrushDialogBackgroundActive ) ; // todo eventually change for inactive - } + event.Skip() ; } else - { - m_macBackgroundBrush.SetColour( m_backgroundColour ) ; - } - - return m_macBackgroundBrush ; - -} - -void wxWindowMac::OnEraseBackground(wxEraseEvent& event) -{ - event.GetDC()->Clear() ; + event.GetDC()->Clear() ; } void wxWindowMac::OnNcPaint( wxNcPaintEvent& event ) @@ -1055,61 +2172,78 @@ void wxWindowMac::MacPaintBorders( int left , int top ) { if( IsTopLevel() ) return ; + + int major,minor; + wxGetOsVersion( &major, &minor ); RGBColor white = { 0xFFFF, 0xFFFF , 0xFFFF } ; - RGBColor black = { 0x0000, 0x0000 , 0x0000 } ; RGBColor face = { 0xDDDD, 0xDDDD , 0xDDDD } ; - RGBColor shadow = { 0x4444, 0x4444 , 0x4444 } ; + + RGBColor darkShadow = { 0x0000, 0x0000 , 0x0000 } ; + RGBColor lightShadow = { 0x4444, 0x4444 , 0x4444 } ; + // OS X has lighter border edges than classic: + if (major >= 10) + { + darkShadow.red = 0x8E8E; + darkShadow.green = 0x8E8E; + darkShadow.blue = 0x8E8E; + lightShadow.red = 0xBDBD; + lightShadow.green = 0xBDBD; + lightShadow.blue = 0xBDBD; + } + PenNormal() ; + int w , h ; + GetSize( &w , &h ) ; if (HasFlag(wxRAISED_BORDER) || HasFlag( wxSUNKEN_BORDER) || HasFlag(wxDOUBLE_BORDER) ) { #if wxMAC_USE_THEME_BORDER - Rect rect = { top , left , m_height + top , m_width + left } ; - SInt32 border = 0 ; - /* - GetThemeMetric( kThemeMetricListBoxFrameOutset , &border ) ; - InsetRect( &rect , border , border ); - DrawThemeListBoxFrame(&rect,IsEnabled() ? kThemeStateActive : kThemeStateInactive) ; - */ + Rect rect = { top , left , m_height + top , m_width + left } ; + SInt32 border = 0 ; + /* + GetThemeMetric( kThemeMetricListBoxFrameOutset , &border ) ; + InsetRect( &rect , border , border ); + DrawThemeListBoxFrame(&rect,IsEnabled() ? kThemeStateActive : kThemeStateInactive) ; + */ DrawThemePrimaryGroup(&rect ,IsEnabled() ? kThemeStateActive : kThemeStateInactive) ; #else bool sunken = HasFlag( wxSUNKEN_BORDER ) ; RGBForeColor( &face ); - MoveTo( left + 0 , top + m_height - 2 ); + MoveTo( left + 0 , top + h - 2 ); LineTo( left + 0 , top + 0 ); - LineTo( left + m_width - 2 , top + 0 ); + LineTo( left + w - 2 , top + 0 ); - MoveTo( left + 2 , top + m_height - 3 ); - LineTo( left + m_width - 3 , top + m_height - 3 ); - LineTo( left + m_width - 3 , top + 2 ); + MoveTo( left + 2 , top + h - 3 ); + LineTo( left + w - 3 , top + h - 3 ); + LineTo( left + w - 3 , top + 2 ); - RGBForeColor( sunken ? &face : &black ); - MoveTo( left + 0 , top + m_height - 1 ); - LineTo( left + m_width - 1 , top + m_height - 1 ); - LineTo( left + m_width - 1 , top + 0 ); + RGBForeColor( sunken ? &face : &darkShadow ); + MoveTo( left + 0 , top + h - 1 ); + LineTo( left + w - 1 , top + h - 1 ); + LineTo( left + w - 1 , top + 0 ); - RGBForeColor( sunken ? &shadow : &white ); - MoveTo( left + 1 , top + m_height - 3 ); + RGBForeColor( sunken ? &lightShadow : &white ); + MoveTo( left + 1 , top + h - 3 ); LineTo( left + 1, top + 1 ); - LineTo( left + m_width - 3 , top + 1 ); + LineTo( left + w - 3 , top + 1 ); - RGBForeColor( sunken ? &white : &shadow ); - MoveTo( left + 1 , top + m_height - 2 ); - LineTo( left + m_width - 2 , top + m_height - 2 ); - LineTo( left + m_width - 2 , top + 1 ); + RGBForeColor( sunken ? &white : &lightShadow ); + MoveTo( left + 1 , top + h - 2 ); + LineTo( left + w - 2 , top + h - 2 ); + LineTo( left + w - 2 , top + 1 ); - RGBForeColor( sunken ? &black : &face ); - MoveTo( left + 2 , top + m_height - 4 ); + RGBForeColor( sunken ? &darkShadow : &face ); + MoveTo( left + 2 , top + h - 4 ); LineTo( left + 2 , top + 2 ); - LineTo( left + m_width - 4 , top + 2 ); + LineTo( left + w - 4 , top + 2 ); #endif } else if (HasFlag(wxSIMPLE_BORDER)) { - Rect rect = { top , left , m_height + top , m_width + left } ; - RGBForeColor( &black ) ; + Rect rect = { top , left , h + top , w + left } ; + RGBForeColor( &darkShadow ) ; FrameRect( &rect ) ; } } @@ -1128,80 +2262,132 @@ void wxWindowMac::RemoveChild( wxWindowBase *child ) void wxWindowMac::SetScrollbar(int orient, int pos, int thumbVisible, int range, bool refresh) { - if ( orient == wxHORIZONTAL ) + if ( orient == wxHORIZONTAL ) + { + if ( m_hScrollBar ) { - if ( m_hScrollBar ) + if ( range == 0 || thumbVisible >= range ) { - if ( range == 0 || thumbVisible >= range ) - { - if ( m_hScrollBar->IsShown() ) - m_hScrollBar->Show(false) ; - } - else - { - if ( !m_hScrollBar->IsShown() ) - m_hScrollBar->Show(true) ; - m_hScrollBar->SetScrollbar( pos , thumbVisible , range , thumbVisible , refresh ) ; - } + if ( m_hScrollBar->IsShown() ) + m_hScrollBar->Show(false) ; + } + else + { + if ( !m_hScrollBar->IsShown() ) + m_hScrollBar->Show(true) ; + m_hScrollBar->SetScrollbar( pos , thumbVisible , range , thumbVisible , refresh ) ; } } - else + } + else + { + if ( m_vScrollBar ) { - if ( m_vScrollBar ) + if ( range == 0 || thumbVisible >= range ) { - if ( range == 0 || thumbVisible >= range ) - { - if ( m_vScrollBar->IsShown() ) - m_vScrollBar->Show(false) ; - } - else - { - if ( !m_vScrollBar->IsShown() ) - m_vScrollBar->Show(true) ; - m_vScrollBar->SetScrollbar( pos , thumbVisible , range , thumbVisible , refresh ) ; - } + if ( m_vScrollBar->IsShown() ) + m_vScrollBar->Show(false) ; + } + else + { + if ( !m_vScrollBar->IsShown() ) + m_vScrollBar->Show(true) ; + m_vScrollBar->SetScrollbar( pos , thumbVisible , range , thumbVisible , refresh ) ; } } - MacRepositionScrollBars() ; + } + MacRepositionScrollBars() ; } // Does a physical scroll void wxWindowMac::ScrollWindow(int dx, int dy, const wxRect *rect) { - wxClientDC dc(this) ; - wxMacPortSetter helper(&dc) ; + if( dx == 0 && dy ==0 ) + return ; + { + int width , height ; GetClientSize( &width , &height ) ; +#if TARGET_API_MAC_OSX + // note there currently is a bug in OSX which makes inefficient refreshes in case an entire control + // area is scrolled, this does not occur if width and height are 2 pixels less, + // TODO write optimal workaround + HIRect scrollrect = CGRectMake( MacGetLeftBorderSize() , MacGetTopBorderSize() , width , height ) ; + if ( rect ) + { + HIRect scrollarea = CGRectMake( rect->x , rect->y , rect->width , rect->height) ; + scrollrect = CGRectIntersection( scrollrect , scrollarea ) ; + } + if ( HIViewGetNeedsDisplay( *m_peer ) ) + { + // becuase HIViewScrollRect does not scroll the already invalidated area we have two options + // either immediate redraw or full invalidate +#if 1 + // is the better overall solution, as it does not slow down scrolling + HIViewSetNeedsDisplay( *m_peer , true ) ; +#else + // this would be the preferred version for fast drawing controls + if( UMAGetSystemVersion() < 0x1030 ) + Update() ; + else + HIViewRender(*m_peer) ; +#endif + } + HIViewScrollRect ( *m_peer , &scrollrect , dx ,dy ) ; +#else - Rect scrollrect = { dc.YLOG2DEVMAC(0) , dc.XLOG2DEVMAC(0) , dc.YLOG2DEVMAC(height) , dc.XLOG2DEVMAC(width) } ; + wxPoint pos; + pos.x = pos.y = 0; + + Rect scrollrect; RgnHandle updateRgn = NewRgn() ; - ClipRect( &scrollrect ) ; - if ( rect ) - { - Rect r = { dc.YLOG2DEVMAC(rect->y) , dc.XLOG2DEVMAC(rect->x) , dc.YLOG2DEVMAC(rect->y + rect->height) , - dc.XLOG2DEVMAC(rect->x + rect->width) } ; - SectRect( &scrollrect , &r , &scrollrect ) ; + + { + wxClientDC dc(this) ; + wxMacPortSetter helper(&dc) ; + + GetControlBounds( *m_peer, &scrollrect); + scrollrect.top += MacGetTopBorderSize() ; + scrollrect.left += MacGetLeftBorderSize() ; + scrollrect.bottom = scrollrect.top + height ; + scrollrect.right = scrollrect.left + width ; + + if ( rect ) + { + Rect r = { dc.YLOG2DEVMAC(rect->y) , dc.XLOG2DEVMAC(rect->x) , dc.YLOG2DEVMAC(rect->y + rect->height) , + dc.XLOG2DEVMAC(rect->x + rect->width) } ; + SectRect( &scrollrect , &r , &scrollrect ) ; + } + ScrollRect( &scrollrect , dx , dy , updateRgn ) ; } - ScrollRect( &scrollrect , dx , dy , updateRgn ) ; - InvalWindowRgn( (WindowRef) MacGetRootWindow() , updateRgn ) ; - DisposeRgn( updateRgn ) ; + // ScrollWindowRect( (WindowRef) MacGetTopLevelWindowRef() , &scrollrect , dx , dy , kScrollWindowInvalidate, updateRgn ) ; +#endif } - for (wxNode *node = GetChildren().First(); node; node = node->Next()) + for (wxWindowListNode *node = GetChildren().GetFirst(); node; node = node->GetNext()) { - wxWindowMac *child = (wxWindowMac*)node->Data(); + wxWindowMac *child = node->GetData(); if (child == m_vScrollBar) continue; if (child == m_hScrollBar) continue; if (child->IsTopLevel()) continue; + int x,y; child->GetPosition( &x, &y ); int w,h; child->GetSize( &w, &h ); - child->SetSize( x+dx, y+dy, w, h ); + if (rect) + { + wxRect rc(x,y,w,h); + if (rect->Intersects(rc)) + child->SetSize( x+dx, y+dy, w, h ); + } + else + { + child->SetSize( x+dx, y+dy, w, h ); + } } - } void wxWindowMac::MacOnScroll(wxScrollEvent &event ) @@ -1213,27 +2399,22 @@ void wxWindowMac::MacOnScroll(wxScrollEvent &event ) wevent.SetOrientation(event.GetOrientation()); wevent.m_eventObject = this; - if (event.m_eventType == wxEVT_SCROLL_TOP) { + if (event.m_eventType == wxEVT_SCROLL_TOP) wevent.m_eventType = wxEVT_SCROLLWIN_TOP; - } else - if (event.m_eventType == wxEVT_SCROLL_BOTTOM) { + else if (event.m_eventType == wxEVT_SCROLL_BOTTOM) wevent.m_eventType = wxEVT_SCROLLWIN_BOTTOM; - } else - if (event.m_eventType == wxEVT_SCROLL_LINEUP) { + else if (event.m_eventType == wxEVT_SCROLL_LINEUP) wevent.m_eventType = wxEVT_SCROLLWIN_LINEUP; - } else - if (event.m_eventType == wxEVT_SCROLL_LINEDOWN) { + else if (event.m_eventType == wxEVT_SCROLL_LINEDOWN) wevent.m_eventType = wxEVT_SCROLLWIN_LINEDOWN; - } else - if (event.m_eventType == wxEVT_SCROLL_PAGEUP) { + else if (event.m_eventType == wxEVT_SCROLL_PAGEUP) wevent.m_eventType = wxEVT_SCROLLWIN_PAGEUP; - } else - if (event.m_eventType == wxEVT_SCROLL_PAGEDOWN) { + else if (event.m_eventType == wxEVT_SCROLL_PAGEDOWN) wevent.m_eventType = wxEVT_SCROLLWIN_PAGEDOWN; - } else - if (event.m_eventType == wxEVT_SCROLL_THUMBTRACK) { + else if (event.m_eventType == wxEVT_SCROLL_THUMBTRACK) wevent.m_eventType = wxEVT_SCROLLWIN_THUMBTRACK; - } + else if (event.m_eventType == wxEVT_SCROLL_THUMBRELEASE) + wevent.m_eventType = wxEVT_SCROLLWIN_THUMBRELEASE; GetEventHandler()->ProcessEvent(wevent); } @@ -1242,39 +2423,10 @@ void wxWindowMac::MacOnScroll(wxScrollEvent &event ) // Get the window with the focus wxWindowMac *wxWindowBase::FindFocus() { - return gFocusWindow ; -} - -#if WXWIN_COMPATIBILITY -// If nothing defined for this, try the parent. -// E.g. we may be a button loaded from a resource, with no callback function -// defined. -void wxWindowMac::OnCommand(wxWindowMac& win, wxCommandEvent& event) -{ - if ( GetEventHandler()->ProcessEvent(event) ) - return; - if ( m_parent ) - m_parent->GetEventHandler()->OnCommand(win, event); -} -#endif // WXWIN_COMPATIBILITY_2 - -#if WXWIN_COMPATIBILITY -wxObject* wxWindowMac::GetChild(int number) const -{ - // Return a pointer to the Nth object in the Panel - wxNode *node = GetChildren().First(); - int n = number; - while (node && n--) - node = node->Next(); - if ( node ) - { - wxObject *obj = (wxObject *)node->Data(); - return(obj); - } - else - return NULL; + ControlRef control ; + GetKeyboardFocus( GetUserFocusWindow() , &control ) ; + return wxFindControlFromMacControl( control ) ; } -#endif // WXWIN_COMPATIBILITY void wxWindowMac::OnSetFocus(wxFocusEvent& event) { @@ -1292,215 +2444,73 @@ void wxWindowMac::OnSetFocus(wxFocusEvent& event) event.Skip(); } -void wxWindowMac::Clear() -{ - wxClientDC dc(this); - wxBrush brush(GetBackgroundColour(), wxSOLID); - dc.SetBackground(brush); - dc.Clear(); -} - -// Setup background and foreground colours correctly -void wxWindowMac::SetupColours() -{ - if ( GetParent() ) - SetBackgroundColour(GetParent()->GetBackgroundColour()); -} - -void wxWindowMac::OnIdle(wxIdleEvent& event) +void wxWindowMac::OnInternalIdle() { -/* - // Check if we need to send a LEAVE event - if (m_mouseInWindow) - { - POINT pt; - ::GetCursorPos(&pt); - if (::WindowFromPoint(pt) != (HWND) GetHWND()) - { - // Generate a LEAVE event - m_mouseInWindow = FALSE; - MSWOnMouseLeave(pt.x, pt.y, 0); - } - } -*/ - // This calls the UI-update mechanism (querying windows for // menu/toolbar/control state information) - UpdateWindowUI(); + if (wxUpdateUIEvent::CanUpdate(this)) + UpdateWindowUI(wxUPDATE_UI_FROMIDLE); } // Raise the window to the top of the Z order void wxWindowMac::Raise() { +#if TARGET_API_MAC_OSX + HIViewSetZOrder(*m_peer,kHIViewZOrderAbove, NULL) ; +#endif } // Lower the window to the bottom of the Z order void wxWindowMac::Lower() { -} - -void wxWindowMac::DoSetClientSize(int width, int height) -{ - if ( width != -1 || height != -1 ) - { - - if ( width != -1 && m_vScrollBar ) - width += MAC_SCROLLBAR_SIZE ; - if ( height != -1 && m_vScrollBar ) - height += MAC_SCROLLBAR_SIZE ; - - width += MacGetLeftBorderSize( ) + MacGetRightBorderSize( ) ; - height += MacGetTopBorderSize( ) + MacGetBottomBorderSize( ) ; - - DoSetSize( -1 , -1 , width , height ) ; - } +#if TARGET_API_MAC_OSX + HIViewSetZOrder(*m_peer,kHIViewZOrderBelow, NULL) ; +#endif } -wxWindowMac* wxWindowMac::s_lastMouseWindow = NULL ; +// static wxWindow *gs_lastWhich = NULL; -bool wxWindowMac::MacGetWindowFromPointSub( const wxPoint &point , wxWindowMac** outWin ) +bool wxWindowMac::MacSetupCursor( const wxPoint& pt) { - if ( IsTopLevel() ) - { - if ((point.x < 0) || (point.y < 0) || - (point.x > (m_width)) || (point.y > (m_height))) - return FALSE; - } - else - { - if ((point.x < m_x) || (point.y < m_y) || - (point.x > (m_x + m_width)) || (point.y > (m_y + m_height))) - return FALSE; - } - - WindowRef window = (WindowRef) MacGetRootWindow() ; + // first trigger a set cursor event - wxPoint newPoint( point ) ; - - if ( !IsTopLevel() ) + wxPoint clientorigin = GetClientAreaOrigin() ; + wxSize clientsize = GetClientSize() ; + wxCursor cursor ; + if ( wxRect2DInt( clientorigin.x , clientorigin.y , clientsize.x , clientsize.y ).Contains( wxPoint2DInt( pt ) ) ) { - newPoint.x -= m_x; - newPoint.y -= m_y; - } + wxSetCursorEvent event( pt.x , pt.y ); - for (wxNode *node = GetChildren().First(); node; node = node->Next()) - { - wxWindowMac *child = (wxWindowMac*)node->Data(); - // added the m_isShown test --dmazzoni - if ( child->MacGetRootWindow() == window && child->m_isShown ) + bool processedEvtSetCursor = GetEventHandler()->ProcessEvent(event); + if ( processedEvtSetCursor && event.HasCursor() ) { - if (child->MacGetWindowFromPointSub(newPoint , outWin )) - return TRUE; + cursor = event.GetCursor() ; } - } - - *outWin = this ; - return TRUE; -} - -bool wxWindowMac::MacGetWindowFromPoint( const wxPoint &screenpoint , wxWindowMac** outWin ) -{ - WindowRef window ; - - Point pt = { screenpoint.y , screenpoint.x } ; - if ( ::FindWindow( pt , &window ) == 3 ) - { - - wxWindowMac* win = wxFindWinFromMacWindow( window ) ; - if ( win ) + else { - // No, this yields the CLIENT are, we need the whole frame. RR. - // point = win->ScreenToClient( point ) ; - - GrafPtr port; - ::GetPort( &port ) ; - ::SetPort( UMAGetWindowPort( window ) ) ; - ::GlobalToLocal( &pt ) ; - ::SetPort( port ) ; - - wxPoint point( pt.h, pt.v ) ; - - return win->MacGetWindowFromPointSub( point , outWin ) ; - } - } - return FALSE ; -} - -extern int wxBusyCursorCount ; -static wxWindow *gs_lastWhich = NULL; - -bool wxWindowMac::MacDispatchMouseEvent(wxMouseEvent& event) -{ - if ((event.m_x < m_x) || (event.m_y < m_y) || - (event.m_x > (m_x + m_width)) || (event.m_y > (m_y + m_height))) - return FALSE; - - - if ( IsKindOf( CLASSINFO ( wxStaticBox ) ) || IsKindOf( CLASSINFO( wxSpinCtrl ) )) - return FALSE ; - WindowRef window = (WindowRef) MacGetRootWindow() ; - - event.m_x -= m_x; - event.m_y -= m_y; - - int x = event.m_x ; - int y = event.m_y ; - - for (wxNode *node = GetChildren().First(); node; node = node->Next()) - { - wxWindowMac *child = (wxWindowMac*)node->Data(); - if ( child->MacGetRootWindow() == window && child->IsShown() && child->IsEnabled() ) - { - if (child->MacDispatchMouseEvent(event)) - return TRUE; + // the test for processedEvtSetCursor is here to prevent using m_cursor + // if the user code caught EVT_SET_CURSOR() and returned nothing from + // it - this is a way to say that our cursor shouldn't be used for this + // point + if ( !processedEvtSetCursor && m_cursor.Ok() ) + { + cursor = m_cursor ; + } + if ( wxIsBusy() ) + { + } + else + { + if ( !GetParent() ) + cursor = *wxSTANDARD_CURSOR ; + } } + if ( cursor.Ok() ) + cursor.MacInstall() ; } - - event.m_x = x ; - event.m_y = y ; - event.SetEventObject( this ) ; - - if ( wxBusyCursorCount == 0 ) - { - m_cursor.MacInstall() ; - } - - if ( event.GetEventType() == wxEVT_LEFT_DOWN ) - { - // set focus to this window - if (AcceptsFocus() && FindFocus()!=this) - SetFocus(); - } - -#if wxUSE_TOOLTIPS - if ( event.GetEventType() == wxEVT_MOTION - || event.GetEventType() == wxEVT_ENTER_WINDOW - || event.GetEventType() == wxEVT_LEAVE_WINDOW ) - wxToolTip::RelayEvent( this , event); -#endif // wxUSE_TOOLTIPS - - if (gs_lastWhich != this) - { - gs_lastWhich = this; - - // Double clicks must always occur on the same window - if (event.GetEventType() == wxEVT_LEFT_DCLICK) - event.SetEventType( wxEVT_LEFT_DOWN ); - if (event.GetEventType() == wxEVT_RIGHT_DCLICK) - event.SetEventType( wxEVT_RIGHT_DOWN ); - - // Same for mouse up events - if (event.GetEventType() == wxEVT_LEFT_UP) - return TRUE; - if (event.GetEventType() == wxEVT_RIGHT_UP) - return TRUE; - } - - GetEventHandler()->ProcessEvent( event ) ; - - return TRUE; + return cursor.Ok() ; } wxString wxWindowMac::MacGetToolTipString( wxPoint &pt ) @@ -1509,182 +2519,208 @@ wxString wxWindowMac::MacGetToolTipString( wxPoint &pt ) { return m_tooltip->GetTip() ; } - return "" ; + return wxEmptyString ; } void wxWindowMac::Update() { - wxTopLevelWindowMac* win = MacGetTopLevelWindow( ) ; - if ( win ) +#if TARGET_API_MAC_OSX + WindowRef window = (WindowRef)MacGetTopLevelWindowRef() ; +#if MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_3 + // for composited windows this also triggers a redraw of all + // invalid views in the window + if( UMAGetSystemVersion() >= 0x1030 ) + HIWindowFlush(window) ; + else +#endif { - win->MacUpdate( 0 ) ; -#if TARGET_API_MAC_CARBON - if ( QDIsPortBuffered( GetWindowPort( (WindowRef) win->MacGetWindowRef() ) ) ) + // the only way to trigger the redrawing on earlier systems is to call + // ReceiveNextEvent + + EventRef currentEvent = (EventRef) wxTheApp->MacGetCurrentEvent() ; + UInt32 currentEventClass = 0 ; + UInt32 currentEventKind = 0 ; + if ( currentEvent != NULL ) + { + currentEventClass = ::GetEventClass( currentEvent ) ; + currentEventKind = ::GetEventKind( currentEvent ) ; + } + if ( currentEventClass != kEventClassMenu ) { - QDFlushPortBuffer( GetWindowPort( (WindowRef) win->MacGetWindowRef() ) , NULL ) ; + // when tracking a menu, strange redraw errors occur if we flush now, so leave.. + + EventRef theEvent; + OSStatus status = noErr ; + status = ReceiveNextEvent( 0 , NULL , kEventDurationNoWait , false , &theEvent ) ; } + else + HIViewSetNeedsDisplay( *m_peer , true ) ; + } +#else + ::Draw1Control( *m_peer ) ; #endif - } } wxTopLevelWindowMac* wxWindowMac::MacGetTopLevelWindow() const { wxTopLevelWindowMac* win = NULL ; - WindowRef window = (WindowRef) MacGetRootWindow() ; + WindowRef window = (WindowRef) MacGetTopLevelWindowRef() ; if ( window ) { win = wxFindWinFromMacWindow( window ) ; } return win ; } - -const wxRegion& wxWindowMac::MacGetVisibleRegion() +wxRegion wxWindowMac::MacGetVisibleRegion( bool includeOuterStructures ) { - RgnHandle visRgn = NewRgn() ; - RgnHandle tempRgn = NewRgn() ; - - SetRectRgn( visRgn , 0 , 0 , m_width , m_height ) ; - //TODO : as soon as the new scheme has proven to work correctly, move this to wxStaticBox - if ( IsKindOf( CLASSINFO( wxStaticBox ) ) ) - { - int borderTop = 14 ; - int borderOther = 4 ; - - SetRectRgn( tempRgn , borderOther , borderTop , m_width - borderOther , m_height - borderOther ) ; - DiffRgn( visRgn , tempRgn , visRgn ) ; - } - - if ( !IsTopLevel() ) - { - wxWindow* parent = GetParent() ; - while( parent ) - { - wxSize size = parent->GetSize() ; - int x , y ; - x = y = 0 ; - parent->MacWindowToRootWindow( &x, &y ) ; - MacRootWindowToWindow( &x , &y ) ; - SetRectRgn( tempRgn , x , y , x + size.x , y + size.y ) ; - SectRgn( visRgn , tempRgn , visRgn ) ; - if ( parent->IsTopLevel() ) - break ; - parent = parent->GetParent() ; - } - } - if ( GetWindowStyle() & wxCLIP_CHILDREN ) - { - for (wxNode *node = GetChildren().First(); node; node = node->Next()) + Rect r ; + RgnHandle visRgn = NewRgn() ; + RgnHandle tempRgn = NewRgn() ; + if ( IsControlVisible( *m_peer ) ) { - wxWindowMac *child = (wxWindowMac*)node->Data(); - - if ( !child->IsTopLevel() && child->IsShown() ) + GetControlBounds( *m_peer , &r ) ; + if (! MacGetTopLevelWindow()->MacUsesCompositing() ) { - SetRectRgn( tempRgn , child->m_x , child->m_y , child->m_x + child->m_width , child->m_y + child->m_height ) ; - DiffRgn( visRgn , tempRgn , visRgn ) ; + MacRootWindowToWindow( &r.left , & r.top ) ; + MacRootWindowToWindow( &r.right , & r.bottom ) ; } - } - } - - if ( (GetWindowStyle() & wxCLIP_SIBLINGS) && GetParent() ) - { - bool thisWindowThrough = false ; - for (wxNode *node = GetParent()->GetChildren().First(); node; node = node->Next()) - { - wxWindowMac *sibling = (wxWindowMac*)node->Data(); - if ( sibling == this ) + else { - thisWindowThrough = true ; - continue ; + r.right -= r.left ; + r.bottom -= r.top ; + r.left = 0 ; + r.top = 0 ; } - if( !thisWindowThrough ) + if ( includeOuterStructures ) + InsetRect( &r , -3 , -3 ) ; + RectRgn( visRgn , &r ) ; + if ( !IsTopLevel() ) { - continue ; - } + wxWindow* child = this ; + wxWindow* parent = child->GetParent() ; + while( parent ) + { + int x , y ; + wxSize size ; + // we have to find a better clipping algorithm here, in order not to clip things + // positioned like status and toolbar + if ( 1 /* parent->IsTopLevel() && child->IsKindOf( CLASSINFO( wxToolBar ) ) */ ) + { + size = parent->GetSize() ; + x = y = 0 ; + } + else + { + size = parent->GetClientSize() ; + wxPoint origin = parent->GetClientAreaOrigin() ; + x = origin.x ; + y = origin.y ; + } + parent->MacWindowToRootWindow( &x, &y ) ; + MacRootWindowToWindow( &x , &y ) ; - if ( !sibling->IsTopLevel() && sibling->IsShown() ) - { - SetRectRgn( tempRgn , sibling->m_x - m_x , sibling->m_y - m_y , sibling->m_x + sibling->m_width - m_x , sibling->m_y + sibling->m_height - m_y ) ; - DiffRgn( visRgn , tempRgn , visRgn ) ; + SetRectRgn( tempRgn , + x + parent->MacGetLeftBorderSize() , y + parent->MacGetTopBorderSize() , + x + size.x - parent->MacGetRightBorderSize(), + y + size.y - parent->MacGetBottomBorderSize()) ; + + SectRgn( visRgn , tempRgn , visRgn ) ; + if ( parent->IsTopLevel() ) + break ; + child = parent ; + parent = child->GetParent() ; + } } } - } - m_macVisibleRegion = visRgn ; - DisposeRgn( visRgn ) ; - DisposeRgn( tempRgn ) ; - return m_macVisibleRegion ; + + wxRegion vis = visRgn ; + DisposeRgn( visRgn ) ; + DisposeRgn( tempRgn ) ; + return vis ; } -void wxWindowMac::MacRedraw( WXHRGN updatergnr , long time, bool erase) +/* + This function must not change the updatergn ! + */ +bool wxWindowMac::MacDoRedraw( WXHRGN updatergnr , long time ) { RgnHandle updatergn = (RgnHandle) updatergnr ; - // updatergn is always already clipped to our boundaries - // it is in window coordinates, not in client coordinates - - WindowRef window = (WindowRef) MacGetRootWindow() ; - + bool handled = false ; + + // calculate a client-origin version of the update rgn and set m_updateRegion to that { - // ownUpdateRgn is the area that this window has to repaint, it is in window coordinates - RgnHandle ownUpdateRgn = NewRgn() ; - CopyRgn( updatergn , ownUpdateRgn ) ; - - SectRgn( ownUpdateRgn , (RgnHandle) MacGetVisibleRegion().GetWXHRGN() , ownUpdateRgn ) ; - - // newupdate is the update region in client coordinates RgnHandle newupdate = NewRgn() ; wxSize point = GetClientSize() ; wxPoint origin = GetClientAreaOrigin() ; SetRectRgn( newupdate , origin.x , origin.y , origin.x + point.x , origin.y+point.y ) ; - SectRgn( newupdate , ownUpdateRgn , newupdate ) ; + SectRgn( newupdate , updatergn , newupdate ) ; OffsetRgn( newupdate , -origin.x , -origin.y ) ; m_updateRegion = newupdate ; - DisposeRgn( newupdate ) ; // it's been cloned to m_updateRegion + DisposeRgn( newupdate ) ; + } - if ( erase && !EmptyRgn(ownUpdateRgn) ) - { - wxWindowDC dc(this); - dc.SetClippingRegion(wxRegion(ownUpdateRgn)); - wxEraseEvent eevent( GetId(), &dc ); - eevent.SetEventObject( this ); - GetEventHandler()->ProcessEvent( eevent ); - - wxNcPaintEvent eventNc( GetId() ); - eventNc.SetEventObject( this ); - GetEventHandler()->ProcessEvent( eventNc ); - } - DisposeRgn( ownUpdateRgn ) ; + if ( !EmptyRgn(updatergn) ) + { + wxWindowDC dc(this); + if (!EmptyRgn(updatergn)) + dc.SetClippingRegion(wxRegion(updatergn)); + + wxEraseEvent eevent( GetId(), &dc ); + eevent.SetEventObject( this ); + GetEventHandler()->ProcessEvent( eevent ); + if ( !m_updateRegion.Empty() ) { - wxPaintEvent event; - event.m_timeStamp = time ; - event.SetEventObject(this); - GetEventHandler()->ProcessEvent(event); + // paint the window itself + wxPaintEvent event; + event.m_timeStamp = time ; + event.SetEventObject(this); + handled = GetEventHandler()->ProcessEvent(event); + + // paint custom borders + wxNcPaintEvent eventNc( GetId() ); + eventNc.SetEventObject( this ); + GetEventHandler()->ProcessEvent( eventNc ); } } + return handled ; +} - // now intersect for each of the children their rect with the updateRgn and call MacRedraw recursively +void wxWindowMac::MacRedraw( WXHRGN updatergnr , long time, bool erase) +{ + RgnHandle updatergn = (RgnHandle) updatergnr ; + // updatergn is always already clipped to our boundaries + // if we are in compositing mode then it is in relative to the upper left of the control + // if we are in non-compositing, then it is relatvie to the uppder left of the content area + // of the toplevel window + // it is in window coordinates, not in client coordinates - RgnHandle childupdate = NewRgn() ; - for (wxNode *node = GetChildren().First(); node; node = node->Next()) - { - // calculate the update region for the child windows by intersecting the window rectangle with our own - // passed in update region and then offset it to be client-wise window coordinates again - wxWindowMac *child = (wxWindowMac*)node->Data(); - SetRectRgn( childupdate , child->m_x , child->m_y , child->m_x + child->m_width , child->m_y + child->m_height ) ; - SectRgn( childupdate , updatergn , childupdate ) ; - OffsetRgn( childupdate , -child->m_x , -child->m_y ) ; - if ( child->MacGetRootWindow() == window && child->IsShown() && !EmptyRgn( childupdate ) ) - { - // because dialogs may also be children - child->MacRedraw( childupdate , time , erase ) ; - } + // ownUpdateRgn is the area that this window has to repaint, it is in window coordinates + RgnHandle ownUpdateRgn = NewRgn() ; + CopyRgn( updatergn , ownUpdateRgn ) ; + + if ( MacGetTopLevelWindow()->MacUsesCompositing() == false ) + { + Rect bounds; + UMAGetControlBoundsInWindowCoords( *m_peer, &bounds ); + RgnHandle controlRgn = NewRgn(); + RectRgn( controlRgn, &bounds ); + //KO: This sets the ownUpdateRgn to the area of this control that is inside + // the window update region + SectRgn( ownUpdateRgn, controlRgn, ownUpdateRgn ); + DisposeRgn( controlRgn ); + + //KO: convert ownUpdateRgn to local coordinates + OffsetRgn( ownUpdateRgn, -bounds.left, -bounds.top ); } - DisposeRgn( childupdate ) ; - // eventually a draw grow box here + + MacDoRedraw( ownUpdateRgn , time ) ; + DisposeRgn( ownUpdateRgn ) ; } -WXHWND wxWindowMac::MacGetRootWindow() const +WXWindow wxWindowMac::MacGetTopLevelWindowRef() const { wxWindowMac *iter = (wxWindowMac*)this ; @@ -1695,13 +2731,13 @@ WXHWND wxWindowMac::MacGetRootWindow() const iter = iter->GetParent() ; } - wxASSERT_MSG( 1 , "No valid mac root window" ) ; + wxASSERT_MSG( 1 , wxT("No valid mac root window") ) ; return NULL ; } void wxWindowMac::MacCreateScrollBars( long style ) { - wxASSERT_MSG( m_vScrollBar == NULL && m_hScrollBar == NULL , "attempt to create window twice" ) ; + wxASSERT_MSG( m_vScrollBar == NULL && m_hScrollBar == NULL , wxT("attempt to create window twice") ) ; bool hasBoth = ( style & wxVSCROLL ) && ( style & wxHSCROLL ) ; int adjust = hasBoth ? MAC_SCROLLBAR_SIZE - 1: 0 ; @@ -1745,8 +2781,9 @@ void wxWindowMac::MacRepositionScrollBars() // get real client area - int width = m_width ; - int height = m_height ; + int width ; + int height ; + GetSize( &width , &height ) ; width -= MacGetLeftBorderSize() + MacGetRightBorderSize(); height -= MacGetTopBorderSize() + MacGetBottomBorderSize(); @@ -1758,8 +2795,9 @@ void wxWindowMac::MacRepositionScrollBars() int x = 0 ; int y = 0 ; - int w = m_width ; - int h = m_height ; + int w ; + int h ; + GetSize( &w , &h ) ; MacClientToRootWindow( &x , &y ) ; MacClientToRootWindow( &w , &h ) ; @@ -1771,8 +2809,7 @@ void wxWindowMac::MacRepositionScrollBars() { if ( iter->IsTopLevel() ) { - totW = iter->m_width ; - totH = iter->m_height ; + iter->GetSize( &totW , &totH ) ; break ; } @@ -1817,21 +2854,16 @@ bool wxWindowMac::AcceptsFocus() const return MacCanFocus() && wxWindowBase::AcceptsFocus(); } -WXWidget wxWindowMac::MacGetContainerForEmbedding() -{ - return GetParent()->MacGetContainerForEmbedding() ; -} - void wxWindowMac::MacSuperChangedPosition() { // only window-absolute structures have to be moved i.e. controls - wxNode *node = GetChildren().First(); + wxWindowListNode *node = GetChildren().GetFirst(); while ( node ) { - wxWindowMac *child = (wxWindowMac *)node->Data(); + wxWindowMac *child = node->GetData(); child->MacSuperChangedPosition() ; - node = node->Next(); + node = node->GetNext(); } } @@ -1839,14 +2871,15 @@ void wxWindowMac::MacTopLevelWindowChangedPosition() { // only screen-absolute structures have to be moved i.e. glcanvas - wxNode *node = GetChildren().First(); + wxWindowListNode *node = GetChildren().GetFirst(); while ( node ) { - wxWindowMac *child = (wxWindowMac *)node->Data(); + wxWindowMac *child = node->GetData(); child->MacTopLevelWindowChangedPosition() ; - node = node->Next(); + node = node->GetNext(); } } + long wxWindowMac::MacGetLeftBorderSize( ) const { if( IsTopLevel() ) @@ -1856,9 +2889,7 @@ long wxWindowMac::MacGetLeftBorderSize( ) const { SInt32 border = 3 ; #if wxMAC_USE_THEME_BORDER -#if TARGET_CARBON GetThemeMetric( kThemeMetricListBoxFrameOutset , &border ) ; -#endif #endif return border ; } @@ -1866,9 +2897,7 @@ long wxWindowMac::MacGetLeftBorderSize( ) const { SInt32 border = 3 ; #if wxMAC_USE_THEME_BORDER -#if TARGET_CARBON GetThemeMetric( kThemeMetricListBoxFrameOutset , &border ) ; -#endif #endif return border ; } @@ -1919,3 +2948,45 @@ wxPoint wxGetMousePosition() return wxPoint(x, y); } +void wxWindowMac::OnMouseEvent( wxMouseEvent &event ) +{ + if ( event.GetEventType() == wxEVT_RIGHT_DOWN ) + { + // copied from wxGTK : CS + // generate a "context menu" event: this is similar to wxEVT_RIGHT_DOWN + // except that: + // + // (a) it's a command event and so is propagated to the parent + // (b) under MSW it can be generated from kbd too + // (c) it uses screen coords (because of (a)) + wxContextMenuEvent evtCtx(wxEVT_CONTEXT_MENU, + this->GetId(), + this->ClientToScreen(event.GetPosition())); + if ( ! GetEventHandler()->ProcessEvent(evtCtx) ) + event.Skip() ; + } + else + { + event.Skip() ; + } +} + +void wxWindowMac::MacHandleControlClick( WXWidget control , wxInt16 controlpart , bool WXUNUSED( mouseStillDown ) ) +{ +} + +Rect wxMacGetBoundsForControl( wxWindow* window , const wxPoint& pos , const wxSize &size , bool adjustForOrigin ) +{ + int x ,y , w ,h ; + + window->MacGetBoundsForControl( pos , size , x , y, w, h , adjustForOrigin) ; + Rect bounds = { y , x , y+h , x+w }; + return bounds ; +} + +wxInt32 wxWindowMac::MacControlHit(WXEVENTHANDLERREF WXUNUSED(handler) , WXEVENTREF WXUNUSED(event) ) +{ + return eventNotHandledErr ; +} + +