1 /////////////////////////////////////////////////////////////////////////////
 
   4 // Author:      David Elliott
 
   8 // Copyright:   (c) David Elliott
 
   9 // Licence:     wxWidgets licence
 
  10 /////////////////////////////////////////////////////////////////////////////
 
  12 #include "wx/wxprec.h"
 
  21 #include "wx/module.h"
 
  23 #include "wx/cocoa/ObjcPose.h"
 
  24 #include "wx/cocoa/autorelease.h"
 
  25 #include "wx/cocoa/mbarman.h"
 
  26 #include "wx/cocoa/NSApplication.h"
 
  28 #if wxUSE_WX_RESOURCES
 
  29 #  include "wx/resource.h"
 
  32 #import <AppKit/NSApplication.h>
 
  33 #import <Foundation/NSRunLoop.h>
 
  34 #import <Foundation/NSThread.h>
 
  35 #import <AppKit/NSEvent.h>
 
  36 #import <Foundation/NSString.h>
 
  37 #import <Foundation/NSNotification.h>
 
  38 #import <AppKit/NSCell.h>
 
  40 // ========================================================================
 
  41 // wxPoseAsInitializer
 
  42 // ========================================================================
 
  43 wxPoseAsInitializer *wxPoseAsInitializer::sm_first = NULL;
 
  45 static bool sg_needIdle = true;
 
  47 // ========================================================================
 
  48 // wxPoserNSApplication
 
  49 // ========================================================================
 
  50 @interface wxPoserNSApplication : NSApplication
 
  54 - (NSEvent *)nextEventMatchingMask:(unsigned int)mask untilDate:(NSDate *)expiration inMode:(NSString *)mode dequeue:(BOOL)flag;
 
  55 - (void)sendEvent: (NSEvent*)anEvent;
 
  56 @end // wxPoserNSApplication
 
  58 WX_IMPLEMENT_POSER(wxPoserNSApplication);
 
  60 @implementation wxPoserNSApplication : NSApplication
 
  62 /* NOTE: The old method of idle event handling added the handler using the
 
  63     [NSRunLoop -performSelector:target:argument:order:modes] which caused
 
  64     the invocation to occur at the begining of [NSApplication
 
  65     -nextEventMatchingMask:untilDate:expiration:inMode:dequeue:].  However,
 
  66     the code would be scheduled for invocation with every iteration of
 
  67     the event loop.  This new method simply overrides the method.  The
 
  68     same caveats apply.  In particular, by the time the event loop has
 
  69     called this method, it usually expects to receive an event.  If you
 
  70     plan on stopping the event loop, it is wise to send an event through
 
  71     the queue to ensure this method will return.
 
  72     See wxEventLoop::Exit() for more information.
 
  74     This overridden method calls the superclass method with an untilDate
 
  75     parameter that indicates nil should be returned if there are no pending
 
  76     events.  That is, nextEventMatchingMask: should not wait for an event.
 
  77     If nil is returned then idle event processing occurs until the user
 
  78     does not request anymore idle events or until a real event comes through.
 
  80     RN: Even though Apple documentation states that nil can be passed in place 
 
  81     of [NSDate distantPast] in the untilDate parameter, this causes Jaguar (10.2)
 
  82     to get stuck in some kind of loop deep within nextEventMatchingMask:, thus we 
 
  83     need to explicitly pass [NSDate distantPast] instead.
 
  86 - (NSEvent *)nextEventMatchingMask:(unsigned int)mask untilDate:(NSDate *)expiration inMode:(NSString *)mode dequeue:(BOOL)flag
 
  88     // Get the same events except don't block
 
  89     NSEvent *event = [super nextEventMatchingMask:mask untilDate:[NSDate distantPast] inMode:mode dequeue:flag];
 
  90     // If we got one, simply return it
 
  93     // No events, try doing some idle stuff
 
  96         && !wxTheApp->IsInAssert()
 
  98         && ([NSDefaultRunLoopMode isEqualToString:mode] || [NSModalPanelRunLoopMode isEqualToString:mode]))
 
 101         wxLogTrace(wxTRACE_COCOA,wxT("Processing idle events"));
 
 102         while(wxTheApp->ProcessIdle())
 
 104             // Get the same events except don't block
 
 105             NSEvent *event = [super nextEventMatchingMask:mask untilDate:[NSDate distantPast] inMode:mode dequeue:flag];
 
 106             // If we got one, simply return it
 
 109             // we didn't get one, do some idle work
 
 110             wxLogTrace(wxTRACE_COCOA,wxT("Looping idle events"));
 
 112         // No more idle work requested, block
 
 113         wxLogTrace(wxTRACE_COCOA,wxT("Finished idle processing"));
 
 116         wxLogTrace(wxTRACE_COCOA,wxT("Avoiding idle processing sg_needIdle=%d"),sg_needIdle);
 
 117     return [super nextEventMatchingMask:mask untilDate:expiration inMode:mode dequeue:flag];
 
 120 - (void)sendEvent: (NSEvent*)anEvent
 
 122     wxLogTrace(wxTRACE_COCOA,wxT("SendEvent"));
 
 124     [super sendEvent: anEvent];
 
 127 @end // wxPoserNSApplication
 
 129 // ========================================================================
 
 130 // wxNSApplicationDelegate
 
 131 // ========================================================================
 
 132 @implementation wxNSApplicationDelegate : NSObject
 
 134 // NOTE: Terminate means that the event loop does NOT return and thus
 
 135 // cleanup code doesn't properly execute.  Furthermore, wxWidgets has its
 
 136 // own exit on frame delete mechanism.
 
 137 - (BOOL)applicationShouldTerminateAfterLastWindowClosed:(NSApplication *)theApplication
 
 142 - (void)applicationWillBecomeActive:(NSNotification *)notification
 
 144     wxTheApp->CocoaDelegate_applicationWillBecomeActive();
 
 147 - (void)applicationDidBecomeActive:(NSNotification *)notification
 
 149     wxTheApp->CocoaDelegate_applicationDidBecomeActive();
 
 152 - (void)applicationWillResignActive:(NSNotification *)notification
 
 154     wxTheApp->CocoaDelegate_applicationWillResignActive();
 
 157 - (void)applicationDidResignActive:(NSNotification *)notification
 
 159     wxTheApp->CocoaDelegate_applicationDidResignActive();
 
 162 - (void)controlTintChanged:(NSNotification *)notification
 
 164     wxLogDebug(wxT("TODO: send EVT_SYS_COLOUR_CHANGED as appropriate"));
 
 167 @end // implementation wxNSApplicationDelegate : NSObject
 
 169 // ========================================================================
 
 171 // ========================================================================
 
 173 // ----------------------------------------------------------------------------
 
 174 // wxApp Static member initialization
 
 175 // ----------------------------------------------------------------------------
 
 176 IMPLEMENT_DYNAMIC_CLASS(wxApp, wxEvtHandler)
 
 177 BEGIN_EVENT_TABLE(wxApp, wxEvtHandler)
 
 178     EVT_IDLE(wxAppBase::OnIdle)
 
 179 //    EVT_END_SESSION(wxApp::OnEndSession)
 
 180 //    EVT_QUERY_END_SESSION(wxApp::OnQueryEndSession)
 
 183 // ----------------------------------------------------------------------------
 
 184 // wxApp initialization/cleanup
 
 185 // ----------------------------------------------------------------------------
 
 186 bool wxApp::Initialize(int& argc, wxChar **argv)
 
 188     wxAutoNSAutoreleasePool pool;
 
 189     m_cocoaMainThread = [NSThread currentThread];
 
 190     // Mac OS X passes a process serial number command line argument when
 
 191     // the application is launched from the Finder. This argument must be
 
 192     // removed from the command line arguments before being handled by the
 
 193     // application (otherwise applications would need to handle it)
 
 196         static const wxChar *ARG_PSN = _T("-psn_");
 
 197         if ( wxStrncmp(argv[1], ARG_PSN, wxStrlen(ARG_PSN)) == 0 )
 
 199             // remove this argument
 
 201             memmove(argv + 1, argv + 2, argc * sizeof(wxChar *));
 
 205     // Posing must be completed before any instances of the Objective-C
 
 206     // classes being posed as are created.
 
 207     wxPoseAsInitializer::InitializePosers();
 
 209     return wxAppBase::Initialize(argc, argv);
 
 212 void wxApp::CleanUp()
 
 214     wxAutoNSAutoreleasePool pool;
 
 216     wxDC::CocoaShutdownTextSystem();
 
 217     wxMenuBarManager::DestroyInstance();
 
 219     [m_cocoaApp setDelegate:nil];
 
 220     [[NSNotificationCenter defaultCenter] removeObserver:m_cocoaAppDelegate
 
 221         name:NSControlTintDidChangeNotification object:nil];
 
 222     [m_cocoaAppDelegate release];
 
 223     m_cocoaAppDelegate = NULL;
 
 225     wxAppBase::CleanUp();
 
 228 // ----------------------------------------------------------------------------
 
 230 // ----------------------------------------------------------------------------
 
 235 #if WXWIN_COMPATIBILITY_2_2
 
 236     m_wantDebugOutput = TRUE;
 
 239     m_isInAssert = FALSE;
 
 240 #endif // __WXDEBUG__
 
 245     m_cocoaAppDelegate = NULL;
 
 248 void wxApp::CocoaDelegate_applicationWillBecomeActive()
 
 252 void wxApp::CocoaDelegate_applicationDidBecomeActive()
 
 256 void wxApp::CocoaDelegate_applicationWillResignActive()
 
 258     wxTopLevelWindowCocoa::DeactivatePendingWindow();
 
 261 void wxApp::CocoaDelegate_applicationDidResignActive()
 
 265 bool wxApp::OnInitGui()
 
 267     wxAutoNSAutoreleasePool pool;
 
 268     if(!wxAppBase::OnInitGui())
 
 271     // Create the app using the sharedApplication method
 
 272     m_cocoaApp = [NSApplication sharedApplication];
 
 273     m_cocoaAppDelegate = [[wxNSApplicationDelegate alloc] init];
 
 274     [m_cocoaApp setDelegate:m_cocoaAppDelegate];
 
 275     [[NSNotificationCenter defaultCenter] addObserver:m_cocoaAppDelegate
 
 276         selector:@selector(controlTintChanged:)
 
 277         name:NSControlTintDidChangeNotification object:nil];
 
 279     wxMenuBarManager::CreateInstance();
 
 281     wxDC::CocoaInitializeTextSystem();
 
 285 bool wxApp::CallOnInit()
 
 287 //    wxAutoNSAutoreleasePool pool;
 
 293     if(!wxAppBase::OnInit())
 
 303     wxAppConsole::Exit();
 
 306 // Yield to other processes
 
 307 bool wxApp::Yield(bool onlyIfNeeded)
 
 310     static bool s_inYield = false;
 
 313     // disable log flushing from here because a call to wxYield() shouldn't
 
 314     // normally result in message boxes popping up &c
 
 322             wxFAIL_MSG( wxT("wxYield called recursively" ) );
 
 330     // Run the event loop until it is out of events
 
 333         wxAutoNSAutoreleasePool pool;
 
 334         NSEvent *event = [GetNSApplication()
 
 335                 nextEventMatchingMask:NSAnyEventMask
 
 336                 untilDate:[NSDate distantPast]
 
 337                 inMode:NSDefaultRunLoopMode
 
 341         [GetNSApplication() sendEvent: event];
 
 345     // let the logs be flashed again
 
 354 void wxApp::WakeUpIdle()
 
 356     [m_cocoaApp postEvent:[NSEvent otherEventWithType:NSApplicationDefined
 
 357             location:NSZeroPoint modifierFlags:NSAnyEventMask
 
 358             timestamp:0 windowNumber:0 context:nil
 
 359             subtype:0 data1:0 data2:0] atStart:NO];
 
 363 void wxApp::OnAssert(const wxChar *file, int line, const wxChar* cond, const wxChar *msg)
 
 366     wxAppBase::OnAssert(file, line, cond, msg);
 
 367     m_isInAssert = FALSE;
 
 369 #endif // __WXDEBUG__