1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: UMA support
4 // Author: Stefan Csomor
8 // Copyright: (c) Stefan Csomor
9 // Licence: The wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
17 #include <MacTextEditor.h>
20 # include <Navigation.h>
21 # if defined(TARGET_CARBON)
22 # if PM_USE_SESSION_APIS
25 # include <PMApplication.h>
27 # include <Printing.h>
34 #include "wx/mac/uma.h"
37 // since we have decided that we only support 8.6 upwards we are
38 // checking for these minimum requirements in the startup code of
39 // the application so all wxWindows code can safely assume that appearance 1.1
40 // windows manager, control manager, navigation services etc. are
43 static bool sUMAHasAppearance
= false ;
44 static long sUMAAppearanceVersion
= 0 ;
45 static long sUMASystemVersion
= 0 ;
46 static bool sUMAHasAquaLayout
= false ;
48 static bool sUMAHasInittedAppearance
= false;
50 extern int gAGABackgroundColor
;
51 bool UMAHasAppearance() { return sUMAHasAppearance
; }
52 long UMAGetAppearanceVersion() { return sUMAAppearanceVersion
; }
53 long UMAGetSystemVersion() { return sUMASystemVersion
; }
55 static bool sUMAHasWindowManager
= false ;
56 static long sUMAWindowManagerAttr
= 0 ;
58 bool UMAHasWindowManager() { return sUMAHasWindowManager
; }
59 long UMAGetWindowManagerAttr() { return sUMAWindowManagerAttr
; }
60 bool UMAHasAquaLayout() { return sUMAHasAquaLayout
; }
63 void UMACleanupToolbox()
65 if (sUMAHasInittedAppearance
)
67 UnregisterAppearanceClient() ;
69 if ( NavServicesAvailable() )
73 if ( TXNTerminateTextension
!= (void*) kUnresolvedCFragSymbolAddress
)
74 TXNTerminateTextension( ) ;
76 void UMAInitToolbox( UInt16 inMoreMastersCalls
, bool isEmbedded
)
80 for (long i
= 1; i
<= inMoreMastersCalls
; i
++)
85 ::InitGraf(&qd
.thePort
);
90 ::FlushEvents(everyEvent
, 0);
94 PurgeSpace(&total
, &contig
);
99 if ( Gestalt(gestaltSystemVersion
, &sUMASystemVersion
) != noErr
)
100 sUMASystemVersion
= 0x0000 ;
103 if ( Gestalt( gestaltAppearanceAttr
, &theAppearance
) == noErr
)
105 sUMAHasAppearance
= true ;
106 OSStatus status
= RegisterAppearanceClient();
107 // If status equals appearanceProcessRegisteredErr it means the
108 // appearance client already was registered (For example if we run
109 // embedded, the host might have registered it). In such a case
110 // we don't unregister it later on.
111 if (status
!= appearanceProcessRegisteredErr
)
113 // Appearance client wasn't registered yet.
114 sUMAHasInittedAppearance
= true;
117 if ( Gestalt( gestaltAppearanceVersion
, &theAppearance
) == noErr
)
119 sUMAAppearanceVersion
= theAppearance
;
123 sUMAAppearanceVersion
= 0x0100 ;
126 if ( Gestalt( gestaltWindowMgrAttr
, &sUMAWindowManagerAttr
) == noErr
)
128 sUMAHasWindowManager
= sUMAWindowManagerAttr
& gestaltWindowMgrPresent
;
132 // Call currently implicitely done : InitFloatingWindows() ;
136 if ( sUMAHasWindowManager
)
137 InitFloatingWindows() ;
143 if ( NavServicesAvailable() )
149 Gestalt( gestaltMenuMgrAttr
, &menuMgrAttr
) ;
150 if ( menuMgrAttr
& gestaltMenuMgrAquaLayoutMask
)
151 sUMAHasAquaLayout
= true ;
153 if ( TXNInitTextension
!= (void*) kUnresolvedCFragSymbolAddress
)
155 FontFamilyID fontId
;
159 GetThemeFont(kThemeSmallSystemFont
, GetApplicationScript() , fontName
, &fontSize
, &fontStyle
) ;
160 GetFNum( fontName
, &fontId
);
162 TXNMacOSPreferredFontDescription fontDescriptions
[] =
164 { fontId
, (fontSize
<< 16) ,kTXNDefaultFontStyle
, kTXNSystemDefaultEncoding
}
166 int noOfFontDescriptions
= sizeof( fontDescriptions
) / sizeof(TXNMacOSPreferredFontDescription
) ;
168 OptionBits options
= 0 ;
170 if ( UMAGetSystemVersion() < 0x1000 )
172 options
|= kTXNAlwaysUseQuickDrawTextMask
;
174 TXNInitTextension(fontDescriptions
, noOfFontDescriptions
, options
);
178 UMASetSystemIsInitialized(true);
183 Boolean CanUseATSUI()
186 OSErr err = Gestalt(gestaltATSUVersion, &result);
187 return (err == noErr);
191 long UMAGetProcessMode()
194 ProcessInfoRec processinfo
;
195 ProcessSerialNumber procno
;
197 procno
.highLongOfPSN
= NULL
;
198 procno
.lowLongOfPSN
= kCurrentProcess
;
199 processinfo
.processInfoLength
= sizeof(ProcessInfoRec
);
200 processinfo
.processName
= NULL
;
201 processinfo
.processAppSpec
= NULL
;
203 err
= ::GetProcessInformation( &procno
, &processinfo
) ;
204 wxASSERT( err
== noErr
) ;
205 return processinfo
.processMode
;
208 bool UMAGetProcessModeDoesActivateOnFGSwitch()
210 return UMAGetProcessMode() & modeDoesActivateOnFGSwitch
;
215 MenuRef
UMANewMenu( SInt16 id
, const wxString
& title
, wxFontEncoding encoding
)
217 wxString str
= wxStripMenuCodes( title
) ;
220 CreateNewMenu( id
, 0 , &menu
) ;
221 SetMenuTitleWithCFString( menu
, wxMacCFStringHolder(str
, encoding
) ) ;
224 wxMacStringToPascal( str
, ptitle
) ;
225 menu
= ::NewMenu( id
, ptitle
) ;
230 void UMASetMenuTitle( MenuRef menu
, const wxString
& title
, wxFontEncoding encoding
)
232 wxString str
= wxStripMenuCodes( title
) ;
234 SetMenuTitleWithCFString( menu
, wxMacCFStringHolder(str
, encoding
) ) ;
237 wxMacStringToPascal( str
, ptitle
) ;
238 SetMenuTitle( menu
, ptitle
) ;
242 void UMASetMenuItemText( MenuRef menu
, MenuItemIndex item
, const wxString
& title
, wxFontEncoding encoding
)
244 wxString str
= wxStripMenuCodes( title
) ;
246 SetMenuItemTextWithCFString( menu
, item
, wxMacCFStringHolder(str
, encoding
) ) ;
249 wxMacStringToPascal( str
, ptitle
) ;
250 SetMenuItemText( menu
, item
, ptitle
) ;
255 UInt32
UMAMenuEvent( EventRecord
*inEvent
)
257 return MenuEvent( inEvent
) ;
260 void UMAEnableMenuItem( MenuRef inMenu
, MenuItemIndex inItem
, bool enable
)
263 EnableMenuItem( inMenu
, inItem
) ;
265 DisableMenuItem( inMenu
, inItem
) ;
268 void UMAAppendSubMenuItem( MenuRef menu
, const wxString
& title
, wxFontEncoding encoding
, SInt16 id
)
270 MacAppendMenu(menu
, "\pA");
271 UMASetMenuItemText(menu
, (SInt16
) ::CountMenuItems(menu
), title
, encoding
);
272 SetMenuItemHierarchicalID( menu
, CountMenuItems( menu
) , id
) ;
275 void UMAInsertSubMenuItem( MenuRef menu
, const wxString
& title
, wxFontEncoding encoding
, MenuItemIndex item
, SInt16 id
)
277 MacInsertMenuItem(menu
, "\pA" , item
);
278 UMASetMenuItemText(menu
, item
, title
, encoding
);
279 SetMenuItemHierarchicalID( menu
, item
, id
) ;
282 void UMASetMenuItemShortcut( MenuRef menu
, MenuItemIndex item
, wxAcceleratorEntry
*entry
)
287 UInt8 modifiers
= 0 ;
288 SInt16 key
= entry
->GetKeyCode() ;
291 bool explicitCommandKey
= false ;
293 if ( entry
->GetFlags() & wxACCEL_CTRL
)
295 explicitCommandKey
= true ;
298 if (entry
->GetFlags() & wxACCEL_ALT
)
300 modifiers
|= kMenuOptionModifier
;
303 if (entry
->GetFlags() & wxACCEL_SHIFT
)
305 modifiers
|= kMenuShiftModifier
;
309 SInt16 macKey
= key
;
310 if ( key
>= WXK_F1
&& key
<= WXK_F15
)
312 // for some reasons this must be 0 right now
313 // everything else leads to just the first function key item
314 // to be selected. Thanks to Ryan Wilcox for finding out.
316 glyph
= kMenuF1Glyph
+ ( key
- WXK_F1
) ;
317 if ( key
>= WXK_F13
)
319 if ( !explicitCommandKey
)
320 modifiers
|= kMenuNoCommandModifier
;
327 macKey
= kBackspaceCharCode
;
328 glyph
= kMenuDeleteLeftGlyph
;
331 macKey
= kTabCharCode
;
332 glyph
= kMenuTabRightGlyph
;
334 case kEnterCharCode
:
335 macKey
= kEnterCharCode
;
336 glyph
= kMenuEnterGlyph
;
339 macKey
= kReturnCharCode
;
340 glyph
= kMenuReturnGlyph
;
343 macKey
= kEscapeCharCode
;
344 glyph
= kMenuEscapeGlyph
;
348 glyph
= kMenuSpaceGlyph
;
351 macKey
= kDeleteCharCode
;
352 glyph
= kMenuDeleteRightGlyph
;
355 macKey
= kClearCharCode
;
356 glyph
= kMenuClearGlyph
;
358 case WXK_PRIOR
: // PAGE UP
359 macKey
= kPageUpCharCode
;
360 glyph
= kMenuPageUpGlyph
;
363 macKey
= kPageDownCharCode
;
364 glyph
= kMenuPageDownGlyph
;
367 macKey
= kLeftArrowCharCode
;
368 glyph
= kMenuLeftArrowGlyph
;
371 macKey
= kUpArrowCharCode
;
372 glyph
= kMenuUpArrowGlyph
;
375 macKey
= kRightArrowCharCode
;
376 glyph
= kMenuRightArrowGlyph
;
379 macKey
= kDownArrowCharCode
;
380 glyph
= kMenuDownArrowGlyph
;
385 SetItemCmd( menu
, item
, macKey
);
386 SetMenuItemModifiers(menu
, item
, modifiers
) ;
389 SetMenuItemKeyGlyph(menu
, item
, glyph
) ;
393 void UMAAppendMenuItem( MenuRef menu
, const wxString
& title
, wxFontEncoding encoding
, wxAcceleratorEntry
*entry
)
395 MacAppendMenu(menu
, "\pA");
396 UMASetMenuItemText(menu
, (SInt16
) ::CountMenuItems(menu
), title
, encoding
);
397 UMASetMenuItemShortcut( menu
, (SInt16
) ::CountMenuItems(menu
), entry
) ;
400 void UMAInsertMenuItem( MenuRef menu
, const wxString
& title
, wxFontEncoding encoding
, MenuItemIndex item
, wxAcceleratorEntry
*entry
)
402 MacInsertMenuItem( menu
, "\pA" , item
) ;
403 UMASetMenuItemText(menu
, item
+1 , title
, encoding
);
404 UMASetMenuItemShortcut( menu
, item
+1 , entry
) ;
411 int gPrOpenCounter
= 0 ;
417 if ( gPrOpenCounter
== 1 )
421 wxASSERT( err
== noErr
) ;
426 OSStatus
UMAPrClose()
429 wxASSERT( gPrOpenCounter
>= 1 ) ;
430 if ( gPrOpenCounter
== 1 )
434 wxASSERT( err
== noErr
) ;
440 pascal QDGlobalsPtr
GetQDGlobalsPtr (void) ;
441 pascal QDGlobalsPtr
GetQDGlobalsPtr (void)
443 return QDGlobalsPtr (* (Ptr
*) LMGetCurrentA5 ( ) - 0xCA);
448 void UMAShowWatchCursor()
452 CursHandle watchFob
= GetCursor (watchCursor
);
459 // Cursor preservedArrow;
460 // GetQDGlobalsArrow (&preservedArrow);
461 // SetQDGlobalsArrow (*watchFob);
463 // SetQDGlobalsArrow (&preservedArrow);
464 SetCursor (*watchFob
);
466 SetCursor (*watchFob
);
471 void UMAShowArrowCursor()
475 SetCursor (GetQDGlobalsArrow (&arrow
));
477 SetCursor (&(qd
.arrow
));
483 GrafPtr
UMAGetWindowPort( WindowRef inWindowRef
)
485 wxASSERT( inWindowRef
!= NULL
) ;
487 return (GrafPtr
) GetWindowPort( inWindowRef
) ;
489 return (GrafPtr
) inWindowRef
;
493 void UMADisposeWindow( WindowRef inWindowRef
)
495 wxASSERT( inWindowRef
!= NULL
) ;
496 DisposeWindow( inWindowRef
) ;
499 void UMASetWTitle( WindowRef inWindowRef
, const wxString
& title
, wxFontEncoding encoding
)
502 SetWindowTitleWithCFString( inWindowRef
, wxMacCFStringHolder(title
, encoding
) ) ;
505 wxMacStringToPascal( title
, ptitle
) ;
506 SetWTitle( inWindowRef
, ptitle
) ;
510 // appearance additions
512 void UMASetControlTitle( ControlRef inControl
, const wxString
& title
, wxFontEncoding encoding
)
515 SetControlTitleWithCFString( inControl
, wxMacCFStringHolder(title
, encoding
) ) ;
518 wxMacStringToPascal( title
, ptitle
) ;
519 SetControlTitle( inControl
, ptitle
) ;
523 void UMAActivateControl( ControlRef inControl
)
525 #if !TARGET_API_MAC_OSX
526 // we have to add the control after again to the update rgn
527 // otherwise updates get lost
528 if ( !IsControlActive( inControl
) )
530 bool visible
= IsControlVisible( inControl
) ;
532 SetControlVisibility( inControl
, false , false ) ;
534 ::ActivateControl( inControl
) ;
535 #if !TARGET_API_MAC_OSX
537 SetControlVisibility( inControl
, true , false ) ;
540 InvalWindowRect(GetControlOwner(inControl
),UMAGetControlBoundsInWindowCoords(inControl
,&ctrlBounds
) ) ;
546 void UMADrawControl( ControlRef inControl
)
548 #if TARGET_API_MAC_CARBON
549 ::Draw1Control( inControl
);
551 WindowRef theWindow
= GetControlOwner(inControl
) ;
552 wxMacPortStateHelper
help( (GrafPtr
) GetWindowPort(theWindow
) ) ;
553 RgnHandle updateRgn
= NewRgn() ;
554 GetWindowUpdateRgn( theWindow
, updateRgn
) ;
555 Point zero
= { 0 , 0 } ;
556 LocalToGlobal( &zero
) ;
557 OffsetRgn( updateRgn
, -zero
.h
, -zero
.v
) ;
558 ::DrawControlInCurrentPort( inControl
) ;
559 InvalWindowRgn( theWindow
, updateRgn
) ;
560 DisposeRgn( updateRgn
) ;
564 void UMAMoveControl( ControlRef inControl
, short x
, short y
)
566 #if !TARGET_API_MAC_OSX
567 bool visible
= IsControlVisible( inControl
) ;
569 SetControlVisibility( inControl
, false , false ) ;
571 InvalWindowRect(GetControlOwner(inControl
),GetControlBounds(inControl
,&ctrlBounds
) ) ;
574 ::MoveControl( inControl
, x
, y
) ;
575 #if !TARGET_API_MAC_OSX
577 SetControlVisibility( inControl
, true , false ) ;
579 InvalWindowRect(GetControlOwner(inControl
),GetControlBounds(inControl
,&ctrlBounds
) ) ;
584 void UMASizeControl( ControlRef inControl
, short x
, short y
)
586 #if !TARGET_API_MAC_OSX
587 bool visible
= IsControlVisible( inControl
) ;
589 SetControlVisibility( inControl
, false , false ) ;
591 InvalWindowRect(GetControlOwner(inControl
),GetControlBounds(inControl
,&ctrlBounds
) ) ;
594 ::SizeControl( inControl
, x
, y
) ;
595 #if !TARGET_API_MAC_OSX
597 SetControlVisibility( inControl
, true , false ) ;
599 InvalWindowRect(GetControlOwner(inControl
),GetControlBounds(inControl
,&ctrlBounds
) ) ;
604 void UMADeactivateControl( ControlRef inControl
)
606 #if !TARGET_API_MAC_OSX
607 // we have to add the control after again to the update rgn
608 // otherwise updates get lost
609 bool visible
= IsControlVisible( inControl
) ;
611 SetControlVisibility( inControl
, false , false ) ;
613 ::DeactivateControl( inControl
) ;
614 #if !TARGET_API_MAC_OSX
616 SetControlVisibility( inControl
, true , false ) ;
618 InvalWindowRect(GetControlOwner(inControl
),UMAGetControlBoundsInWindowCoords(inControl
,&ctrlBounds
) ) ;
622 // shows the control and adds the region to the update region
623 void UMAShowControl (ControlRef inControl
)
625 SetControlVisibility( inControl
, true , false ) ;
627 InvalWindowRect(GetControlOwner(inControl
),UMAGetControlBoundsInWindowCoords(inControl
,&ctrlBounds
) ) ;
630 // hides the control and adds the region to the update region
631 void UMAHideControl (ControlRef inControl
)
633 SetControlVisibility( inControl
, false , false ) ;
635 InvalWindowRect(GetControlOwner(inControl
),UMAGetControlBoundsInWindowCoords(inControl
,&ctrlBounds
) ) ;
638 OSErr
UMASetKeyboardFocus (WindowPtr inWindow
,
639 ControlRef inControl
,
640 ControlFocusPart inPart
)
646 SetPortWindowPort( inWindow
) ;
648 err
= SetKeyboardFocus( inWindow
, inControl
, inPart
) ;
654 void UMAUpdateControls( WindowPtr inWindow
, RgnHandle inRgn
)
656 wxMacPortStateHelper
help( (GrafPtr
) GetWindowPort( (WindowRef
) inWindow
) ) ;
657 RgnHandle updateRgn
= NewRgn() ;
658 GetWindowUpdateRgn( inWindow
, updateRgn
) ;
660 Point zero
= { 0 , 0 } ;
661 LocalToGlobal( &zero
) ;
662 OffsetRgn( updateRgn
, -zero
.h
, -zero
.v
) ;
664 UpdateControls( inWindow
, inRgn
) ;
665 InvalWindowRgn( inWindow
, updateRgn
) ;
666 DisposeRgn( updateRgn
) ;
669 bool UMAIsWindowFloating( WindowRef inWindow
)
673 GetWindowClass( inWindow
, &cl
) ;
674 return cl
== kFloatingWindowClass
;
677 bool UMAIsWindowModal( WindowRef inWindow
)
681 GetWindowClass( inWindow
, &cl
) ;
682 return cl
< kFloatingWindowClass
;
687 void UMAHighlightAndActivateWindow( WindowRef inWindowRef
, bool inActivate
)
691 // bool isHighlighted = IsWindowHighlited( inWindowRef ) ;
692 // if ( inActivate != isHightlited )
695 SetPortWindowPort( inWindowRef
) ;
696 HiliteWindow( inWindowRef
, inActivate
) ;
697 ControlRef control
= NULL
;
698 ::GetRootControl( inWindowRef
, & control
) ;
702 UMAActivateControl( control
) ;
704 UMADeactivateControl( control
) ;
710 OSStatus
UMADrawThemePlacard( const Rect
*inRect
, ThemeDrawState inState
)
712 return ::DrawThemePlacard( inRect
, inState
) ;
716 static OSStatus helpMenuStatus
= noErr
;
717 static MenuItemIndex firstCustomItemIndex
= 0 ;
720 OSStatus
UMAGetHelpMenu(
721 MenuRef
* outHelpMenu
,
722 MenuItemIndex
* outFirstCustomItemIndex
)
725 return HMGetHelpMenu( outHelpMenu
, outFirstCustomItemIndex
) ;
727 MenuRef helpMenuHandle
;
728 helpMenuStatus
= HMGetHelpMenuHandle( &helpMenuHandle
) ;
729 if ( firstCustomItemIndex
== 0 && helpMenuStatus
== noErr
)
731 firstCustomItemIndex
= CountMenuItems( helpMenuHandle
) + 1 ;
733 if ( outFirstCustomItemIndex
)
735 *outFirstCustomItemIndex
= firstCustomItemIndex
;
737 *outHelpMenu
= helpMenuHandle
;
738 return helpMenuStatus
;
742 wxMacPortStateHelper::wxMacPortStateHelper( GrafPtr newport
)
748 wxMacPortStateHelper::wxMacPortStateHelper()
753 void wxMacPortStateHelper::Setup( GrafPtr newport
)
755 GetPort( &m_oldPort
) ;
758 wxASSERT_MSG( m_clip
== NULL
, wxT("Cannot call setup twice") ) ;
761 m_textFont
= GetPortTextFont( (CGrafPtr
) newport
);
762 m_textSize
= GetPortTextSize( (CGrafPtr
) newport
);
763 m_textStyle
= GetPortTextFace( (CGrafPtr
) newport
);
764 m_textMode
= GetPortTextMode( (CGrafPtr
) newport
);
765 GetThemeDrawingState( &m_drawingState
) ;
766 m_currentPort
= newport
;
768 void wxMacPortStateHelper::Clear()
772 DisposeRgn( m_clip
) ;
773 DisposeThemeDrawingState( m_drawingState
) ;
778 wxMacPortStateHelper::~wxMacPortStateHelper()
782 SetPort( m_currentPort
) ;
784 DisposeRgn( m_clip
) ;
785 TextFont( m_textFont
);
786 TextSize( m_textSize
);
787 TextFace( m_textStyle
);
788 TextMode( m_textMode
);
789 SetThemeDrawingState( m_drawingState
, true ) ;
790 SetPort( m_oldPort
) ;
794 OSStatus
UMAPutScrap( Size size
, OSType type
, void *data
)
796 OSStatus err
= noErr
;
798 err
= PutScrap( size
, type
, data
) ;
801 err
= GetCurrentScrap (&scrap
);
804 err
= PutScrapFlavor (scrap
, type
, 0, size
, data
);
810 Rect
* UMAGetControlBoundsInWindowCoords(ControlRef theControl
, Rect
*bounds
)
812 wxWindow
* win
= wxFindControlFromMacControl( theControl
) ;
814 GetControlBounds( theControl
, bounds
) ;
815 #if TARGET_API_MAC_OSX
816 if ( win
!= NULL
&& win
->MacGetTopLevelWindow() != NULL
)
822 win
->GetParent()->MacWindowToRootWindow( &x
, & y
) ;
826 bounds
->bottom
+= y
;
837 static bool sUMASystemInitialized
= false ;
839 bool UMASystemIsInitialized()
841 return sUMASystemInitialized
;
844 void UMASetSystemIsInitialized(bool val
)
846 sUMASystemInitialized
= val
;