/************************************************************************* * * OpenOffice.org - a multi-platform office productivity suite * * $RCSfile: salinst.cxx,v $ * * $Revision: 1.46 $ * * last change: $Author: obo $ $Date: 2008-03-10 07:37:29 $ * * The Contents of this file are made available subject to * the terms of GNU Lesser General Public License Version 2.1. * * * GNU Lesser General Public License Version 2.1 * ============================================= * Copyright 2005 by Sun Microsystems, Inc. * 901 San Antonio Road, Palo Alto, CA 94303, USA * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License version 2.1, as published by the Free Software Foundation. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, * MA 02111-1307 USA * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_vcl.hxx" #include #include "tools/fsys.hxx" #include "osl/process.h" #include "rtl/ustrbuf.hxx" #include "vcl/svapp.hxx" #include "vcl/print.h" #include "vcl/salimestatus.hxx" #include "vcl/window.hxx" #include "vcl/timer.hxx" #include "saldata.hxx" #include "salinst.h" #include "salframe.h" #include "salobj.h" #include "salsys.h" #include "salvd.h" #include "salbmp.h" #include "salprn.h" #include "salogl.h" #include "saltimer.h" #include "vclnsapp.h" #include "premac.h" #include #include "postmac.h" using namespace std; extern BOOL ImplSVMain(); static BOOL* gpbInit = 0; static NSMenu* pDockMenu = nil; static bool bNoSVMain = true; static bool bLeftMain = false; // ----------------------------------------------------------------------- class AquaDelayedSettingsChanged : public Timer { bool mbInvalidate; public: AquaDelayedSettingsChanged( bool bInvalidate ) : mbInvalidate( bInvalidate ) { } virtual void Timeout() { SalData* pSalData = GetSalData(); if( ! pSalData->maFrames.empty() ) pSalData->maFrames.front()->CallCallback( SALEVENT_SETTINGSCHANGED, NULL ); if( mbInvalidate ) { for( std::list< AquaSalFrame* >::iterator it = pSalData->maFrames.begin(); it != pSalData->maFrames.end(); ++it ) { if( (*it)->mbShown ) (*it)->SendPaintEvent( NULL ); } } Stop(); delete this; } }; void AquaSalInstance::delayedSettingsChanged( bool bInvalidate ) { vos::OGuard aGuard( *mpSalYieldMutex ); AquaDelayedSettingsChanged* pTimer = new AquaDelayedSettingsChanged( bInvalidate ); pTimer->SetTimeout( 50 ); pTimer->Start(); } // the AppEventList must be available before any SalData/SalInst/etc. objects are ready typedef std::list AppEventList; AppEventList AquaSalInstance::aAppEventList; NSMenu* AquaSalInstance::GetDynamicDockMenu() { if( ! pDockMenu && ! bLeftMain ) pDockMenu = [[NSMenu alloc] initWithTitle: @""]; return pDockMenu; } bool AquaSalInstance::isOnCommandLine( const rtl::OUString& rArg ) { sal_uInt32 nArgs = osl_getCommandArgCount(); for( sal_uInt32 i = 0; i < nArgs; i++ ) { rtl::OUString aArg; osl_getCommandArg( i, &aArg.pData ); if( aArg.equals( rArg ) ) return true; } return false; } // initialize the cocoa VCL_NSApplication object // returns an NSAutoreleasePool that must be released when the event loop begins static void initNSApp() { // create our cocoa NSApplication [VCL_NSApplication sharedApplication]; SalData::ensureThreadAutoreleasePool(); // put cocoa into multithreaded mode [NSThread detachNewThreadSelector:@selector(enableCocoaThreads:) toTarget:[[CocoaThreadEnabler alloc] init] withObject:nil]; // activate our delegate methods [NSApp setDelegate: NSApp]; [[NSNotificationCenter defaultCenter] addObserver: NSApp selector: @selector(systemColorsChanged:) name: NSSystemColorsDidChangeNotification object: nil ]; [[NSNotificationCenter defaultCenter] addObserver: NSApp selector: @selector(screenParametersChanged:) name: NSApplicationDidChangeScreenParametersNotification object: nil ]; // add observers for some settings changes that affect vcl's settings // scrollbar variant [[NSDistributedNotificationCenter defaultCenter] addObserver: NSApp selector: @selector(scrollbarVariantChanged:) name: @"AppleAquaScrollBarVariantChanged" object: nil ]; // scrollbar page behavior ("jump to here" or not) [[NSDistributedNotificationCenter defaultCenter] addObserver: NSApp selector: @selector(scrollbarSettingsChanged:) name: @"AppleNoRedisplayAppearancePreferenceChanged" object: nil ]; } BOOL ImplSVMainHook( BOOL * pbInit ) { gpbInit = pbInit; bNoSVMain = false; initNSApp(); NSPoint aPt = { 0, 0 }; NSEvent* pEvent = [NSEvent otherEventWithType: NSApplicationDefined location: aPt modifierFlags: 0 timestamp: 0 windowNumber: 0 context: nil subtype: AquaSalInstance::AppExecuteSVMain data1: 0 data2: 0 ]; if( pEvent ) { [NSApp postEvent: pEvent atStart: NO]; rtl::OUString aExeURL, aExe; osl_getExecutableFile( &aExeURL.pData ); osl_getSystemPathFromFileURL( aExeURL.pData, &aExe.pData ); rtl::OString aByteExe( rtl::OUStringToOString( aExe, osl_getThreadTextEncoding() ) ); const char* pArgv[] = { aByteExe.getStr(), NULL }; NSApplicationMain( 1, pArgv ); } else { DBG_ERROR( "NSApplication initialization could not be done" ); } return TRUE; // indicate that ImplSVMainHook is implemented } // ======================================================================= void SalAbort( const XubString& rErrorText ) { if( !rErrorText.Len() ) fprintf( stderr, "Application Error " ); else fprintf( stderr, "%s ", ByteString( rErrorText, gsl_getSystemTextEncoding() ).GetBuffer() ); abort(); } // ----------------------------------------------------------------------- void InitSalData() { SalData *pSalData = new SalData; SetSalData( pSalData ); } // ----------------------------------------------------------------------- const ::rtl::OUString& SalGetDesktopEnvironment() { static OUString aDesktopEnvironment(RTL_CONSTASCII_USTRINGPARAM( "MacOSX" )); return aDesktopEnvironment; } // ----------------------------------------------------------------------- void DeInitSalData() { SalData *pSalData = GetSalData(); delete pSalData; SetSalData( NULL ); } // ----------------------------------------------------------------------- extern "C" { #include } // ----------------------------------------------------------------------- void InitSalMain() { rtl_uString *urlWorkDir = NULL; rtl_uString *sysWorkDir = NULL; oslProcessError err1 = osl_getProcessWorkingDir(&urlWorkDir); if (err1 == osl_Process_E_None) { oslFileError err2 = osl_getSystemPathFromFileURL(urlWorkDir, &sysWorkDir); if (err2 == osl_File_E_None) { ByteString aPath( getenv( "PATH" ) ); ByteString aResPath( getenv( "STAR_RESOURCEPATH" ) ); ByteString aLibPath( getenv( "DYLD_LIBRARY_PATH" ) ); ByteString aCmdPath( OUStringToOString(OUString(sysWorkDir), RTL_TEXTENCODING_UTF8).getStr() ); ByteString aTmpPath; // Get absolute path of command's directory if ( aCmdPath.Len() ) { DirEntry aCmdDirEntry( aCmdPath ); aCmdDirEntry.ToAbs(); aCmdPath = ByteString( aCmdDirEntry.GetPath().GetFull(), RTL_TEXTENCODING_ASCII_US ); } // Assign to PATH environment variable if ( aCmdPath.Len() ) { aTmpPath = ByteString( "PATH=" ); aTmpPath += aCmdPath; if ( aPath.Len() ) aTmpPath += ByteString( DirEntry::GetSearchDelimiter(), RTL_TEXTENCODING_ASCII_US ); aTmpPath += aPath; putenv( (char*)aTmpPath.GetBuffer() ); } // Assign to STAR_RESOURCEPATH environment variable if ( aCmdPath.Len() ) { aTmpPath = ByteString( "STAR_RESOURCEPATH=" ); aTmpPath += aCmdPath; if ( aResPath.Len() ) aTmpPath += ByteString( DirEntry::GetSearchDelimiter(), RTL_TEXTENCODING_ASCII_US ); aTmpPath += aResPath; putenv( (char*)aTmpPath.GetBuffer() ); } // Assign to DYLD_LIBRARY_PATH environment variable if ( aCmdPath.Len() ) { aTmpPath = ByteString( "DYLD_LIBRARY_PATH=" ); aTmpPath += aCmdPath; if ( aLibPath.Len() ) aTmpPath += ByteString( DirEntry::GetSearchDelimiter(), RTL_TEXTENCODING_ASCII_US ); aTmpPath += aLibPath; putenv( (char*)aTmpPath.GetBuffer() ); } } } } // ----------------------------------------------------------------------- void DeInitSalMain() { } // ======================================================================= SalYieldMutex::SalYieldMutex() { mnCount = 0; mnThreadId = 0; } void SalYieldMutex::acquire() { OMutex::acquire(); mnThreadId = NAMESPACE_VOS(OThread)::getCurrentIdentifier(); mnCount++; } void SalYieldMutex::release() { if ( mnThreadId == NAMESPACE_VOS(OThread)::getCurrentIdentifier() ) { if ( mnCount == 1 ) mnThreadId = 0; mnCount--; } OMutex::release(); } sal_Bool SalYieldMutex::tryToAcquire() { if ( OMutex::tryToAcquire() ) { mnThreadId = NAMESPACE_VOS(OThread)::getCurrentIdentifier(); mnCount++; return sal_True; } else return sal_False; } // ----------------------------------------------------------------------- // some convenience functions regarding the yield mutex, aka solar mutex BOOL ImplSalYieldMutexTryToAcquire() { AquaSalInstance* pInst = (AquaSalInstance*) GetSalData()->mpFirstInstance; if ( pInst ) return pInst->mpSalYieldMutex->tryToAcquire(); else return FALSE; } void ImplSalYieldMutexAcquire() { AquaSalInstance* pInst = (AquaSalInstance*) GetSalData()->mpFirstInstance; if ( pInst ) pInst->mpSalYieldMutex->acquire(); } void ImplSalYieldMutexRelease() { AquaSalInstance* pInst = (AquaSalInstance*) GetSalData()->mpFirstInstance; if ( pInst ) pInst->mpSalYieldMutex->release(); } // ======================================================================= SalInstance* CreateSalInstance() { // this is the case for not using SVMain // not so good if( bNoSVMain ) initNSApp(); SalData* pSalData = GetSalData(); DBG_ASSERT( pSalData->mpFirstInstance == NULL, "more than one instance created" ); AquaSalInstance* pInst = new AquaSalInstance; // init instance (only one instance in this version !!!) pSalData->mpFirstInstance = pInst; // this one is for outside AquaSalInstance::Yield SalData::ensureThreadAutoreleasePool(); // no focus rects on NWF aqua ImplGetSVData()->maNWFData.mbNoFocusRects = true; ImplGetSVData()->maNWFData.mbNoBoldTabFocus = true; ImplGetSVData()->maNWFData.mbCenteredTabs = true; ImplGetSVData()->maNWFData.mbProgressNeedsErase = true; ImplGetSVData()->maNWFData.mbCheckBoxNeedsErase = true; ImplGetSVData()->maGDIData.mbPrinterPullModel = true; return pInst; } // ----------------------------------------------------------------------- void DestroySalInstance( SalInstance* pInst ) { delete pInst; } // ----------------------------------------------------------------------- AquaSalInstance::AquaSalInstance() { mpSalYieldMutex = new SalYieldMutex; mpSalYieldMutex->acquire(); maMainThread = vos::OThread::getCurrentIdentifier(); mbWaitingYield = false; maUserEventListMutex = osl_createMutex(); } // ----------------------------------------------------------------------- AquaSalInstance::~AquaSalInstance() { mpSalYieldMutex->release(); delete mpSalYieldMutex; osl_destroyMutex( maUserEventListMutex ); } // ----------------------------------------------------------------------- void AquaSalInstance::wakeupYield() { // wakeup :Yield if( mbWaitingYield ) { SalData::ensureThreadAutoreleasePool(); NSPoint aPt = { 0, 0 }; NSEvent* pEvent = [NSEvent otherEventWithType: NSApplicationDefined location: aPt modifierFlags: 0 timestamp: 0 windowNumber: 0 context: nil subtype: AquaSalInstance::YieldWakeupEvent data1: 0 data2: 0 ]; if( pEvent ) [NSApp postEvent: pEvent atStart: NO]; } } // ----------------------------------------------------------------------- void AquaSalInstance::PostUserEvent( AquaSalFrame* pFrame, USHORT nType, void* pData ) { osl_acquireMutex( maUserEventListMutex ); maUserEvents.push_back( SalUserEvent( pFrame, pData, nType ) ); osl_releaseMutex( maUserEventListMutex ); // notify main loop that an event has arrived wakeupYield(); } // ----------------------------------------------------------------------- vos::IMutex* AquaSalInstance::GetYieldMutex() { return mpSalYieldMutex; } // ----------------------------------------------------------------------- ULONG AquaSalInstance::ReleaseYieldMutex() { SalYieldMutex* pYieldMutex = mpSalYieldMutex; if ( pYieldMutex->GetThreadId() == NAMESPACE_VOS(OThread)::getCurrentIdentifier() ) { ULONG nCount = pYieldMutex->GetAcquireCount(); ULONG n = nCount; while ( n ) { pYieldMutex->release(); n--; } return nCount; } else return 0; } // ----------------------------------------------------------------------- void AquaSalInstance::AcquireYieldMutex( ULONG nCount ) { SalYieldMutex* pYieldMutex = mpSalYieldMutex; while ( nCount ) { pYieldMutex->acquire(); nCount--; } } // ----------------------------------------------------------------------- bool AquaSalInstance::isNSAppThread() const { return vos::OThread::getCurrentIdentifier() == maMainThread; } // ----------------------------------------------------------------------- void AquaSalInstance::handleAppDefinedEvent( NSEvent* pEvent ) { switch( [pEvent subtype] ) { case AppStartTimerEvent: AquaSalTimer::handleStartTimerEvent( pEvent ); break; case AppEndLoopEvent: [NSApp stop: NSApp]; break; case AppExecuteSVMain: { BOOL bResult = ImplSVMain(); if( gpbInit ) *gpbInit = bResult; [NSApp stop: NSApp]; bLeftMain = true; if( pDockMenu ) { [pDockMenu release]; pDockMenu = nil; } } break; case YieldWakeupEvent: // do nothing, fall out of Yield break; default: DBG_ERROR( "unhandled NSApplicationDefined event" ); break; }; } // ----------------------------------------------------------------------- class ReleasePoolHolder { NSAutoreleasePool* mpPool; public: ReleasePoolHolder() : mpPool( [[NSAutoreleasePool alloc] init] ) {} ~ReleasePoolHolder() { [mpPool release]; } }; void AquaSalInstance::Yield( bool bWait, bool bHandleAllCurrentEvents ) { // ensure that the per thread autorelease pool is top level and // will therefore not be destroyed by cocoa implicitly SalData::ensureThreadAutoreleasePool(); // NSAutoreleasePool documentation suggests we should have // an own pool for each yield level ReleasePoolHolder aReleasePool; // Release all locks so that we don't deadlock when we pull pending // events from the event queue bool bDispatchUser = true; while( bDispatchUser ) { ULONG nCount = ReleaseYieldMutex(); // get one user event osl_acquireMutex( maUserEventListMutex ); SalUserEvent aEvent( NULL, NULL, 0 ); if( ! maUserEvents.empty() ) { aEvent = maUserEvents.front(); maUserEvents.pop_front(); } else bDispatchUser = false; osl_releaseMutex( maUserEventListMutex ); AcquireYieldMutex( nCount ); // dispatch it if( aEvent.mpFrame && AquaSalFrame::isAlive( aEvent.mpFrame ) ) { aEvent.mpFrame->CallCallback( aEvent.mnType, aEvent.mpData ); // return if only one event is asked for if( ! bHandleAllCurrentEvents ) return; } } // handle cocoa event queue // cocoa events mye be only handled in the thread the NSApp was created if( isNSAppThread() ) { // we need to be woken up by a cocoa-event // if a user event should be posted by the event handling below bool bOldWaitingYield = mbWaitingYield; mbWaitingYield = bWait; // handle available events NSEvent* pEvent = nil; bool bHadEvent = false; do { ULONG nCount = ReleaseYieldMutex(); pEvent = [NSApp nextEventMatchingMask: NSAnyEventMask untilDate: nil inMode: NSDefaultRunLoopMode dequeue: YES]; if( pEvent ) { [NSApp sendEvent: pEvent]; bHadEvent = true; } [NSApp updateWindows]; AcquireYieldMutex( nCount ); } while( bHandleAllCurrentEvents && pEvent ); // if we had no event yet, wait for one if requested if( bWait && ! bHadEvent ) { ULONG nCount = ReleaseYieldMutex(); pEvent = [NSApp nextEventMatchingMask: NSAnyEventMask untilDate: [NSDate distantFuture] inMode: NSDefaultRunLoopMode dequeue: YES]; if( pEvent ) [NSApp sendEvent: pEvent]; [NSApp updateWindows]; AcquireYieldMutex( nCount ); } mbWaitingYield = bOldWaitingYield; // collect update rectangles const std::list< AquaSalFrame* > rFrames( GetSalData()->maFrames ); for( std::list< AquaSalFrame* >::const_iterator it = rFrames.begin(); it != rFrames.end(); ++it ) { if( (*it)->mbShown && ! (*it)->maInvalidRect.IsEmpty() ) { (*it)->Flush( (*it)->maInvalidRect ); (*it)->maInvalidRect.SetEmpty(); } } } // we get some apple events way too early // before the application is ready to handle them, // so their corresponding application events need to be delayed // now is a good time to handle at least one of them if( bWait && !aAppEventList.empty() && ImplGetSVData()->maAppData.mbInAppExecute ) { // make sure that only one application event is active at a time static bool bInAppEvent = false; if( !bInAppEvent ) { bInAppEvent = true; // get the next delayed application event const ApplicationEvent* pAppEvent = aAppEventList.front(); aAppEventList.pop_front(); // handle one application event (no recursion) const ImplSVData* pSVData = ImplGetSVData(); pSVData->mpApp->AppEvent( *pAppEvent ); delete pAppEvent; // allow the next delayed application event bInAppEvent = false; } } } // ----------------------------------------------------------------------- bool AquaSalInstance::AnyInput( USHORT nType ) { if( nType & INPUT_APPEVENT ) { if( ! aAppEventList.empty() ) return true; if( nType == INPUT_APPEVENT ) return false; } unsigned/*NSUInteger*/ nEventMask = 0; if( nType & INPUT_MOUSE) nEventMask |= NSLeftMouseDownMask | NSRightMouseDownMask | NSOtherMouseDownMask | NSLeftMouseUpMask | NSRightMouseUpMask | NSOtherMouseUpMask | NSLeftMouseDraggedMask | NSRightMouseDraggedMask | NSOtherMouseDraggedMask | NSScrollWheelMask | // NSMouseMovedMask | NSMouseEnteredMask | NSMouseExitedMask; if( nType & INPUT_KEYBOARD) nEventMask |= NSKeyDownMask | NSKeyUpMask | NSFlagsChangedMask; if( nType & INPUT_OTHER) nEventMask |= NSTabletPoint; // TODO: INPUT_PAINT / INPUT_TIMER / more INPUT_OTHER if( !nType) return false; NSEvent* pEvent = [NSApp nextEventMatchingMask: nEventMask untilDate: nil inMode: NSDefaultRunLoopMode dequeue: NO]; return (pEvent != NULL); } // ----------------------------------------------------------------------- SalFrame* AquaSalInstance::CreateChildFrame( SystemParentData* pSystemParentData, ULONG nSalFrameStyle ) { return NULL; } // ----------------------------------------------------------------------- SalFrame* AquaSalInstance::CreateFrame( SalFrame* pParent, ULONG nSalFrameStyle ) { SalData::ensureThreadAutoreleasePool(); SalFrame* pFrame = new AquaSalFrame( pParent, nSalFrameStyle ); return pFrame; } // ----------------------------------------------------------------------- void AquaSalInstance::DestroyFrame( SalFrame* pFrame ) { delete pFrame; } // ----------------------------------------------------------------------- SalObject* AquaSalInstance::CreateObject( SalFrame* pParent, SystemWindowData* /* pWindowData */, BOOL /* bShow */ ) { // SystemWindowData is meaningless on Mac OS X AquaSalObject *pObject = NULL; if ( pParent ) pObject = new AquaSalObject( static_cast(pParent) ); return pObject; } // ----------------------------------------------------------------------- void AquaSalInstance::DestroyObject( SalObject* pObject ) { delete ( pObject ); } // ----------------------------------------------------------------------- SalPrinter* AquaSalInstance::CreatePrinter( SalInfoPrinter* pInfoPrinter ) { return new AquaSalPrinter( dynamic_cast(pInfoPrinter) ); } // ----------------------------------------------------------------------- void AquaSalInstance::DestroyPrinter( SalPrinter* pPrinter ) { delete pPrinter; } // ----------------------------------------------------------------------- void AquaSalInstance::GetPrinterQueueInfo( ImplPrnQueueList* pList ) { NSArray* pNames = [NSPrinter printerNames]; NSArray* pTypes = [NSPrinter printerTypes]; unsigned int nNameCount = pNames ? [pNames count] : 0; unsigned int nTypeCount = pTypes ? [pTypes count] : 0; DBG_ASSERT( nTypeCount == nNameCount, "type count not equal to printer count" ); for( unsigned int i = 0; i < nNameCount; i++ ) { NSString* pName = [pNames objectAtIndex: i]; NSString* pType = i < nTypeCount ? [pTypes objectAtIndex: i] : nil; if( pName ) { SalPrinterQueueInfo* pInfo = new SalPrinterQueueInfo; pInfo->maPrinterName = GetOUString( pName ); if( pType ) pInfo->maDriver = GetOUString( pType ); pInfo->mnStatus = 0; pInfo->mnJobs = 0; pInfo->mpSysData = NULL; pList->Add( pInfo ); } } } // ----------------------------------------------------------------------- void AquaSalInstance::GetPrinterQueueState( SalPrinterQueueInfo* pInfo ) { } // ----------------------------------------------------------------------- void AquaSalInstance::DeletePrinterQueueInfo( SalPrinterQueueInfo* pInfo ) { delete pInfo; } // ----------------------------------------------------------------------- XubString AquaSalInstance::GetDefaultPrinter() { if( ! maDefaultPrinter.getLength() ) { NSPrintInfo* pPI = [NSPrintInfo sharedPrintInfo]; DBG_ASSERT( pPI, "no print info" ); if( pPI ) { NSPrinter* pPr = [pPI printer]; DBG_ASSERT( pPr, "no printer in default info" ); if( pPr ) { NSString* pDefName = [pPr name]; DBG_ASSERT( pDefName, "printer has no name" ); maDefaultPrinter = GetOUString( pDefName ); } } } return maDefaultPrinter; } // ----------------------------------------------------------------------- SalInfoPrinter* AquaSalInstance::CreateInfoPrinter( SalPrinterQueueInfo* pQueueInfo, ImplJobSetup* pSetupData ) { SalInfoPrinter* pNewInfoPrinter = NULL; if( pQueueInfo ) { pNewInfoPrinter = new AquaSalInfoPrinter( *pQueueInfo ); if( pSetupData ) pNewInfoPrinter->SetPrinterData( pSetupData ); } return pNewInfoPrinter; } // ----------------------------------------------------------------------- void AquaSalInstance::DestroyInfoPrinter( SalInfoPrinter* pPrinter ) { delete pPrinter; } // ----------------------------------------------------------------------- SalSystem* AquaSalInstance::CreateSystem() { return new AquaSalSystem(); } // ----------------------------------------------------------------------- void AquaSalInstance::DestroySystem( SalSystem* pSystem ) { delete pSystem; } // ----------------------------------------------------------------------- void AquaSalInstance::SetEventCallback( void* pInstance, bool(*pCallback)(void*,void*,int) ) { } // ----------------------------------------------------------------------- void AquaSalInstance::SetErrorEventCallback( void* pInstance, bool(*pCallback)(void*,void*,int) ) { } // ----------------------------------------------------------------------- void* AquaSalInstance::GetConnectionIdentifier( ConnectionIdentifierType& rReturnedType, int& rReturnedBytes ) { rReturnedBytes = 1; rReturnedType = AsciiCString; return (void*)""; } // ----------------------------------------------------------------------- SalTimer* AquaSalInstance::CreateSalTimer() { return new AquaSalTimer(); } // ----------------------------------------------------------------------- SalOpenGL* AquaSalInstance::CreateSalOpenGL( SalGraphics* pGraphics ) { return new AquaSalOpenGL( pGraphics ); } // ----------------------------------------------------------------------- SalSystem* AquaSalInstance::CreateSalSystem() { return new AquaSalSystem(); } // ----------------------------------------------------------------------- SalBitmap* AquaSalInstance::CreateSalBitmap() { return new AquaSalBitmap(); } // ----------------------------------------------------------------------- SalSession* AquaSalInstance::CreateSalSession() { return NULL; } // ----------------------------------------------------------------------- class MacImeStatus : public SalI18NImeStatus { public: MacImeStatus() {} virtual ~MacImeStatus() {} // asks whether there is a status window available // to toggle into menubar virtual bool canToggle() { return false; } virtual void toggle() {} }; // ----------------------------------------------------------------------- SalI18NImeStatus* AquaSalInstance::CreateI18NImeStatus() { return new MacImeStatus(); } // YieldMutexReleaser YieldMutexReleaser::YieldMutexReleaser() : mnCount( 0 ) { SalData* pSalData = GetSalData(); if( ! pSalData->mpFirstInstance->isNSAppThread() ) { SalData::ensureThreadAutoreleasePool(); mnCount = pSalData->mpFirstInstance->ReleaseYieldMutex(); } } YieldMutexReleaser::~YieldMutexReleaser() { if( mnCount != 0 ) GetSalData()->mpFirstInstance->AcquireYieldMutex( mnCount ); } ////////////////////////////////////////////////////////////// rtl::OUString GetOUString( CFStringRef rStr ) { if( rStr == 0 ) return rtl::OUString(); CFIndex nLength = CFStringGetLength( rStr ); if( nLength == 0 ) return rtl::OUString(); const UniChar* pConstStr = CFStringGetCharactersPtr( rStr ); if( pConstStr ) return rtl::OUString( pConstStr, nLength ); UniChar* pStr = reinterpret_cast( rtl_allocateMemory( sizeof(UniChar)*nLength ) ); CFRange aRange = { 0, nLength }; CFStringGetCharacters( rStr, aRange, pStr ); rtl::OUString aRet( pStr, nLength ); rtl_freeMemory( pStr ); return aRet; } rtl::OUString GetOUString( NSString* pStr ) { if( ! pStr ) return rtl::OUString(); int nLen = [pStr length]; if( nLen == 0 ) return rtl::OUString(); rtl::OUStringBuffer aBuf( nLen+1 ); aBuf.setLength( nLen ); [pStr getCharacters: const_cast(aBuf.getStr())]; return aBuf.makeStringAndClear(); } CFStringRef CreateCFString( const rtl::OUString& rStr ) { return CFStringCreateWithCharacters(kCFAllocatorDefault, rStr.getStr(), rStr.getLength() ); } NSString* CreateNSString( const rtl::OUString& rStr ) { return [[NSString alloc] initWithCharacters: rStr.getStr() length: rStr.getLength()]; }