1 /////////////////////////////////////////////////////////////////////////////
 
   2 // Name:        src/osx/cocoa/nonownedwnd.mm
 
   3 // Purpose:     non owned window for cocoa
 
   4 // Author:      DavidStefan Csomor
 
   7 // RCS-ID:      $Id: nonownedwnd.mm 48805 2007-09-19 14:52:25Z SC $
 
   8 // Copyright:   (c) Stefan Csomor
 
   9 // Licence:     wxWindows licence
 
  10 /////////////////////////////////////////////////////////////////////////////
 
  12 #include "wx/wxprec.h"
 
  14     #include "wx/nonownedwnd.h"
 
  18 #include "wx/osx/private.h"
 
  20 NSRect wxToNSRect( NSView* parent, const wxRect& r )
 
  22     NSRect frame = parent ? [parent bounds] : [[NSScreen mainScreen] frame];
 
  25     if ( parent == NULL || ![ parent isFlipped ] )
 
  26         y = frame.size.height - ( r.y + r.height );
 
  27     return NSMakeRect(x, y, r.width , r.height);
 
  30 wxRect wxFromNSRect( NSView* parent, const NSRect& rect )
 
  32     NSRect frame = parent ? [parent bounds] : [[NSScreen mainScreen] frame];
 
  33     int y = rect.origin.y;
 
  34     int x = rect.origin.x;
 
  35     if ( parent == NULL || ![ parent isFlipped ] )
 
  36         y = frame.size.height - (rect.origin.y + rect.size.height);
 
  37     return wxRect( x, y, rect.size.width, rect.size.height );
 
  40 NSPoint wxToNSPoint( NSView* parent, const wxPoint& p )
 
  42     NSRect frame = parent ? [parent bounds] : [[NSScreen mainScreen] frame];
 
  45     if ( parent == NULL || ![ parent isFlipped ] )
 
  46         y = frame.size.height - ( p.y );
 
  47     return NSMakePoint(x, y);
 
  50 wxPoint wxFromNSPoint( NSView* parent, const NSPoint& p )
 
  52     NSRect frame = parent ? [parent bounds] : [[NSScreen mainScreen] frame];
 
  55     if ( parent == NULL || ![ parent isFlipped ] )
 
  56         y = frame.size.height - ( p.y );
 
  57     return wxPoint( x, y);
 
  60 bool shouldHandleSelector(SEL selector)
 
  62     if (selector == @selector(noop:) 
 
  63             || selector == @selector(complete:)
 
  64             || selector == @selector(deleteBackward:)
 
  65             || selector == @selector(deleteForward:)
 
  66             || selector == @selector(insertNewline:)
 
  67             || selector == @selector(insertTab:)
 
  68             || selector == @selector(keyDown:)
 
  69             || selector == @selector(keyUp:)
 
  70             || selector == @selector(scrollPageUp:)
 
  71             || selector == @selector(scrollPageDown:)
 
  72             || selector == @selector(scrollToBeginningOfDocument:)
 
  73             || selector == @selector(scrollToEndOfDocument:))
 
  81 // wx native implementation classes
 
  84 typedef void (*wxOSX_NoResponderHandlerPtr)(NSView* self, SEL _cmd, SEL selector);
 
  86 @interface wxNSWindow : NSWindow
 
  88     wxNonOwnedWindowCocoaImpl* impl;
 
  91 - (void)setImplementation: (wxNonOwnedWindowCocoaImpl *) theImplementation;
 
  92 - (wxNonOwnedWindowCocoaImpl*) implementation;
 
  93 - (void)noResponderFor: (SEL) selector;
 
  96 @implementation wxNSWindow
 
  98 - (void)setImplementation: (wxNonOwnedWindowCocoaImpl *) theImplementation
 
 100     impl = theImplementation;
 
 103 - (wxNonOwnedWindowCocoaImpl*) implementation
 
 108 - (void)doCommandBySelector:(SEL)selector
 
 110     if (shouldHandleSelector(selector) && 
 
 111         !(selector == @selector(cancel:) || selector == @selector(cancelOperation:)) )
 
 112         [super doCommandBySelector:selector];
 
 116 // NB: if we don't do this, all key downs that get handled lead to a NSBeep
 
 117 - (void)noResponderFor: (SEL) selector
 
 119     if (selector != @selector(keyDown:) && selector != @selector(keyUp:))
 
 121         [super noResponderFor:selector];
 
 122 //        wxOSX_NoResponderHandlerPtr superimpl = (wxOSX_NoResponderHandlerPtr) [[self superclass] instanceMethodForSelector:@selector(noResponderFor:)];
 
 123 //        superimpl(self, @selector(noResponderFor:), selector);
 
 129 @interface wxNSPanel : NSPanel
 
 132     wxNonOwnedWindowCocoaImpl* impl;
 
 135 - (void)setImplementation: (wxNonOwnedWindowCocoaImpl *) theImplementation;
 
 136 - (wxNonOwnedWindowCocoaImpl*) implementation;
 
 137 - (void)noResponderFor: (SEL) selector;
 
 140 @implementation wxNSPanel 
 
 142 - (void)setImplementation: (wxNonOwnedWindowCocoaImpl *) theImplementation
 
 144     impl = theImplementation;
 
 147 - (BOOL)canBecomeKeyWindow
 
 152 - (wxNonOwnedWindowCocoaImpl*) implementation
 
 157 - (void)doCommandBySelector:(SEL)selector
 
 159     if (shouldHandleSelector(selector))
 
 160         [super doCommandBySelector:selector];
 
 163 // NB: if we don't do this, it seems that all events that end here lead to a NSBeep
 
 164 - (void)noResponderFor: (SEL) selector
 
 166     if (selector != @selector(keyDown:) && selector != @selector(keyUp:))
 
 168         [super noResponderFor:selector];
 
 169 //        wxOSX_NoResponderHandlerPtr superimpl = (wxOSX_NoResponderHandlerPtr) [[self superclass] instanceMethodForSelector:@selector(noResponderFor:)];
 
 170 //        superimpl(self, @selector(noResponderFor:), selector);
 
 181 @interface wxNonOwnedWindowController : NSObject
 
 185 - (void)windowDidResize:(NSNotification *)notification;
 
 186 - (NSSize)windowWillResize:(NSWindow *)window toSize:(NSSize)proposedFrameSize;
 
 187 - (void)windowDidResignKey:(NSNotification *)notification;
 
 188 - (void)windowDidBecomeKey:(NSNotification *)notification;
 
 189 - (void)windowDidMove:(NSNotification *)notification;
 
 190 - (BOOL)windowShouldClose:(id)window;
 
 194 @implementation wxNonOwnedWindowController
 
 202 - (BOOL)windowShouldClose:(id)nwindow
 
 204     wxNSWindow* window = (wxNSWindow*) nwindow;
 
 205     wxNonOwnedWindowCocoaImpl* windowimpl = [window implementation];
 
 208         wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
 
 215 - (NSSize)windowWillResize:(NSWindow *)win
 
 216                     toSize:(NSSize)proposedFrameSize
 
 218     NSRect frame = [win frame];
 
 219     wxRect wxframe = wxFromNSRect( NULL, frame );
 
 220     wxframe.SetWidth( proposedFrameSize.width );
 
 221     wxframe.SetHeight( proposedFrameSize.height );
 
 222     wxNSWindow* window = (wxNSWindow*) win;
 
 223     wxNonOwnedWindowCocoaImpl* windowimpl = [window implementation];
 
 226         wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
 
 229             wxpeer->HandleResizing( 0, &wxframe );
 
 230             NSSize newSize = NSMakeSize(wxframe.GetWidth(), wxframe.GetHeight());
 
 235     return proposedFrameSize;
 
 238 - (void)windowDidResize:(NSNotification *)notification
 
 240     wxNSWindow* window = (wxNSWindow*) [notification object];
 
 241     wxNonOwnedWindowCocoaImpl* windowimpl = [window implementation];
 
 244         wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
 
 246             wxpeer->HandleResized(0);
 
 250 - (void)windowDidMove:(NSNotification *)notification
 
 252     wxNSWindow* window = (wxNSWindow*) [notification object];
 
 253     wxNonOwnedWindowCocoaImpl* windowimpl = [window implementation];
 
 256         wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
 
 258             wxpeer->HandleMoved(0);
 
 262 - (void)windowDidBecomeKey:(NSNotification *)notification
 
 264     wxNSWindow* window = (wxNSWindow*) [notification object];
 
 265     wxNonOwnedWindowCocoaImpl* windowimpl = [window implementation];
 
 268         wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
 
 270             wxpeer->HandleActivated(0, true);
 
 274 - (void)windowDidResignKey:(NSNotification *)notification
 
 276     wxNSWindow* window = (wxNSWindow*) [notification object];
 
 277     wxNonOwnedWindowCocoaImpl* windowimpl = [window implementation];
 
 280         wxNonOwnedWindow* wxpeer = windowimpl->GetWXPeer();
 
 283             wxpeer->HandleActivated(0, false);
 
 284             // Needed for popup window since the firstResponder 
 
 285             // (focus in wx) doesn't change when this 
 
 286             // TLW becomes inactive.
 
 287             wxFocusEvent event( wxEVT_KILL_FOCUS, wxpeer->GetId());
 
 288             event.SetEventObject(wxpeer);
 
 289             wxpeer->HandleWindowEvent(event);
 
 296 IMPLEMENT_DYNAMIC_CLASS( wxNonOwnedWindowCocoaImpl , wxNonOwnedWindowImpl )
 
 298 wxNonOwnedWindowCocoaImpl::wxNonOwnedWindowCocoaImpl( wxNonOwnedWindow* nonownedwnd) : 
 
 299     wxNonOwnedWindowImpl(nonownedwnd)
 
 302     m_macFullScreenData = NULL;
 
 305 wxNonOwnedWindowCocoaImpl::wxNonOwnedWindowCocoaImpl() 
 
 308     m_macFullScreenData = NULL;
 
 311 wxNonOwnedWindowCocoaImpl::~wxNonOwnedWindowCocoaImpl()
 
 313     [m_macWindow setImplementation:nil];
 
 314     [m_macWindow setDelegate:nil];
 
 315     [m_macWindow release];
 
 318 void wxNonOwnedWindowCocoaImpl::Destroy()
 
 320     wxPendingDelete.Append( new wxDeferredObjectDeleter( this ) );
 
 323 void wxNonOwnedWindowCocoaImpl::Create( wxWindow* WXUNUSED(parent), const wxPoint& pos, const wxSize& size,
 
 324 long style, long extraStyle, const wxString& WXUNUSED(name) )
 
 326     static wxNonOwnedWindowController* controller = NULL;
 
 329         controller =[[wxNonOwnedWindowController alloc] init];
 
 332     int windowstyle = NSBorderlessWindowMask;
 
 334     if ( style & wxFRAME_TOOL_WINDOW || ( style & wxPOPUP_WINDOW ) || 
 
 335             GetWXPeer()->GetExtraStyle() & wxTOPLEVEL_EX_DIALOG )
 
 337         m_macWindow = [wxNSPanel alloc];
 
 340         m_macWindow = [wxNSWindow alloc];
 
 342     CGWindowLevel level = kCGNormalWindowLevel;
 
 344     if ( style & wxFRAME_TOOL_WINDOW )
 
 346         windowstyle |= NSUtilityWindowMask;
 
 347         if ( ( style & wxMINIMIZE_BOX ) || ( style & wxMAXIMIZE_BOX ) ||
 
 348             ( style & wxCLOSE_BOX ) || ( style & wxSYSTEM_MENU ) )
 
 350             windowstyle |= NSTitledWindowMask ;
 
 353     else if ( ( style & wxPOPUP_WINDOW ) )
 
 355         level = kCGPopUpMenuWindowLevel;
 
 357         if ( ( style & wxBORDER_NONE ) )
 
 359             wclass = kHelpWindowClass ;   // has no border
 
 360             attr |= kWindowNoShadowAttribute;
 
 364             wclass = kPlainWindowClass ;  // has a single line border, it will have to do for now
 
 368     else if ( ( style & wxCAPTION ) )
 
 370         windowstyle |= NSTitledWindowMask ;
 
 372     else if ( ( style & wxFRAME_DRAWER ) )
 
 375         wclass = kDrawerWindowClass;
 
 380         // set these even if we have no title, otherwise the controls won't be visible
 
 381         if ( ( style & wxMINIMIZE_BOX ) || ( style & wxMAXIMIZE_BOX ) ||
 
 382             ( style & wxCLOSE_BOX ) || ( style & wxSYSTEM_MENU ) )
 
 384             windowstyle |= NSTitledWindowMask ;
 
 387         else if ( ( style & wxNO_BORDER ) )
 
 389             wclass = kSimpleWindowClass ;
 
 393             wclass = kPlainWindowClass ;
 
 398     if ( windowstyle & NSTitledWindowMask )
 
 400         if ( ( style & wxMINIMIZE_BOX ) )
 
 401             windowstyle |= NSMiniaturizableWindowMask ;
 
 403         if ( ( style & wxMAXIMIZE_BOX ) )
 
 404             windowstyle |= NSResizableWindowMask ; // TODO showing ZOOM ?
 
 406         if ( ( style & wxRESIZE_BORDER ) )
 
 407             windowstyle |= NSResizableWindowMask ;
 
 409         if ( ( style & wxCLOSE_BOX) )
 
 410             windowstyle |= NSClosableWindowMask ;
 
 412     if ( extraStyle & wxFRAME_EX_METAL)
 
 413         windowstyle |= NSTexturedBackgroundWindowMask;
 
 415     if ( ( style & wxFRAME_FLOAT_ON_PARENT ) || ( style & wxFRAME_TOOL_WINDOW ) )
 
 416         level = kCGFloatingWindowLevel;
 
 418     if ( ( style & wxSTAY_ON_TOP ) )
 
 419         level = kCGUtilityWindowLevel;
 
 421     NSRect r = wxToNSRect( NULL, wxRect( pos, size) );
 
 423     [m_macWindow setImplementation:this];
 
 425     [m_macWindow initWithContentRect:r
 
 426         styleMask:windowstyle
 
 427         backing:NSBackingStoreBuffered
 
 431     [m_macWindow setLevel:level];
 
 433     [m_macWindow setDelegate:controller];
 
 435     [m_macWindow setAcceptsMouseMovedEvents: YES];
 
 439 WXWindow wxNonOwnedWindowCocoaImpl::GetWXWindow() const
 
 444 void wxNonOwnedWindowCocoaImpl::Raise()
 
 446     [m_macWindow orderWindow:NSWindowAbove relativeTo:0];
 
 449 void wxNonOwnedWindowCocoaImpl::Lower()
 
 451     [m_macWindow orderWindow:NSWindowBelow relativeTo:0];
 
 454 bool wxNonOwnedWindowCocoaImpl::Show(bool show)
 
 458         [m_macWindow makeKeyAndOrderFront:nil];
 
 459         [[m_macWindow contentView] setNeedsDisplay:YES];
 
 462         [m_macWindow orderOut:nil];
 
 466 bool wxNonOwnedWindowCocoaImpl::ShowWithEffect(bool show, wxShowEffect WXUNUSED(effect), unsigned WXUNUSED(timeout))
 
 471 void wxNonOwnedWindowCocoaImpl::Update()
 
 473     [m_macWindow displayIfNeeded];
 
 476 bool wxNonOwnedWindowCocoaImpl::SetTransparent(wxByte alpha)
 
 478     [m_macWindow setAlphaValue:(CGFloat) alpha/255.0];
 
 482 bool wxNonOwnedWindowCocoaImpl::SetBackgroundColour(const wxColour& WXUNUSED(col) )
 
 487 void wxNonOwnedWindowCocoaImpl::SetExtraStyle( long exStyle )
 
 491         bool metal = exStyle & wxFRAME_EX_METAL ;
 
 492         int windowStyle = [ m_macWindow styleMask];
 
 493         if ( metal && !(windowStyle & NSTexturedBackgroundWindowMask) )
 
 495             wxFAIL_MSG( _T("Metal Style cannot be changed after creation") );
 
 497         else if ( !metal && (windowStyle & NSTexturedBackgroundWindowMask ) )
 
 499             wxFAIL_MSG( _T("Metal Style cannot be changed after creation") );
 
 504 bool wxNonOwnedWindowCocoaImpl::SetBackgroundStyle(wxBackgroundStyle WXUNUSED(style))
 
 509 bool wxNonOwnedWindowCocoaImpl::CanSetTransparent()
 
 514 void wxNonOwnedWindowCocoaImpl::MoveWindow(int x, int y, int width, int height)
 
 516     NSRect r = wxToNSRect( NULL, wxRect(x,y,width, height) );
 
 517     // do not trigger refreshes upon invisible and possible partly created objects
 
 518     [m_macWindow setFrame:r display:GetWXPeer()->IsShownOnScreen()];
 
 521 void wxNonOwnedWindowCocoaImpl::GetPosition( int &x, int &y ) const
 
 523     wxRect r = wxFromNSRect( NULL, [m_macWindow frame] );
 
 528 void wxNonOwnedWindowCocoaImpl::GetSize( int &width, int &height ) const
 
 530     NSRect rect = [m_macWindow frame];
 
 531     width = rect.size.width;
 
 532     height = rect.size.height;
 
 535 void wxNonOwnedWindowCocoaImpl::GetContentArea( int& left, int &top, int &width, int &height ) const
 
 537     NSRect outer = NSMakeRect(100,100,100,100);
 
 538     NSRect content = [NSWindow contentRectForFrameRect:outer styleMask:[m_macWindow styleMask] ];
 
 539     NSRect rect = [[m_macWindow contentView] frame];
 
 540     left = rect.origin.x;
 
 542     width = rect.size.width;
 
 543     height = rect.size.height;
 
 546 bool wxNonOwnedWindowCocoaImpl::SetShape(const wxRegion& WXUNUSED(region))
 
 551 void wxNonOwnedWindowCocoaImpl::SetTitle( const wxString& title, wxFontEncoding encoding ) 
 
 553     [m_macWindow setTitle:wxCFStringRef( title , encoding ).AsNSString()];
 
 556 bool wxNonOwnedWindowCocoaImpl::IsMaximized() const
 
 558     return [m_macWindow isZoomed];
 
 561 bool wxNonOwnedWindowCocoaImpl::IsIconized() const
 
 563     return [m_macWindow isMiniaturized];
 
 566 void wxNonOwnedWindowCocoaImpl::Iconize( bool iconize )
 
 569         [m_macWindow miniaturize:nil];
 
 571         [m_macWindow deminiaturize:nil];
 
 574 void wxNonOwnedWindowCocoaImpl::Maximize(bool WXUNUSED(maximize))
 
 576     [m_macWindow zoom:nil];
 
 580 // http://cocoadevcentral.com/articles/000028.php
 
 585     NSRect m_formerFrame;
 
 588 bool wxNonOwnedWindowCocoaImpl::IsFullScreen() const
 
 590     return m_macFullScreenData != NULL ;
 
 593 bool wxNonOwnedWindowCocoaImpl::ShowFullScreen(bool show, long WXUNUSED(style))
 
 597         FullScreenData *data = (FullScreenData *)m_macFullScreenData ;
 
 599         data = new FullScreenData();
 
 601         m_macFullScreenData = data ;
 
 602         data->m_formerLevel = [m_macWindow level];
 
 603         data->m_formerFrame = [m_macWindow frame];
 
 604         CGDisplayCapture( kCGDirectMainDisplay );
 
 605         [m_macWindow setLevel:CGShieldingWindowLevel()];
 
 606         [m_macWindow setFrame:[[NSScreen mainScreen] frame] display:YES];
 
 608     else if ( m_macFullScreenData != NULL )
 
 610         FullScreenData *data = (FullScreenData *) m_macFullScreenData ;
 
 611         CGDisplayRelease( kCGDirectMainDisplay );
 
 612         [m_macWindow setLevel:data->m_formerLevel];
 
 613         [m_macWindow setFrame:data->m_formerFrame display:YES];
 
 615         m_macFullScreenData = NULL ;
 
 621 void wxNonOwnedWindowCocoaImpl::RequestUserAttention(int WXUNUSED(flags))
 
 625 void wxNonOwnedWindowCocoaImpl::ScreenToWindow( int *x, int *y )
 
 627     wxPoint p((x ? *x : 0), (y ? *y : 0) );
 
 628     NSPoint nspt = wxToNSPoint( NULL, p );
 
 629     nspt = [m_macWindow convertScreenToBase:nspt];
 
 630     nspt = [[m_macWindow contentView] convertPoint:nspt fromView:nil];
 
 631     p = wxFromNSPoint([m_macWindow contentView], nspt);
 
 638 void wxNonOwnedWindowCocoaImpl::WindowToScreen( int *x, int *y )
 
 640     wxPoint p((x ? *x : 0), (y ? *y : 0) );
 
 641     NSPoint nspt = wxToNSPoint( [m_macWindow contentView], p );
 
 642     nspt = [[m_macWindow contentView] convertPoint:nspt toView:nil];
 
 643     nspt = [m_macWindow convertBaseToScreen:nspt];
 
 644     p = wxFromNSPoint( NULL, nspt);
 
 651 wxNonOwnedWindowImpl* wxNonOwnedWindowImpl::CreateNonOwnedWindow( wxNonOwnedWindow* wxpeer, wxWindow* parent, const wxPoint& pos, const wxSize& size,
 
 652     long style, long extraStyle, const wxString& name )
 
 654     wxNonOwnedWindowImpl* now = new wxNonOwnedWindowCocoaImpl( wxpeer );
 
 655     now->Create( parent, pos, size, style , extraStyle, name );