1 /////////////////////////////////////////////////////////////////////////////
4 // Author: David Elliott
8 // Copyright: (c) David Elliott
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 #include "wx/wxprec.h"
17 #include "wx/dialog.h"
23 #include "wx/module.h"
25 #include "wx/cocoa/ObjcPose.h"
26 #include "wx/cocoa/autorelease.h"
27 #include "wx/cocoa/mbarman.h"
28 #include "wx/cocoa/NSApplication.h"
30 #if wxUSE_WX_RESOURCES
31 # include "wx/resource.h"
34 #import <AppKit/NSApplication.h>
35 #import <Foundation/NSRunLoop.h>
36 #import <Foundation/NSArray.h>
37 #import <Foundation/NSAutoreleasePool.h>
38 #import <Foundation/NSThread.h>
39 #import <AppKit/NSEvent.h>
40 #import <Foundation/NSString.h>
42 // ========================================================================
43 // wxPoseAsInitializer
44 // ========================================================================
45 wxPoseAsInitializer *wxPoseAsInitializer::sm_first = NULL;
47 static bool sg_needIdle = true;
49 // ========================================================================
50 // wxPoserNSApplication
51 // ========================================================================
52 @interface wxPoserNSApplication : NSApplication
56 - (NSEvent *)nextEventMatchingMask:(unsigned int)mask untilDate:(NSDate *)expiration inMode:(NSString *)mode dequeue:(BOOL)flag;
57 - (void)sendEvent: (NSEvent*)anEvent;
58 @end // wxPoserNSApplication
60 WX_IMPLEMENT_POSER(wxPoserNSApplication);
62 @implementation wxPoserNSApplication : NSApplication
64 /* NOTE: The old method of idle event handling added the handler using the
65 [NSRunLoop -performSelector:target:argument:order:modes] which caused
66 the invocation to occur at the begining of [NSApplication
67 -nextEventMatchingMask:untilDate:expiration:inMode:dequeue:]. However,
68 the code would be scheduled for invocation with every iteration of
69 the event loop. This new method simply overrides the method. The
70 same caveats apply. In particular, by the time the event loop has
71 called this method, it usually expects to receive an event. If you
72 plan on stopping the event loop, it is wise to send an event through
73 the queue to ensure this method will return.
74 See wxEventLoop::Exit() for more information.
77 - (NSEvent *)nextEventMatchingMask:(unsigned int)mask untilDate:(NSDate *)expiration inMode:(NSString *)mode dequeue:(BOOL)flag
79 // Get the same events except don't block
80 NSEvent *event = [super nextEventMatchingMask:mask untilDate:nil/* equivalent to [NSDate distantPast] */ inMode:mode dequeue:flag];
81 // If we got one, simply return it
84 // No events, try doing some idle stuff
87 && !wxTheApp->IsInAssert()
89 && ([NSDefaultRunLoopMode isEqualToString:mode] || [NSModalPanelRunLoopMode isEqualToString:mode]))
92 wxLogDebug(wxT("Processing idle events"));
93 while(wxTheApp->ProcessIdle())
95 // Get the same events except don't block
96 NSEvent *event = [super nextEventMatchingMask:mask untilDate:nil/* equivalent to [NSDate distantPast] */ inMode:mode dequeue:flag];
97 // If we got one, simply return it
100 // we didn't get one, do some idle work
101 wxLogDebug(wxT("Looping idle events"));
103 // No more idle work requested, block
104 wxLogDebug(wxT("Finished idle processing"));
107 wxLogDebug(wxT("Avoiding idle processing sg_needIdle=%d"),sg_needIdle);
108 return [super nextEventMatchingMask:mask untilDate:expiration inMode:mode dequeue:flag];
111 - (void)sendEvent: (NSEvent*)anEvent
113 wxLogDebug(wxT("SendEvent"));
115 [super sendEvent: anEvent];
118 @end // wxPoserNSApplication
120 // ========================================================================
121 // wxNSApplicationDelegate
122 // ========================================================================
123 @implementation wxNSApplicationDelegate : NSObject
125 // NOTE: Terminate means that the event loop does NOT return and thus
126 // cleanup code doesn't properly execute. Furthermore, wxWindows has its
127 // own exit on frame delete mechanism.
128 - (BOOL)applicationShouldTerminateAfterLastWindowClosed:(NSApplication *)theApplication
133 - (void)applicationWillBecomeActive:(NSNotification *)notification
135 wxTheApp->CocoaDelegate_applicationWillBecomeActive();
138 - (void)applicationDidBecomeActive:(NSNotification *)notification
140 wxTheApp->CocoaDelegate_applicationDidBecomeActive();
143 - (void)applicationWillResignActive:(NSNotification *)notification
145 wxTheApp->CocoaDelegate_applicationWillResignActive();
148 - (void)applicationDidResignActive:(NSNotification *)notification
150 wxTheApp->CocoaDelegate_applicationDidResignActive();
153 @end // implementation wxNSApplicationDelegate : NSObject
155 // ========================================================================
157 // ========================================================================
159 // ----------------------------------------------------------------------------
160 // wxApp Static member initialization
161 // ----------------------------------------------------------------------------
162 IMPLEMENT_DYNAMIC_CLASS(wxApp, wxEvtHandler)
163 BEGIN_EVENT_TABLE(wxApp, wxEvtHandler)
164 EVT_IDLE(wxAppBase::OnIdle)
165 // EVT_END_SESSION(wxApp::OnEndSession)
166 // EVT_QUERY_END_SESSION(wxApp::OnQueryEndSession)
169 // ----------------------------------------------------------------------------
170 // wxApp initialization/cleanup
171 // ----------------------------------------------------------------------------
172 bool wxApp::Initialize(int& argc, wxChar **argv)
174 wxAutoNSAutoreleasePool pool;
175 m_cocoaMainThread = [NSThread currentThread];
176 // Mac OS X passes a process serial number command line argument when
177 // the application is launched from the Finder. This argument must be
178 // removed from the command line arguments before being handled by the
179 // application (otherwise applications would need to handle it)
182 static const wxChar *ARG_PSN = _T("-psn_");
183 if ( wxStrncmp(argv[1], ARG_PSN, wxStrlen(ARG_PSN)) == 0 )
185 // remove this argument
187 memmove(argv + 1, argv + 2, argc * sizeof(wxChar *));
191 // Posing must be completed before any instances of the Objective-C
192 // classes being posed as are created.
193 wxPoseAsInitializer::InitializePosers();
195 return wxAppBase::Initialize(argc, argv);
198 void wxApp::CleanUp()
200 wxAutoNSAutoreleasePool pool;
202 wxDC::CocoaShutdownTextSystem();
203 wxMenuBarManager::DestroyInstance();
205 [m_cocoaApp setDelegate:nil];
206 [m_cocoaAppDelegate release];
207 m_cocoaAppDelegate = NULL;
209 wxAppBase::CleanUp();
212 // ----------------------------------------------------------------------------
214 // ----------------------------------------------------------------------------
219 #if WXWIN_COMPATIBILITY_2_2
220 m_wantDebugOutput = TRUE;
223 m_isInAssert = FALSE;
224 #endif // __WXDEBUG__
229 m_cocoaAppDelegate = NULL;
232 void wxApp::CocoaDelegate_applicationWillBecomeActive()
236 void wxApp::CocoaDelegate_applicationDidBecomeActive()
240 void wxApp::CocoaDelegate_applicationWillResignActive()
242 wxTopLevelWindowCocoa::DeactivatePendingWindow();
245 void wxApp::CocoaDelegate_applicationDidResignActive()
249 bool wxApp::OnInitGui()
251 wxAutoNSAutoreleasePool pool;
252 if(!wxAppBase::OnInitGui())
255 // Create the app using the sharedApplication method
256 m_cocoaApp = [NSApplication sharedApplication];
257 m_cocoaAppDelegate = [[wxNSApplicationDelegate alloc] init];
258 [m_cocoaApp setDelegate:m_cocoaAppDelegate];
260 wxMenuBarManager::CreateInstance();
262 wxDC::CocoaInitializeTextSystem();
263 // [ m_cocoaApp setDelegate:m_cocoaApp ];
267 bool wxApp::CallOnInit()
269 // wxAutoNSAutoreleasePool pool;
275 if(!wxAppBase::OnInit())
285 wxAppConsole::Exit();
288 // Yield to other processes
289 bool wxApp::Yield(bool onlyIfNeeded)
292 static bool s_inYield = false;
295 // disable log flushing from here because a call to wxYield() shouldn't
296 // normally result in message boxes popping up &c
304 wxFAIL_MSG( wxT("wxYield called recursively" ) );
312 // Run the event loop until it is out of events
313 while(NSEvent *event = [GetNSApplication()
314 nextEventMatchingMask:NSAnyEventMask
315 untilDate:[NSDate distantPast]
316 inMode:NSDefaultRunLoopMode
319 [GetNSApplication() sendEvent: event];
323 // let the logs be flashed again
332 void wxApp::WakeUpIdle()
334 [m_cocoaApp postEvent:[NSEvent otherEventWithType:NSApplicationDefined
335 location:NSZeroPoint modifierFlags:NSAnyEventMask
336 timestamp:0 windowNumber:0 context:nil
337 subtype:0 data1:0 data2:0] atStart:NO];
341 void wxApp::OnAssert(const wxChar *file, int line, const wxChar* cond, const wxChar *msg)
344 wxAppBase::OnAssert(file, line, cond, msg);
345 m_isInAssert = FALSE;
347 #endif // __WXDEBUG__