]> git.saurik.com Git - wxWidgets.git/blame - src/osx/cocoa/nonownedwnd.mm
supporting nested window disablers on the same window
[wxWidgets.git] / src / osx / cocoa / nonownedwnd.mm
CommitLineData
33e90275 1/////////////////////////////////////////////////////////////////////////////
0f9b48d1 2// Name: src/osx/cocoa/nonownedwnd.mm
33e90275
SC
3// Purpose: non owned window for cocoa
4// Author: DavidStefan Csomor
5// Modified by:
6// Created: 2008-06-20
a9a4f229 7// RCS-ID: $Id$
33e90275
SC
8// Copyright: (c) Stefan Csomor
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12#include "wx/wxprec.h"
05cf95ed
SC
13#ifndef WX_PRECOMP
14 #include "wx/nonownedwnd.h"
15 #include "wx/frame.h"
697dce56 16 #include "wx/app.h"
40c4350f 17 #include "wx/dialog.h"
d814b237
SC
18 #include "wx/menuitem.h"
19 #include "wx/menu.h"
33e90275
SC
20#endif
21
33e90275 22#include "wx/osx/private.h"
33e90275 23
5d57348e
SC
24NSScreen* wxOSXGetMenuScreen()
25{
26 if ( [NSScreen screens] == nil )
27 return [NSScreen mainScreen];
28 else
29 {
30 return [[NSScreen screens] objectAtIndex:0];
31 }
32}
33
33e90275
SC
34NSRect wxToNSRect( NSView* parent, const wxRect& r )
35{
5d57348e 36 NSRect frame = parent ? [parent bounds] : [wxOSXGetMenuScreen() frame];
33e90275
SC
37 int y = r.y;
38 int x = r.x ;
94929e7b 39 if ( parent == NULL || ![ parent isFlipped ] )
e490b0d2 40 y = (int)(frame.size.height - ( r.y + r.height ));
33e90275
SC
41 return NSMakeRect(x, y, r.width , r.height);
42}
43
44wxRect wxFromNSRect( NSView* parent, const NSRect& rect )
45{
5d57348e 46 NSRect frame = parent ? [parent bounds] : [wxOSXGetMenuScreen() frame];
e490b0d2
VZ
47 int y = (int)rect.origin.y;
48 int x = (int)rect.origin.x;
94929e7b 49 if ( parent == NULL || ![ parent isFlipped ] )
e490b0d2
VZ
50 y = (int)(frame.size.height - (rect.origin.y + rect.size.height));
51 return wxRect( x, y, (int)rect.size.width, (int)rect.size.height );
33e90275
SC
52}
53
54NSPoint wxToNSPoint( NSView* parent, const wxPoint& p )
55{
5d57348e 56 NSRect frame = parent ? [parent bounds] : [wxOSXGetMenuScreen() frame];
33e90275
SC
57 int x = p.x ;
58 int y = p.y;
94929e7b 59 if ( parent == NULL || ![ parent isFlipped ] )
e490b0d2 60 y = (int)(frame.size.height - ( p.y ));
33e90275
SC
61 return NSMakePoint(x, y);
62}
63
64wxPoint wxFromNSPoint( NSView* parent, const NSPoint& p )
65{
5d57348e 66 NSRect frame = parent ? [parent bounds] : [wxOSXGetMenuScreen() frame];
e490b0d2
VZ
67 int x = (int)p.x;
68 int y = (int)p.y;
94929e7b 69 if ( parent == NULL || ![ parent isFlipped ] )
e490b0d2 70 y = (int)(frame.size.height - ( p.y ));
33e90275
SC
71 return wxPoint( x, y);
72}
73
46f573dc
KO
74bool shouldHandleSelector(SEL selector)
75{
03647350 76 if (selector == @selector(noop:)
46f573dc
KO
77 || selector == @selector(complete:)
78 || selector == @selector(deleteBackward:)
79 || selector == @selector(deleteForward:)
80 || selector == @selector(insertNewline:)
81 || selector == @selector(insertTab:)
a46cbd24 82 || selector == @selector(insertBacktab:)
46f573dc
KO
83 || selector == @selector(keyDown:)
84 || selector == @selector(keyUp:)
85 || selector == @selector(scrollPageUp:)
86 || selector == @selector(scrollPageDown:)
87 || selector == @selector(scrollToBeginningOfDocument:)
88 || selector == @selector(scrollToEndOfDocument:))
89 return false;
03647350 90
46f573dc
KO
91 return true;
92
93}
94
dbeddfb9 95//
e17ac396 96// wx category for NSWindow (our own and wrapped instances)
dbeddfb9
SC
97//
98
e17ac396
SC
99@interface NSWindow (wxNSWindowSupport)
100
101- (wxNonOwnedWindowCocoaImpl*) WX_implementation;
102
103- (bool) WX_filterSendEvent:(NSEvent *) event;
104
105@end
106
107@implementation NSWindow (wxNSWindowSupport)
108
109- (wxNonOwnedWindowCocoaImpl*) WX_implementation
110{
111 return (wxNonOwnedWindowCocoaImpl*) wxNonOwnedWindowImpl::FindFromWXWindow( self );
112}
113
114// TODO in cocoa everything during a drag is sent to the NSWindow the mouse down occured,
115// this does not conform to the wx behaviour if the window is not captured, so try to resend
116// or capture all wx mouse event handling at the tlw as we did for carbon
117
118- (bool) WX_filterSendEvent:(NSEvent *) event
119{
120 bool handled = false;
121 if ( ([event type] >= NSLeftMouseDown) && ([event type] <= NSMouseExited) )
122 {
123 wxWindow* cw = wxWindow::GetCapture();
124 if ( cw != NULL )
125 {
126 ((wxWidgetCocoaImpl*)cw->GetPeer())->DoHandleMouseEvent( event);
127 handled = true;
128 }
129 }
130 return handled;
131}
132@end
133
134//
135// wx native implementation
136//
9130dfd7 137
dbeddfb9 138@interface wxNSWindow : NSWindow
dbeddfb9 139{
dbeddfb9
SC
140}
141
e17ac396 142- (void) sendEvent:(NSEvent *)event;
0bcac6c7 143- (NSRect)constrainFrameRect:(NSRect)frameRect toScreen:(NSScreen *)screen;
9130dfd7 144- (void)noResponderFor: (SEL) selector;
dbeddfb9
SC
145@end
146
147@implementation wxNSWindow
148
e17ac396
SC
149- (void)sendEvent:(NSEvent *) event
150{
151 if ( ![self WX_filterSendEvent: event] )
0b6f851f
SC
152 {
153 WXEVENTREF formerEvent = wxTheApp == NULL ? NULL : wxTheApp->MacGetCurrentEvent();
154 WXEVENTHANDLERCALLREF formerHandler = wxTheApp == NULL ? NULL : wxTheApp->MacGetCurrentEventHandlerCallRef();
155
156 if (wxTheApp)
157 wxTheApp->MacSetCurrentEvent(event, NULL);
158
e17ac396 159 [super sendEvent: event];
0b6f851f
SC
160
161 if (wxTheApp)
162 wxTheApp->MacSetCurrentEvent(formerEvent , formerHandler);
163 }
e17ac396
SC
164}
165
0bcac6c7
KO
166// The default implementation always moves the window back onto the screen,
167// even when the programmer explicitly wants to hide it.
168- (NSRect)constrainFrameRect:(NSRect)frameRect toScreen:(NSScreen *)screen
169{
219665bb 170 wxUnusedVar(screen);
0bcac6c7
KO
171 return frameRect;
172}
173
46f573dc
KO
174- (void)doCommandBySelector:(SEL)selector
175{
03647350 176 if (shouldHandleSelector(selector) &&
46f573dc
KO
177 !(selector == @selector(cancel:) || selector == @selector(cancelOperation:)) )
178 [super doCommandBySelector:selector];
179}
180
181
9130dfd7
KO
182// NB: if we don't do this, all key downs that get handled lead to a NSBeep
183- (void)noResponderFor: (SEL) selector
184{
46f573dc 185 if (selector != @selector(keyDown:) && selector != @selector(keyUp:))
9130dfd7 186 {
20ede392 187 [super noResponderFor:selector];
9130dfd7
KO
188 }
189}
dbeddfb9 190
2c3ce6c4
KO
191// We need this for borderless windows, i.e. shaped windows or windows without
192// a title bar. For more info, see:
193// http://lists.apple.com/archives/cocoa-dev/2008/May/msg02091.html
194- (BOOL)canBecomeKeyWindow
195{
196 return YES;
197}
198
dbeddfb9
SC
199@end
200
bb839504 201@interface wxNSPanel : NSPanel
dbeddfb9
SC
202{
203}
204
247bb510 205- (NSRect)constrainFrameRect:(NSRect)frameRect toScreen:(NSScreen *)screen;
9130dfd7 206- (void)noResponderFor: (SEL) selector;
e17ac396 207- (void)sendEvent:(NSEvent *)event;
dbeddfb9
SC
208@end
209
03647350 210@implementation wxNSPanel
dbeddfb9 211
247bb510
VZ
212- (NSRect)constrainFrameRect:(NSRect)frameRect toScreen:(NSScreen *)screen
213{
214 wxNonOwnedWindowCocoaImpl* impl = (wxNonOwnedWindowCocoaImpl*) wxNonOwnedWindowImpl::FindFromWXWindow( self );
215 if (impl && impl->IsFullScreen())
216 return frameRect;
217 else
218 return [super constrainFrameRect:frameRect toScreen:screen];
219}
220
09a9eb20
KO
221- (BOOL)canBecomeKeyWindow
222{
223 return YES;
224}
225
46f573dc
KO
226- (void)doCommandBySelector:(SEL)selector
227{
228 if (shouldHandleSelector(selector))
229 [super doCommandBySelector:selector];
230}
231
232// NB: if we don't do this, it seems that all events that end here lead to a NSBeep
9130dfd7
KO
233- (void)noResponderFor: (SEL) selector
234{
46f573dc 235 if (selector != @selector(keyDown:) && selector != @selector(keyUp:))
9130dfd7 236 {
20ede392 237 [super noResponderFor:selector];
9130dfd7
KO
238 }
239}
240
e17ac396
SC
241- (void)sendEvent:(NSEvent *) event
242{
243 if ( ![self WX_filterSendEvent: event] )
244 [super sendEvent: event];
245}
246
dbeddfb9
SC
247@end
248
249
250//
251// controller
252//
253
c8fdb345 254@interface wxNonOwnedWindowController : NSObject wxOSX_10_6_AND_LATER(<NSWindowDelegate>)
dbeddfb9
SC
255{
256}
257
258- (void)windowDidResize:(NSNotification *)notification;
259- (NSSize)windowWillResize:(NSWindow *)window toSize:(NSSize)proposedFrameSize;
09a9eb20
KO
260- (void)windowDidResignKey:(NSNotification *)notification;
261- (void)windowDidBecomeKey:(NSNotification *)notification;
dbeddfb9
SC
262- (void)windowDidMove:(NSNotification *)notification;
263- (BOOL)windowShouldClose:(id)window;
1232607d 264- (BOOL)windowShouldZoom:(NSWindow *)window toFrame:(NSRect)newFrame;
dbeddfb9
SC
265
266@end
267
fbbe829a
SC
268extern int wxOSXGetIdFromSelector(SEL action );
269
dbeddfb9
SC
270@implementation wxNonOwnedWindowController
271
272- (id) init
273{
274 [super init];
275 return self;
276}
277
fbbe829a
SC
278- (BOOL) triggerMenu:(SEL) action
279{
280 wxMenuBar* mbar = wxMenuBar::MacGetInstalledMenuBar();
281 if ( mbar )
282 {
283 wxMenu* menu = NULL;
284 wxMenuItem* menuitem = mbar->FindItem(wxOSXGetIdFromSelector(action), &menu);
285 if ( menu != NULL && menuitem != NULL)
286 return menu->HandleCommandProcess(menuitem);
287 }
288 return NO;
289}
290
291- (BOOL)validateMenuItem:(NSMenuItem *)menuItem
292{
293 SEL action = [menuItem action];
294
295 wxMenuBar* mbar = wxMenuBar::MacGetInstalledMenuBar();
296 if ( mbar )
297 {
298 wxMenu* menu = NULL;
299 wxMenuItem* menuitem = mbar->FindItem(wxOSXGetIdFromSelector(action), &menu);
300 if ( menu != NULL && menuitem != NULL)
301 {
40a35c1f
SC
302 menu->HandleCommandUpdateStatus(menuitem);
303 return menuitem->IsEnabled();
fbbe829a
SC
304 }
305 }
306 return YES;
307}
308
309- (void)undo:(id)sender
310{
e7794cf2 311 wxUnusedVar(sender);
fbbe829a
SC
312 [self triggerMenu:_cmd];
313}
314
315- (void)redo:(id)sender
316{
e7794cf2 317 wxUnusedVar(sender);
fbbe829a
SC
318 [self triggerMenu:_cmd];
319}
320
321- (void)cut:(id)sender
322{
e7794cf2 323 wxUnusedVar(sender);
fbbe829a
SC
324 [self triggerMenu:_cmd];
325}
326
327- (void)copy:(id)sender
328{
e7794cf2 329 wxUnusedVar(sender);
fbbe829a
SC
330 [self triggerMenu:_cmd];
331}
332
333- (void)paste:(id)sender
334{
e7794cf2 335 wxUnusedVar(sender);
fbbe829a
SC
336 [self triggerMenu:_cmd];
337}
338
339- (void)delete:(id)sender
340{
e7794cf2 341 wxUnusedVar(sender);
fbbe829a
SC
342 [self triggerMenu:_cmd];
343}
344
345- (void)selectAll:(id)sender
346{
e7794cf2 347 wxUnusedVar(sender);
fbbe829a
SC
348 [self triggerMenu:_cmd];
349}
350
dbeddfb9
SC
351- (BOOL)windowShouldClose:(id)nwindow
352{
e17ac396 353 wxNonOwnedWindowCocoaImpl* windowimpl = [(NSWindow*) nwindow WX_implementation];
dbeddfb9
SC
354 if ( windowimpl )
355 {
356 wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
357 if ( wxpeer )
358 wxpeer->Close();
359 }
360 return NO;
361}
362
e17ac396 363- (NSSize)windowWillResize:(NSWindow *)window
dbeddfb9
SC
364 toSize:(NSSize)proposedFrameSize
365{
e17ac396 366 NSRect frame = [window frame];
5da125c7 367 wxRect wxframe = wxFromNSRect( NULL, frame );
e490b0d2
VZ
368 wxframe.SetWidth( (int)proposedFrameSize.width );
369 wxframe.SetHeight( (int)proposedFrameSize.height );
e17ac396
SC
370
371 wxNonOwnedWindowCocoaImpl* windowimpl = [window WX_implementation];
5da125c7
SC
372 if ( windowimpl )
373 {
374 wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
375 if ( wxpeer )
376 {
377 wxpeer->HandleResizing( 0, &wxframe );
378 NSSize newSize = NSMakeSize(wxframe.GetWidth(), wxframe.GetHeight());
379 return newSize;
380 }
381 }
382
dbeddfb9
SC
383 return proposedFrameSize;
384}
385
386- (void)windowDidResize:(NSNotification *)notification
387{
e17ac396
SC
388 NSWindow* window = (NSWindow*) [notification object];
389 wxNonOwnedWindowCocoaImpl* windowimpl = [window WX_implementation];
dbeddfb9
SC
390 if ( windowimpl )
391 {
392 wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
393 if ( wxpeer )
394 wxpeer->HandleResized(0);
395 }
396}
397
398- (void)windowDidMove:(NSNotification *)notification
399{
400 wxNSWindow* window = (wxNSWindow*) [notification object];
e17ac396 401 wxNonOwnedWindowCocoaImpl* windowimpl = [window WX_implementation];
dbeddfb9
SC
402 if ( windowimpl )
403 {
404 wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
405 if ( wxpeer )
406 wxpeer->HandleMoved(0);
407 }
408}
409
09a9eb20 410- (void)windowDidBecomeKey:(NSNotification *)notification
dbeddfb9 411{
e17ac396
SC
412 NSWindow* window = (NSWindow*) [notification object];
413 wxNonOwnedWindowCocoaImpl* windowimpl = [window WX_implementation];
dbeddfb9
SC
414 if ( windowimpl )
415 {
416 wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
417 if ( wxpeer )
418 wxpeer->HandleActivated(0, true);
419 }
420}
421
09a9eb20 422- (void)windowDidResignKey:(NSNotification *)notification
dbeddfb9 423{
e17ac396
SC
424 NSWindow* window = (NSWindow*) [notification object];
425 wxNonOwnedWindowCocoaImpl* windowimpl = [window WX_implementation];
dbeddfb9
SC
426 if ( windowimpl )
427 {
428 wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
429 if ( wxpeer )
09a9eb20 430 {
dbeddfb9 431 wxpeer->HandleActivated(0, false);
03647350
VZ
432 // Needed for popup window since the firstResponder
433 // (focus in wx) doesn't change when this
09a9eb20
KO
434 // TLW becomes inactive.
435 wxFocusEvent event( wxEVT_KILL_FOCUS, wxpeer->GetId());
436 event.SetEventObject(wxpeer);
437 wxpeer->HandleWindowEvent(event);
438 }
dbeddfb9
SC
439 }
440}
441
7cb2a241
SC
442- (id)windowWillReturnFieldEditor:(NSWindow *)sender toObject:(id)anObject
443{
444 wxUnusedVar(sender);
445
446 if ([anObject isKindOfClass:[wxNSTextField class]])
447 {
448 wxNSTextField* tf = (wxNSTextField*) anObject;
449 wxNSTextFieldEditor* editor = [tf fieldEditor];
450 if ( editor == nil )
451 {
452 editor = [[wxNSTextFieldEditor alloc] init];
453 [editor setFieldEditor:YES];
454 [tf setFieldEditor:editor];
455 }
456 return editor;
457 }
458
459 return nil;
460}
461
1232607d
KO
462- (BOOL)windowShouldZoom:(NSWindow *)window toFrame:(NSRect)newFrame
463{
e17ac396
SC
464 wxUnusedVar(newFrame);
465 wxNonOwnedWindowCocoaImpl* windowimpl = [window WX_implementation];
1232607d
KO
466 if ( windowimpl )
467 {
468 wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
469 wxMaximizeEvent event(wxpeer->GetId());
470 event.SetEventObject(wxpeer);
471 return !wxpeer->HandleWindowEvent(event);
472 }
473 return true;
474}
475
dbeddfb9
SC
476@end
477
33e90275
SC
478IMPLEMENT_DYNAMIC_CLASS( wxNonOwnedWindowCocoaImpl , wxNonOwnedWindowImpl )
479
03647350 480wxNonOwnedWindowCocoaImpl::wxNonOwnedWindowCocoaImpl( wxNonOwnedWindow* nonownedwnd) :
33e90275
SC
481 wxNonOwnedWindowImpl(nonownedwnd)
482{
483 m_macWindow = NULL;
484 m_macFullScreenData = NULL;
485}
03647350
VZ
486
487wxNonOwnedWindowCocoaImpl::wxNonOwnedWindowCocoaImpl()
33e90275
SC
488{
489 m_macWindow = NULL;
490 m_macFullScreenData = NULL;
491}
03647350 492
33e90275
SC
493wxNonOwnedWindowCocoaImpl::~wxNonOwnedWindowCocoaImpl()
494{
17e2694c
SC
495 if ( !m_wxPeer->IsNativeWindowWrapper() )
496 {
17e2694c
SC
497 [m_macWindow setDelegate:nil];
498 [m_macWindow release];
499 }
33e90275
SC
500}
501
0aaa6ace 502void wxNonOwnedWindowCocoaImpl::WillBeDestroyed()
33e90275 503{
17e2694c
SC
504 if ( !m_wxPeer->IsNativeWindowWrapper() )
505 {
506 [m_macWindow setDelegate:nil];
507 }
33e90275
SC
508}
509
40c4350f 510void wxNonOwnedWindowCocoaImpl::Create( wxWindow* parent, const wxPoint& pos, const wxSize& size,
20ede392 511long style, long extraStyle, const wxString& WXUNUSED(name) )
33e90275 512{
dbeddfb9 513 static wxNonOwnedWindowController* controller = NULL;
03647350 514
dbeddfb9
SC
515 if ( !controller )
516 controller =[[wxNonOwnedWindowController alloc] init];
517
518
33e90275 519 int windowstyle = NSBorderlessWindowMask;
03647350
VZ
520
521 if ( style & wxFRAME_TOOL_WINDOW || ( style & wxPOPUP_WINDOW ) ||
09a9eb20
KO
522 GetWXPeer()->GetExtraStyle() & wxTOPLEVEL_EX_DIALOG )
523 {
dbeddfb9 524 m_macWindow = [wxNSPanel alloc];
09a9eb20 525 }
33e90275 526 else
dbeddfb9 527 m_macWindow = [wxNSWindow alloc];
03647350 528
e912ebe3 529 CGWindowLevel level = kCGNormalWindowLevel;
03647350 530
33e90275
SC
531 if ( style & wxFRAME_TOOL_WINDOW )
532 {
aa960026
KO
533 windowstyle |= NSUtilityWindowMask;
534 if ( ( style & wxMINIMIZE_BOX ) || ( style & wxMAXIMIZE_BOX ) ||
535 ( style & wxCLOSE_BOX ) || ( style & wxSYSTEM_MENU ) )
536 {
537 windowstyle |= NSTitledWindowMask ;
538 }
33e90275
SC
539 }
540 else if ( ( style & wxPOPUP_WINDOW ) )
541 {
542 level = kCGPopUpMenuWindowLevel;
543 /*
544 if ( ( style & wxBORDER_NONE ) )
545 {
546 wclass = kHelpWindowClass ; // has no border
547 attr |= kWindowNoShadowAttribute;
548 }
549 else
550 {
551 wclass = kPlainWindowClass ; // has a single line border, it will have to do for now
552 }
553 */
554 }
555 else if ( ( style & wxCAPTION ) )
556 {
557 windowstyle |= NSTitledWindowMask ;
558 }
559 else if ( ( style & wxFRAME_DRAWER ) )
560 {
561 /*
562 wclass = kDrawerWindowClass;
563 */
564 }
565 else
566 {
567 // set these even if we have no title, otherwise the controls won't be visible
568 if ( ( style & wxMINIMIZE_BOX ) || ( style & wxMAXIMIZE_BOX ) ||
569 ( style & wxCLOSE_BOX ) || ( style & wxSYSTEM_MENU ) )
570 {
571 windowstyle |= NSTitledWindowMask ;
572 }
573 /*
574 else if ( ( style & wxNO_BORDER ) )
575 {
576 wclass = kSimpleWindowClass ;
577 }
578 else
579 {
580 wclass = kPlainWindowClass ;
581 }
582 */
583 }
584
585 if ( windowstyle & NSTitledWindowMask )
586 {
587 if ( ( style & wxMINIMIZE_BOX ) )
588 windowstyle |= NSMiniaturizableWindowMask ;
589
590 if ( ( style & wxMAXIMIZE_BOX ) )
591 windowstyle |= NSResizableWindowMask ; // TODO showing ZOOM ?
592
593 if ( ( style & wxRESIZE_BORDER ) )
594 windowstyle |= NSResizableWindowMask ;
595
596 if ( ( style & wxCLOSE_BOX) )
597 windowstyle |= NSClosableWindowMask ;
598 }
599 if ( extraStyle & wxFRAME_EX_METAL)
600 windowstyle |= NSTexturedBackgroundWindowMask;
601
bb839504
KO
602 if ( ( style & wxFRAME_FLOAT_ON_PARENT ) || ( style & wxFRAME_TOOL_WINDOW ) )
603 level = kCGFloatingWindowLevel;
604
33e90275
SC
605 if ( ( style & wxSTAY_ON_TOP ) )
606 level = kCGUtilityWindowLevel;
03647350 607
33e90275 608 NSRect r = wxToNSRect( NULL, wxRect( pos, size) );
03647350 609
01639b08 610 r = [NSWindow contentRectForFrameRect:r styleMask:windowstyle];
03647350 611
33e90275
SC
612 [m_macWindow initWithContentRect:r
613 styleMask:windowstyle
614 backing:NSBackingStoreBuffered
03647350 615 defer:NO
33e90275 616 ];
03647350 617
40c4350f
VZ
618 // If the parent is modal, windows with wxFRAME_FLOAT_ON_PARENT style need
619 // to be in kCGUtilityWindowLevel and not kCGFloatingWindowLevel to stay
620 // above the parent.
621 wxDialog * const parentDialog = wxDynamicCast(parent, wxDialog);
622 if (parentDialog && parentDialog->IsModal())
623 {
624 if (level == kCGFloatingWindowLevel)
625 {
626 level = kCGUtilityWindowLevel;
627 }
628
629 // Cocoa's modal loop does not process other windows by default, but
630 // don't call this on normal window levels so nested modal dialogs will
631 // still behave modally.
632 if (level != kCGNormalWindowLevel)
633 {
634 if ([m_macWindow isKindOfClass:[NSPanel class]])
635 {
636 [(NSPanel*)m_macWindow setWorksWhenModal:YES];
637 }
638 }
639 }
640
33e90275 641 [m_macWindow setLevel:level];
9d243a47 642 m_macWindowLevel = level;
dbeddfb9
SC
643
644 [m_macWindow setDelegate:controller];
03647350 645
826da451 646 [m_macWindow setAcceptsMouseMovedEvents: YES];
b61b8371
SC
647
648 if ( ( style & wxFRAME_SHAPED) )
649 {
650 [m_macWindow setOpaque:NO];
651 [m_macWindow setAlphaValue:1.0];
652 }
4539e903
SC
653
654 if ( !(style & wxFRAME_TOOL_WINDOW) )
655 [m_macWindow setHidesOnDeactivate:NO];
33e90275
SC
656}
657
17e2694c
SC
658void wxNonOwnedWindowCocoaImpl::Create( wxWindow* WXUNUSED(parent), WXWindow nativeWindow )
659{
660 m_macWindow = nativeWindow;
661}
33e90275
SC
662
663WXWindow wxNonOwnedWindowCocoaImpl::GetWXWindow() const
664{
665 return m_macWindow;
666}
667
668void wxNonOwnedWindowCocoaImpl::Raise()
669{
add051eb 670 [m_macWindow makeKeyAndOrderFront:nil];
33e90275 671}
03647350 672
33e90275
SC
673void wxNonOwnedWindowCocoaImpl::Lower()
674{
675 [m_macWindow orderWindow:NSWindowBelow relativeTo:0];
676}
677
dbc7ceb9
KO
678void wxNonOwnedWindowCocoaImpl::ShowWithoutActivating()
679{
a029e98a 680 [m_macWindow orderFront:nil];
eabe8426 681 [[m_macWindow contentView] setNeedsDisplay: YES];
dbc7ceb9
KO
682}
683
33e90275
SC
684bool wxNonOwnedWindowCocoaImpl::Show(bool show)
685{
686 if ( show )
687 {
1f83c631
SC
688 wxNonOwnedWindow* wxpeer = GetWXPeer();
689 if (wxpeer && !(wxpeer->GetWindowStyle() & wxFRAME_TOOL_WINDOW))
dbc7ceb9 690 [m_macWindow makeKeyAndOrderFront:nil];
1f83c631
SC
691 else
692 [m_macWindow orderFront:nil];
eabe8426 693 [[m_macWindow contentView] setNeedsDisplay: YES];
33e90275 694 }
03647350 695 else
33e90275
SC
696 [m_macWindow orderOut:nil];
697 return true;
698}
03647350 699
ab9a0b84
VZ
700bool wxNonOwnedWindowCocoaImpl::ShowWithEffect(bool show,
701 wxShowEffect effect,
702 unsigned timeout)
33e90275 703{
ab9a0b84
VZ
704 return wxWidgetCocoaImpl::
705 ShowViewOrWindowWithEffect(m_wxPeer, show, effect, timeout);
33e90275
SC
706}
707
708void wxNonOwnedWindowCocoaImpl::Update()
709{
710 [m_macWindow displayIfNeeded];
711}
712
713bool wxNonOwnedWindowCocoaImpl::SetTransparent(wxByte alpha)
714{
715 [m_macWindow setAlphaValue:(CGFloat) alpha/255.0];
716 return true;
717}
718
776036d6 719bool wxNonOwnedWindowCocoaImpl::SetBackgroundColour(const wxColour& col )
33e90275 720{
776036d6
SC
721 [m_macWindow setBackgroundColor:[NSColor colorWithCalibratedRed:(CGFloat) (col.Red() / 255.0)
722 green:(CGFloat) (col.Green() / 255.0)
723 blue:(CGFloat) (col.Blue() / 255.0)
724 alpha:(CGFloat) (col.Alpha() / 255.0)]];
33e90275
SC
725 return true;
726}
727
728void wxNonOwnedWindowCocoaImpl::SetExtraStyle( long exStyle )
729{
730 if ( m_macWindow )
731 {
732 bool metal = exStyle & wxFRAME_EX_METAL ;
733 int windowStyle = [ m_macWindow styleMask];
734 if ( metal && !(windowStyle & NSTexturedBackgroundWindowMask) )
735 {
9a83f860 736 wxFAIL_MSG( wxT("Metal Style cannot be changed after creation") );
33e90275
SC
737 }
738 else if ( !metal && (windowStyle & NSTexturedBackgroundWindowMask ) )
739 {
9a83f860 740 wxFAIL_MSG( wxT("Metal Style cannot be changed after creation") );
03647350 741 }
33e90275
SC
742 }
743}
03647350 744
b6dc21e7
KO
745void wxNonOwnedWindowCocoaImpl::SetWindowStyleFlag( long style )
746{
747 if (m_macWindow)
748 {
749 CGWindowLevel level = kCGNormalWindowLevel;
750
751 if (style & wxSTAY_ON_TOP)
752 level = kCGUtilityWindowLevel;
753 else if (( style & wxFRAME_FLOAT_ON_PARENT ) || ( style & wxFRAME_TOOL_WINDOW ))
754 level = kCGFloatingWindowLevel;
755
756 [m_macWindow setLevel: level];
9d243a47 757 m_macWindowLevel = level;
b6dc21e7
KO
758 }
759}
760
eabe8426 761bool wxNonOwnedWindowCocoaImpl::SetBackgroundStyle(wxBackgroundStyle style)
33e90275 762{
eabe8426
KO
763 if ( style == wxBG_STYLE_TRANSPARENT )
764 {
765 [m_macWindow setOpaque:NO];
766 [m_macWindow setBackgroundColor:[NSColor clearColor]];
767 }
768
33e90275
SC
769 return true;
770}
03647350 771
33e90275
SC
772bool wxNonOwnedWindowCocoaImpl::CanSetTransparent()
773{
774 return true;
775}
776
777void wxNonOwnedWindowCocoaImpl::MoveWindow(int x, int y, int width, int height)
778{
779 NSRect r = wxToNSRect( NULL, wxRect(x,y,width, height) );
54f11060
SC
780 // do not trigger refreshes upon invisible and possible partly created objects
781 [m_macWindow setFrame:r display:GetWXPeer()->IsShownOnScreen()];
33e90275
SC
782}
783
784void wxNonOwnedWindowCocoaImpl::GetPosition( int &x, int &y ) const
785{
786 wxRect r = wxFromNSRect( NULL, [m_macWindow frame] );
787 x = r.GetLeft();
788 y = r.GetTop();
789}
790
791void wxNonOwnedWindowCocoaImpl::GetSize( int &width, int &height ) const
792{
793 NSRect rect = [m_macWindow frame];
e490b0d2
VZ
794 width = (int)rect.size.width;
795 height = (int)rect.size.height;
33e90275
SC
796}
797
b0ec7fbb 798void wxNonOwnedWindowCocoaImpl::GetContentArea( int& left, int &top, int &width, int &height ) const
33e90275 799{
33e90275 800 NSRect rect = [[m_macWindow contentView] frame];
e490b0d2
VZ
801 left = (int)rect.origin.x;
802 top = (int)rect.origin.y;
803 width = (int)rect.size.width;
804 height = (int)rect.size.height;
33e90275 805}
03647350 806
20ede392 807bool wxNonOwnedWindowCocoaImpl::SetShape(const wxRegion& WXUNUSED(region))
33e90275 808{
eabe8426
KO
809 [m_macWindow setOpaque:NO];
810 [m_macWindow setBackgroundColor:[NSColor clearColor]];
811
812 return true;
33e90275
SC
813}
814
03647350 815void wxNonOwnedWindowCocoaImpl::SetTitle( const wxString& title, wxFontEncoding encoding )
33e90275
SC
816{
817 [m_macWindow setTitle:wxCFStringRef( title , encoding ).AsNSString()];
818}
03647350 819
33e90275
SC
820bool wxNonOwnedWindowCocoaImpl::IsMaximized() const
821{
cbe06760
VZ
822 if (([m_macWindow styleMask] & NSResizableWindowMask) != 0)
823 {
824 return [m_macWindow isZoomed];
825 }
826 else
827 {
828 NSRect rectScreen = [[NSScreen mainScreen] visibleFrame];
829 NSRect rectWindow = [m_macWindow frame];
830 return (rectScreen.origin.x == rectWindow.origin.x &&
831 rectScreen.origin.y == rectWindow.origin.y &&
832 rectScreen.size.width == rectWindow.size.width &&
833 rectScreen.size.height == rectWindow.size.height);
834 }
33e90275 835}
03647350 836
33e90275
SC
837bool wxNonOwnedWindowCocoaImpl::IsIconized() const
838{
839 return [m_macWindow isMiniaturized];
840}
03647350 841
33e90275
SC
842void wxNonOwnedWindowCocoaImpl::Iconize( bool iconize )
843{
844 if ( iconize )
845 [m_macWindow miniaturize:nil];
846 else
847 [m_macWindow deminiaturize:nil];
848}
03647350 849
20ede392 850void wxNonOwnedWindowCocoaImpl::Maximize(bool WXUNUSED(maximize))
33e90275
SC
851{
852 [m_macWindow zoom:nil];
853}
03647350 854
33e90275
SC
855
856// http://cocoadevcentral.com/articles/000028.php
857
858typedef struct
859{
860 int m_formerLevel;
861 NSRect m_formerFrame;
862} FullScreenData ;
863
864bool wxNonOwnedWindowCocoaImpl::IsFullScreen() const
865{
866 return m_macFullScreenData != NULL ;
867}
03647350 868
20ede392 869bool wxNonOwnedWindowCocoaImpl::ShowFullScreen(bool show, long WXUNUSED(style))
33e90275
SC
870{
871 if ( show )
872 {
873 FullScreenData *data = (FullScreenData *)m_macFullScreenData ;
874 delete data ;
875 data = new FullScreenData();
876
877 m_macFullScreenData = data ;
878 data->m_formerLevel = [m_macWindow level];
879 data->m_formerFrame = [m_macWindow frame];
2077a6ad
SC
880#if 0
881 // CGDisplayCapture( kCGDirectMainDisplay );
882 //[m_macWindow setLevel:NSMainMenuWindowLevel+1/*CGShieldingWindowLevel()*/];
883#endif
247bb510
VZ
884 NSRect screenframe = [[NSScreen mainScreen] frame];
885 NSRect frame = NSMakeRect (0, 0, 100, 100);
886 NSRect contentRect;
887 contentRect = [NSWindow contentRectForFrameRect: frame
d1fc9578 888 styleMask: [m_macWindow styleMask]];
247bb510
VZ
889 screenframe.origin.y += (frame.origin.y - contentRect.origin.y);
890 screenframe.size.height += (frame.size.height - contentRect.size.height);
891 [m_macWindow setFrame:screenframe display:YES];
2077a6ad 892
d1fc9578 893 SetSystemUIMode(kUIModeAllHidden,
2077a6ad 894 kUIOptionDisableAppleMenu
d1fc9578 895 /*
2077a6ad 896 | kUIOptionDisableProcessSwitch
d1fc9578
SC
897 | kUIOptionDisableForceQuit
898 */);
33e90275
SC
899 }
900 else if ( m_macFullScreenData != NULL )
901 {
902 FullScreenData *data = (FullScreenData *) m_macFullScreenData ;
2077a6ad
SC
903#if 0
904 // CGDisplayRelease( kCGDirectMainDisplay );
905 // [m_macWindow setLevel:data->m_formerLevel];
906#endif
907
33e90275
SC
908 [m_macWindow setFrame:data->m_formerFrame display:YES];
909 delete data ;
910 m_macFullScreenData = NULL ;
2077a6ad 911
d1fc9578 912 SetSystemUIMode(kUIModeNormal, 0);
33e90275 913 }
03647350 914
33e90275
SC
915 return true;
916}
917
32b8e701 918void wxNonOwnedWindowCocoaImpl::RequestUserAttention(int flagsWX)
33e90275 919{
98182acc 920 NSRequestUserAttentionType flagsOSX;
32b8e701
VZ
921 switch ( flagsWX )
922 {
923 case wxUSER_ATTENTION_INFO:
924 flagsOSX = NSInformationalRequest;
925 break;
926
927 case wxUSER_ATTENTION_ERROR:
928 flagsOSX = NSCriticalRequest;
929 break;
930
931 default:
932 wxFAIL_MSG( "invalid RequestUserAttention() flags" );
933 return;
934 }
935
936 [NSApp requestUserAttention:flagsOSX];
33e90275 937}
dbeddfb9
SC
938
939void wxNonOwnedWindowCocoaImpl::ScreenToWindow( int *x, int *y )
940{
941 wxPoint p((x ? *x : 0), (y ? *y : 0) );
dbeddfb9 942 NSPoint nspt = wxToNSPoint( NULL, p );
54f11060
SC
943 nspt = [m_macWindow convertScreenToBase:nspt];
944 nspt = [[m_macWindow contentView] convertPoint:nspt fromView:nil];
945 p = wxFromNSPoint([m_macWindow contentView], nspt);
dbeddfb9 946 if ( x )
03647350 947 *x = p.x;
dbeddfb9
SC
948 if ( y )
949 *y = p.y;
950}
951
952void wxNonOwnedWindowCocoaImpl::WindowToScreen( int *x, int *y )
953{
54f11060
SC
954 wxPoint p((x ? *x : 0), (y ? *y : 0) );
955 NSPoint nspt = wxToNSPoint( [m_macWindow contentView], p );
956 nspt = [[m_macWindow contentView] convertPoint:nspt toView:nil];
957 nspt = [m_macWindow convertBaseToScreen:nspt];
958 p = wxFromNSPoint( NULL, nspt);
dbeddfb9
SC
959 if ( x )
960 *x = p.x;
961 if ( y )
962 *y = p.y;
963}
964
dbc7ceb9
KO
965bool wxNonOwnedWindowCocoaImpl::IsActive()
966{
967 return [m_macWindow isKeyWindow];
968}
969
efb2fa41
KO
970void wxNonOwnedWindowCocoaImpl::SetModified(bool modified)
971{
972 [m_macWindow setDocumentEdited:modified];
973}
974
ebf7d5c4 975bool wxNonOwnedWindowCocoaImpl::IsModified() const
efb2fa41
KO
976{
977 return [m_macWindow isDocumentEdited];
978}
979
9d243a47
SC
980void wxNonOwnedWindowCocoaImpl::RestoreWindowLevel()
981{
982 if ( [m_macWindow level] != m_macWindowLevel )
983 [m_macWindow setLevel:m_macWindowLevel];
984}
985
986//
987//
988//
989
17e2694c
SC
990wxNonOwnedWindowImpl* wxNonOwnedWindowImpl::CreateNonOwnedWindow( wxNonOwnedWindow* wxpeer, wxWindow* parent, WXWindow nativeWindow)
991{
992 wxNonOwnedWindowCocoaImpl* now = new wxNonOwnedWindowCocoaImpl( wxpeer );
993 now->Create( parent, nativeWindow );
994 return now;
995}
996
dbeddfb9
SC
997wxNonOwnedWindowImpl* wxNonOwnedWindowImpl::CreateNonOwnedWindow( wxNonOwnedWindow* wxpeer, wxWindow* parent, const wxPoint& pos, const wxSize& size,
998 long style, long extraStyle, const wxString& name )
999{
1000 wxNonOwnedWindowImpl* now = new wxNonOwnedWindowCocoaImpl( wxpeer );
1001 now->Create( parent, pos, size, style , extraStyle, name );
1002 return now;
54f11060 1003}
9d243a47 1004