mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-08 12:37:37 +00:00
1773 lines
55 KiB
Plaintext
1773 lines
55 KiB
Plaintext
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
||
/* ***** BEGIN LICENSE BLOCK *****
|
||
* Version: NPL 1.1/GPL 2.0/LGPL 2.1
|
||
*
|
||
* The contents of this file are subject to the Netscape Public License
|
||
* Version 1.1 (the "License"); you may not use this file except in
|
||
* compliance with the License. You may obtain a copy of the License at
|
||
* http://www.mozilla.org/NPL/
|
||
*
|
||
* Software distributed under the License is distributed on an "AS IS" basis,
|
||
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
||
* for the specific language governing rights and limitations under the
|
||
* License.
|
||
*
|
||
* The Original Code is mozilla.org code.
|
||
*
|
||
* The Initial Developer of the Original Code is
|
||
* Netscape Communications Corporation.
|
||
* Portions created by the Initial Developer are Copyright (C) 1998
|
||
* the Initial Developer. All Rights Reserved.
|
||
*
|
||
* Contributor(s):
|
||
*
|
||
* Alternatively, the contents of this file may be used under the terms of
|
||
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
||
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
||
* in which case the provisions of the GPL or the LGPL are applicable instead
|
||
* of those above. If you wish to allow use of your version of this file only
|
||
* under the terms of either the GPL or the LGPL, and not to allow others to
|
||
* use your version of this file under the terms of the NPL, indicate your
|
||
* decision by deleting the provisions above and replace them with the notice
|
||
* and other provisions required by the GPL or the LGPL. If you do not delete
|
||
* the provisions above, a recipient may use your version of this file under
|
||
* the terms of any one of the NPL, the GPL or the LGPL.
|
||
*
|
||
* ***** END LICENSE BLOCK ***** */
|
||
|
||
#include "nsCocoaWindow.h"
|
||
|
||
#include "nsIServiceManager.h" // for drag and drop
|
||
#include "nsWidgetsCID.h"
|
||
#include "nsIDragService.h"
|
||
#include "nsIDragSession.h"
|
||
#include "nsIDragSessionMac.h"
|
||
#include "nsIScreen.h"
|
||
#include "nsIScreenManager.h"
|
||
#include "nsGUIEvent.h"
|
||
#include "nsCarbonHelpers.h"
|
||
#include "nsGFXUtils.h"
|
||
#include "nsMacResources.h"
|
||
#include "nsRegionMac.h"
|
||
#include "nsIRollupListener.h"
|
||
#import "nsChildView.h"
|
||
|
||
#include "nsIEventQueueService.h"
|
||
|
||
#if TARGET_CARBON
|
||
#include <CFString.h>
|
||
#endif
|
||
|
||
#include <Gestalt.h>
|
||
#include <Quickdraw.h>
|
||
|
||
#if UNIVERSAL_INTERFACES_VERSION < 0x0340
|
||
enum {
|
||
kEventWindowConstrain = 83
|
||
};
|
||
const UInt32 kWindowLiveResizeAttribute = (1L << 28);
|
||
#endif
|
||
|
||
// Define Class IDs -- i hate having to do this
|
||
static NS_DEFINE_CID(kCDragServiceCID, NS_DRAGSERVICE_CID);
|
||
//static const char *sScreenManagerContractID = "@mozilla.org/gfx/screenmanager;1";
|
||
|
||
// from MacHeaders.c
|
||
#ifndef topLeft
|
||
#define topLeft(r) (((Point *) &(r))[0])
|
||
#endif
|
||
#ifndef botRight
|
||
#define botRight(r) (((Point *) &(r))[1])
|
||
#endif
|
||
|
||
// externs defined in nsWindow.cpp
|
||
extern nsIRollupListener * gRollupListener;
|
||
extern nsIWidget * gRollupWidget;
|
||
|
||
static PRBool OnMacOSX();
|
||
|
||
#define kWindowPositionSlop 20
|
||
|
||
|
||
#if 0
|
||
void SetDragActionBasedOnModifiers ( nsIDragService* inDragService, short inModifiers ) ;
|
||
|
||
|
||
//
|
||
// SetDragActionsBasedOnModifiers [static]
|
||
//
|
||
// Examines the MacOS modifier keys and sets the appropriate drag action on the
|
||
// drag session to copy/move/etc
|
||
//
|
||
void
|
||
SetDragActionBasedOnModifiers ( nsIDragService* inDragService, short inModifiers )
|
||
{
|
||
nsCOMPtr<nsIDragSession> dragSession;
|
||
inDragService->GetCurrentSession ( getter_AddRefs(dragSession) );
|
||
if ( dragSession ) {
|
||
PRUint32 action = nsIDragService::DRAGDROP_ACTION_MOVE;
|
||
|
||
// force copy = option, alias = cmd-option, default is move
|
||
if ( inModifiers & optionKey ) {
|
||
if ( inModifiers & cmdKey )
|
||
action = nsIDragService::DRAGDROP_ACTION_LINK;
|
||
else
|
||
action = nsIDragService::DRAGDROP_ACTION_COPY;
|
||
}
|
||
|
||
dragSession->SetDragAction ( action );
|
||
}
|
||
|
||
} // SetDragActionBasedOnModifiers
|
||
|
||
#pragma mark -
|
||
|
||
|
||
//<2F><><EFBFBD> this should probably go into the drag session as a static
|
||
pascal OSErr
|
||
nsCocoaWindow :: DragTrackingHandler ( DragTrackingMessage theMessage, WindowPtr theWindow,
|
||
void *handlerRefCon, DragReference theDrag)
|
||
{
|
||
// holds our drag service across multiple calls to this callback. The reference to
|
||
// the service is obtained when the mouse enters the window and is released when
|
||
// the mouse leaves the window (or there is a drop). This prevents us from having
|
||
// to re-establish the connection to the service manager 15 times a second when
|
||
// handling the |kDragTrackingInWindow| message.
|
||
static nsIDragService* sDragService = nsnull;
|
||
|
||
nsCocoaWindow* geckoWindow = reinterpret_cast<nsCocoaWindow*>(handlerRefCon);
|
||
if ( !theWindow || !geckoWindow )
|
||
return dragNotAcceptedErr;
|
||
|
||
nsresult rv = NS_OK;
|
||
switch ( theMessage ) {
|
||
|
||
case kDragTrackingEnterHandler:
|
||
break;
|
||
|
||
case kDragTrackingEnterWindow:
|
||
{
|
||
// get our drag service for the duration of the drag.
|
||
nsresult rv = nsServiceManager::GetService(kCDragServiceCID,
|
||
NS_GET_IID(nsIDragService),
|
||
(nsISupports **)&sDragService);
|
||
NS_ASSERTION ( sDragService, "Couldn't get a drag service, we're in biiig trouble" );
|
||
|
||
// tell the session about this drag
|
||
if ( sDragService ) {
|
||
sDragService->StartDragSession();
|
||
nsCOMPtr<nsIDragSessionMac> macSession ( do_QueryInterface(sDragService) );
|
||
if ( macSession )
|
||
macSession->SetDragReference ( theDrag );
|
||
}
|
||
|
||
// let gecko know that the mouse has entered the window so it
|
||
// can start tracking and sending enter/exit events to frames.
|
||
Point mouseLocGlobal;
|
||
::GetDragMouse ( theDrag, &mouseLocGlobal, nsnull );
|
||
geckoWindow->DragEvent ( NS_DRAGDROP_ENTER, mouseLocGlobal, 0L );
|
||
break;
|
||
}
|
||
|
||
case kDragTrackingInWindow:
|
||
{
|
||
Point mouseLocGlobal;
|
||
::GetDragMouse ( theDrag, &mouseLocGlobal, nsnull );
|
||
short modifiers;
|
||
::GetDragModifiers ( theDrag, &modifiers, nsnull, nsnull );
|
||
|
||
NS_ASSERTION ( sDragService, "If we don't have a drag service, we're fucked" );
|
||
|
||
// set the drag action on the service so the frames know what is going on
|
||
SetDragActionBasedOnModifiers ( sDragService, modifiers );
|
||
|
||
// clear out the |canDrop| property of the drag session. If it's meant to
|
||
// be, it will be set again.
|
||
nsCOMPtr<nsIDragSession> session;
|
||
sDragService->GetCurrentSession(getter_AddRefs(session));
|
||
NS_ASSERTION ( session, "If we don't have a drag session, we're fucked" );
|
||
if ( session )
|
||
session->SetCanDrop(PR_FALSE);
|
||
|
||
// pass into gecko for handling...
|
||
geckoWindow->DragEvent ( NS_DRAGDROP_OVER, mouseLocGlobal, modifiers );
|
||
break;
|
||
}
|
||
|
||
case kDragTrackingLeaveWindow:
|
||
{
|
||
// tell the drag service that we're done with it.
|
||
if ( sDragService ) {
|
||
sDragService->EndDragSession();
|
||
|
||
// clear out the dragRef in the drag session. We are guaranteed that
|
||
// this will be called _after_ the drop has been processed (if there
|
||
// is one), so we're not destroying valuable information if the drop
|
||
// was in our window.
|
||
nsCOMPtr<nsIDragSessionMac> macSession ( do_QueryInterface(sDragService) );
|
||
if ( macSession )
|
||
macSession->SetDragReference ( 0 );
|
||
}
|
||
|
||
// let gecko know that the mouse has left the window so it
|
||
// can stop tracking and sending enter/exit events to frames.
|
||
Point mouseLocGlobal;
|
||
::GetDragMouse ( theDrag, &mouseLocGlobal, nsnull );
|
||
geckoWindow->DragEvent ( NS_DRAGDROP_EXIT, mouseLocGlobal, 0L );
|
||
|
||
::HideDragHilite ( theDrag );
|
||
|
||
// we're _really_ done with it, so let go of the service.
|
||
if ( sDragService ) {
|
||
nsServiceManager::ReleaseService(kCDragServiceCID, sDragService);
|
||
sDragService = nsnull;
|
||
}
|
||
|
||
break;
|
||
}
|
||
|
||
} // case of each drag message
|
||
|
||
return noErr;
|
||
|
||
} // DragTrackingHandler
|
||
|
||
|
||
//<2F><><EFBFBD> this should probably go into the drag session as a static
|
||
pascal OSErr
|
||
nsCocoaWindow :: DragReceiveHandler (WindowPtr theWindow, void *handlerRefCon,
|
||
DragReference theDragRef)
|
||
{
|
||
// get our window back from the refCon
|
||
nsCocoaWindow* geckoWindow = reinterpret_cast<nsCocoaWindow*>(handlerRefCon);
|
||
if ( !theWindow || !geckoWindow )
|
||
return dragNotAcceptedErr;
|
||
|
||
// We make the assuption that the dragOver handlers have correctly set
|
||
// the |canDrop| property of the Drag Session. Before we dispatch the event
|
||
// into Gecko, check that value and either dispatch it or set the result
|
||
// code to "spring-back" and show the user the drag failed.
|
||
OSErr result = noErr;
|
||
nsCOMPtr<nsIDragService> dragService ( do_GetService(kCDragServiceCID) );
|
||
if ( dragService ) {
|
||
nsCOMPtr<nsIDragSession> dragSession;
|
||
dragService->GetCurrentSession ( getter_AddRefs(dragSession) );
|
||
if ( dragSession ) {
|
||
// if the target has set that it can accept the drag, pass along
|
||
// to gecko, otherwise set phasers for failure.
|
||
PRBool canDrop = PR_FALSE;
|
||
if ( NS_SUCCEEDED(dragSession->GetCanDrop(&canDrop)) )
|
||
if ( canDrop ) {
|
||
// pass the drop event along to Gecko
|
||
Point mouseLocGlobal;
|
||
::GetDragMouse ( theDragRef, &mouseLocGlobal, nsnull );
|
||
short modifiers;
|
||
::GetDragModifiers ( theDragRef, &modifiers, nsnull, nsnull );
|
||
geckoWindow->DragEvent ( NS_DRAGDROP_DROP, mouseLocGlobal, modifiers );
|
||
}
|
||
else
|
||
result = dragNotAcceptedErr;
|
||
} // if a valid drag session
|
||
|
||
// we don't need the drag session anymore, the user has released the
|
||
// mouse and the event has already gone to gecko.
|
||
dragService->EndDragSession();
|
||
}
|
||
|
||
return result;
|
||
|
||
} // DragReceiveHandler
|
||
|
||
#endif
|
||
|
||
|
||
NS_IMPL_ISUPPORTS_INHERITED0(nsCocoaWindow, Inherited);
|
||
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// nsCocoaWindow constructor
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
nsCocoaWindow::nsCocoaWindow()
|
||
#if 0
|
||
, mWindowMadeHere(PR_FALSE)
|
||
, mIsDialog(PR_FALSE)
|
||
, mMacEventHandler(nsnull)
|
||
, mAcceptsActivation(PR_TRUE)
|
||
, mIsActive(PR_FALSE)
|
||
, mZoomOnShow(PR_FALSE)
|
||
#endif
|
||
:
|
||
mOffsetParent(nsnull)
|
||
, mIsDialog(PR_FALSE)
|
||
, mIsResizing(PR_FALSE)
|
||
, mWindowMadeHere(PR_FALSE)
|
||
, mWindow(nil)
|
||
{
|
||
#if 0
|
||
mMacEventHandler.reset(new nsMacEventHandler(this));
|
||
WIDGET_SET_CLASSNAME("nsCocoaWindow");
|
||
|
||
// create handlers for drag&drop
|
||
mDragTrackingHandlerUPP = NewDragTrackingHandlerUPP(DragTrackingHandler);
|
||
mDragReceiveHandlerUPP = NewDragReceiveHandlerUPP(DragReceiveHandler);
|
||
#endif
|
||
}
|
||
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// nsCocoaWindow destructor
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
nsCocoaWindow::~nsCocoaWindow()
|
||
{
|
||
if ( mWindow && mWindowMadeHere ) {
|
||
[mWindow autorelease];
|
||
[mDelegate autorelease];
|
||
}
|
||
|
||
#if 0
|
||
if (mWindowPtr)
|
||
{
|
||
if (mWindowMadeHere)
|
||
::DisposeWindow(mWindowPtr);
|
||
|
||
// clean up DragManager stuff
|
||
if ( mDragTrackingHandlerUPP ) {
|
||
::RemoveTrackingHandler ( mDragTrackingHandlerUPP, mWindowPtr );
|
||
::DisposeDragTrackingHandlerUPP ( mDragTrackingHandlerUPP );
|
||
}
|
||
if ( mDragReceiveHandlerUPP ) {
|
||
::RemoveReceiveHandler ( mDragReceiveHandlerUPP, mWindowPtr );
|
||
::DisposeDragReceiveHandlerUPP ( mDragReceiveHandlerUPP );
|
||
}
|
||
|
||
nsMacMessageSink::RemoveRaptorWindowFromList(mWindowPtr);
|
||
mWindowPtr = nsnull;
|
||
}
|
||
#endif
|
||
|
||
}
|
||
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// Utility method for implementing both Create(nsIWidget ...) and
|
||
// Create(nsNativeWidget...)
|
||
//-------------------------------------------------------------------------
|
||
|
||
nsresult nsCocoaWindow::StandardCreate(nsIWidget *aParent,
|
||
const nsRect &aRect,
|
||
EVENT_CALLBACK aHandleEventFunction,
|
||
nsIDeviceContext *aContext,
|
||
nsIAppShell *aAppShell,
|
||
nsIToolkit *aToolkit,
|
||
nsWidgetInitData *aInitData,
|
||
nsNativeWidget aNativeParent)
|
||
{
|
||
Inherited::BaseCreate ( aParent, aRect, aHandleEventFunction, aContext, aAppShell,
|
||
aToolkit, aInitData );
|
||
|
||
if ( !aNativeParent ) {
|
||
mOffsetParent = aParent;
|
||
|
||
nsWindowType windowType = eWindowType_toplevel;
|
||
if (aInitData) {
|
||
mWindowType = aInitData->mWindowType;
|
||
// if a toplevel window was requested without a titlebar, use a dialog windowproc
|
||
if (aInitData->mWindowType == eWindowType_toplevel &&
|
||
(aInitData->mBorderStyle == eBorderStyle_none ||
|
||
aInitData->mBorderStyle != eBorderStyle_all && !(aInitData->mBorderStyle & eBorderStyle_title)))
|
||
windowType = eWindowType_dialog;
|
||
}
|
||
else
|
||
mWindowType = (mIsDialog ? eWindowType_dialog : eWindowType_toplevel);
|
||
|
||
// create the cocoa window
|
||
NSRect rect;
|
||
rect.origin.x = rect.origin.y = 1.0;
|
||
rect.size.width = rect.size.height = 1.0;
|
||
unsigned int features = NSTitledWindowMask | NSClosableWindowMask | NSMiniaturizableWindowMask
|
||
| NSResizableWindowMask;
|
||
if ( mWindowType == eWindowType_popup || mWindowType == eWindowType_invisible )
|
||
features = 0;
|
||
|
||
// XXXdwh Just don't make popup windows yet. They mess up the world.
|
||
if (mWindowType == eWindowType_popup)
|
||
return NS_OK;
|
||
|
||
mWindow = [[NSWindow alloc] initWithContentRect:rect styleMask:features
|
||
backing:NSBackingStoreBuffered defer:NO];
|
||
|
||
// Popups will receive a "close" message when an app terminates
|
||
// that causes an extra release to occur. Make sure popups
|
||
// are set not to release when closed.
|
||
if (features == 0)
|
||
[mWindow setReleasedWhenClosed: NO];
|
||
|
||
// create a quickdraw view as the toplevel content view of the window
|
||
NSQuickDrawView* content = [[[NSQuickDrawView alloc] init] autorelease];
|
||
[content setFrame:[[mWindow contentView] frame]];
|
||
[mWindow setContentView:content];
|
||
|
||
// register for mouse-moved events. The default is to ignore them for perf reasons.
|
||
[mWindow setAcceptsMouseMovedEvents:YES];
|
||
|
||
// setup our notification delegate. Note that setDelegate: does NOT retain.
|
||
mDelegate = [[WindowDelegate alloc] initWithGeckoWindow:this];
|
||
[mWindow setDelegate:mDelegate];
|
||
|
||
mWindowMadeHere = PR_TRUE;
|
||
}
|
||
|
||
#if 0
|
||
short bottomPinDelta = 0; // # of pixels to subtract to pin window bottom
|
||
nsCOMPtr<nsIToolkit> theToolkit = aToolkit;
|
||
|
||
// build the main native window
|
||
if (aNativeParent == nsnull)
|
||
{
|
||
nsWindowType windowType;
|
||
if (aInitData)
|
||
{
|
||
mWindowType = aInitData->mWindowType;
|
||
// if a toplevel window was requested without a titlebar, use a dialog windowproc
|
||
if (aInitData->mWindowType == eWindowType_toplevel &&
|
||
(aInitData->mBorderStyle == eBorderStyle_none ||
|
||
aInitData->mBorderStyle != eBorderStyle_all && !(aInitData->mBorderStyle & eBorderStyle_title)))
|
||
windowType = eWindowType_dialog;
|
||
} else
|
||
mWindowType = (mIsDialog ? eWindowType_dialog : eWindowType_toplevel);
|
||
|
||
short wDefProcID = kWindowDocumentProc;
|
||
Boolean goAwayFlag;
|
||
short hOffset;
|
||
short vOffset;
|
||
|
||
switch (mWindowType)
|
||
{
|
||
case eWindowType_popup:
|
||
// We're a popup, context menu, etc. Sets
|
||
// mAcceptsActivation to false so we don't activate the window
|
||
// when we show it.
|
||
mOffsetParent = aParent;
|
||
if( !aParent )
|
||
theToolkit = getter_AddRefs(aParent->GetToolkit());
|
||
|
||
mAcceptsActivation = PR_FALSE;
|
||
goAwayFlag = false;
|
||
hOffset = 0;
|
||
vOffset = 0;
|
||
#if TARGET_CARBON
|
||
wDefProcID = kWindowSimpleProc;
|
||
#else
|
||
wDefProcID = plainDBox;
|
||
#endif
|
||
break;
|
||
|
||
case eWindowType_child:
|
||
wDefProcID = plainDBox;
|
||
goAwayFlag = false;
|
||
hOffset = 0;
|
||
vOffset = 0;
|
||
break;
|
||
|
||
case eWindowType_dialog:
|
||
if (aInitData)
|
||
{
|
||
// Prior to Carbon, defProcs were solely about appearance. If told to create a dialog,
|
||
// we could use, for example, |kWindowMovableModalDialogProc| even if the dialog wasn't
|
||
// supposed to be modal. Carbon breaks this assumption, enforcing modality when using these
|
||
// particular proc ids. As a result, when compiling under Carbon we have to use the
|
||
// standard window proc id's and below, after we have a windowPtr, we'll hide the closebox
|
||
// that comes with the standard window proc.
|
||
//
|
||
// I'm making the assumption here that any dialog created w/out a titlebar is modal and am
|
||
// therefore keeping the old modal dialog proc. I'm only special-casing dialogs with a
|
||
// titlebar since those are the only ones that might end up not being modal.
|
||
|
||
switch (aInitData->mBorderStyle)
|
||
{
|
||
case eBorderStyle_none:
|
||
wDefProcID = kWindowModalDialogProc;
|
||
break;
|
||
|
||
case eBorderStyle_all:
|
||
#if TARGET_CARBON
|
||
wDefProcID = kWindowGrowDocumentProc;
|
||
#else
|
||
wDefProcID = kWindowMovableModalGrowProc; // should we add a close box (kWindowGrowDocumentProc) ?
|
||
#endif
|
||
break;
|
||
|
||
case eBorderStyle_default:
|
||
wDefProcID = kWindowModalDialogProc;
|
||
break;
|
||
|
||
default:
|
||
// we ignore the close flag here, since mac dialogs should never have a close box.
|
||
switch(aInitData->mBorderStyle & (eBorderStyle_resizeh | eBorderStyle_title))
|
||
{
|
||
// combinations of individual options.
|
||
case eBorderStyle_title:
|
||
#if TARGET_CARBON
|
||
wDefProcID = kWindowDocumentProc;
|
||
#else
|
||
wDefProcID = kWindowMovableModalDialogProc;
|
||
#endif
|
||
break;
|
||
|
||
case eBorderStyle_resizeh:
|
||
case (eBorderStyle_title | eBorderStyle_resizeh):
|
||
#if TARGET_CARBON
|
||
wDefProcID = kWindowGrowDocumentProc;
|
||
#else
|
||
wDefProcID = kWindowMovableModalGrowProc; // this is the only kind of resizable dialog.
|
||
#endif
|
||
break;
|
||
|
||
default:
|
||
NS_WARNING("Unhandled combination of window flags");
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
wDefProcID = kWindowModalDialogProc;
|
||
goAwayFlag = true; // revisit this below
|
||
}
|
||
|
||
hOffset = kDialogMarginWidth;
|
||
vOffset = kDialogTitleBarHeight;
|
||
break;
|
||
|
||
case eWindowType_toplevel:
|
||
if (aInitData &&
|
||
aInitData->mBorderStyle != eBorderStyle_all &&
|
||
aInitData->mBorderStyle != eBorderStyle_default &&
|
||
(aInitData->mBorderStyle == eBorderStyle_none ||
|
||
!(aInitData->mBorderStyle & eBorderStyle_resizeh)))
|
||
wDefProcID = kWindowDocumentProc;
|
||
else
|
||
wDefProcID = kWindowFullZoomGrowDocumentProc;
|
||
goAwayFlag = true;
|
||
hOffset = kWindowMarginWidth;
|
||
vOffset = kWindowTitleBarHeight;
|
||
break;
|
||
|
||
case eWindowType_invisible:
|
||
// don't do anything
|
||
break;
|
||
}
|
||
|
||
// now turn off some default features if requested by aInitData
|
||
if (aInitData && aInitData->mBorderStyle != eBorderStyle_all)
|
||
{
|
||
if (aInitData->mBorderStyle == eBorderStyle_none ||
|
||
aInitData->mBorderStyle == eBorderStyle_default &&
|
||
windowType == eWindowType_dialog ||
|
||
!(aInitData->mBorderStyle & eBorderStyle_close))
|
||
goAwayFlag = false;
|
||
}
|
||
|
||
Rect wRect;
|
||
nsRectToMacRect(aRect, wRect);
|
||
|
||
if (eWindowType_popup != mWindowType)
|
||
::OffsetRect(&wRect, hOffset, vOffset + ::GetMBarHeight());
|
||
else
|
||
::OffsetRect(&wRect, hOffset, vOffset);
|
||
|
||
nsCOMPtr<nsIScreenManager> screenmgr = do_GetService(sScreenManagerContractID);
|
||
if (screenmgr) {
|
||
nsCOMPtr<nsIScreen> screen;
|
||
//screenmgr->GetPrimaryScreen(getter_AddRefs(screen));
|
||
screenmgr->ScreenForRect(wRect.left, wRect.top,
|
||
wRect.right - wRect.left, wRect.bottom - wRect.top,
|
||
getter_AddRefs(screen));
|
||
if (screen) {
|
||
PRInt32 left, top, width, height;
|
||
screen->GetAvailRect(&left, &top, &width, &height);
|
||
if (wRect.bottom > top+height) {
|
||
bottomPinDelta = wRect.bottom - (top+height);
|
||
wRect.bottom -= bottomPinDelta;
|
||
}
|
||
}
|
||
}
|
||
mWindowPtr = ::NewCWindow(nil, &wRect, "\p", false, wDefProcID, (WindowRef)-1, goAwayFlag, (long)nsnull);
|
||
mWindowMadeHere = PR_TRUE;
|
||
}
|
||
else
|
||
{
|
||
mWindowPtr = (WindowPtr)aNativeParent;
|
||
mWindowMadeHere = PR_FALSE;
|
||
}
|
||
|
||
if (mWindowPtr == nsnull)
|
||
return NS_ERROR_OUT_OF_MEMORY;
|
||
|
||
nsMacMessageSink::AddRaptorWindowToList(mWindowPtr, this);
|
||
|
||
// create the root control
|
||
ControlHandle rootControl = nil;
|
||
if (GetRootControl(mWindowPtr, &rootControl) != noErr)
|
||
{
|
||
OSErr err = CreateRootControl(mWindowPtr, &rootControl);
|
||
NS_ASSERTION(err == noErr, "Error creating window root control");
|
||
}
|
||
|
||
// reset the coordinates to (0,0) because it's the top level widget
|
||
// and adjust for any adjustment required to requested window bottom
|
||
nsRect bounds(0, 0, aRect.width, aRect.height - bottomPinDelta);
|
||
|
||
// init base class
|
||
// (note: aParent is ignored. Mac (real) windows don't want parents)
|
||
Inherited::StandardCreate(nil, bounds, aHandleEventFunction, aContext, aAppShell, theToolkit, aInitData);
|
||
|
||
#if TARGET_CARBON
|
||
if ( mWindowType == eWindowType_popup ) {
|
||
// OSX enforces window layering so we have to make sure that popups can
|
||
// appear over modal dialogs (at the top of the layering chain). Create
|
||
// the popup like normal and change its window class to the modal layer.
|
||
//
|
||
// XXX This needs to use ::SetWindowGroup() when we move to headers that
|
||
// XXX support it.
|
||
::SetWindowClass(mWindowPtr, kModalWindowClass);
|
||
}
|
||
else if ( mWindowType == eWindowType_dialog ) {
|
||
// Dialogs on mac don't have a close box, but we probably used a defproc above that
|
||
// contains one. Thankfully, carbon lets us turn it off after the window has been
|
||
// created. Do so. We currently leave the collapse widget for all dialogs.
|
||
::ChangeWindowAttributes(mWindowPtr, 0L, kWindowCloseBoxAttribute );
|
||
}
|
||
|
||
// Setup the live window resizing
|
||
if ( mWindowType == eWindowType_toplevel || mWindowType == eWindowType_invisible ) {
|
||
WindowAttributes removeAttributes = kWindowNoAttributes;
|
||
if ( mWindowType == eWindowType_invisible )
|
||
removeAttributes |= kWindowInWindowMenuAttribute;
|
||
::ChangeWindowAttributes ( mWindowPtr, kWindowLiveResizeAttribute, removeAttributes );
|
||
|
||
EventTypeSpec windEventList[] = { {kEventClassWindow, kEventWindowBoundsChanged},
|
||
{kEventClassWindow, kEventWindowConstrain} };
|
||
EventTypeSpec scrollEventList[] = { {kEventClassMouse, kEventMouseWheelMoved} };
|
||
OSStatus err1 = ::InstallWindowEventHandler ( mWindowPtr, NewEventHandlerUPP(WindowEventHandler), 2, windEventList, this, NULL );
|
||
OSStatus err2 = ::InstallWindowEventHandler ( mWindowPtr, NewEventHandlerUPP(ScrollEventHandler), 1, scrollEventList, this, NULL );
|
||
// note, passing NULL as the final param to IWEH() causes the UPP to be disposed automatically
|
||
// when the event target (the window) goes away. See CarbonEvents.h for info.
|
||
|
||
NS_ASSERTION(err1 == noErr && err2 == noErr, "Couldn't install Carbon Event handlers");
|
||
}
|
||
#endif
|
||
|
||
|
||
// register tracking and receive handlers with the native Drag Manager
|
||
if ( mDragTrackingHandlerUPP ) {
|
||
OSErr result = ::InstallTrackingHandler ( mDragTrackingHandlerUPP, mWindowPtr, this );
|
||
NS_ASSERTION ( result == noErr, "can't install drag tracking handler");
|
||
}
|
||
if ( mDragReceiveHandlerUPP ) {
|
||
OSErr result = ::InstallReceiveHandler ( mDragReceiveHandlerUPP, mWindowPtr, this );
|
||
NS_ASSERTION ( result == noErr, "can't install drag receive handler");
|
||
}
|
||
|
||
// If we're a popup, we don't want a border (we want CSS to draw it for us). So
|
||
// install our own window defProc.
|
||
if ( mWindowType == eWindowType_popup )
|
||
InstallBorderlessDefProc(mWindowPtr);
|
||
|
||
#endif
|
||
|
||
return NS_OK;
|
||
}
|
||
|
||
|
||
#if 0
|
||
|
||
pascal OSStatus
|
||
nsCocoaWindow :: ScrollEventHandler ( EventHandlerCallRef inHandlerChain, EventRef inEvent, void* userData )
|
||
{
|
||
EventMouseWheelAxis axis = kEventMouseWheelAxisY;
|
||
SInt32 delta = 0;
|
||
Point mouseLoc;
|
||
OSErr err1 = ::GetEventParameter ( inEvent, kEventParamMouseWheelAxis, typeMouseWheelAxis,
|
||
NULL, sizeof(EventMouseWheelAxis), NULL, &axis );
|
||
OSErr err2 = ::GetEventParameter ( inEvent, kEventParamMouseWheelDelta, typeLongInteger,
|
||
NULL, sizeof(SInt32), NULL, &delta );
|
||
OSErr err3 = ::GetEventParameter ( inEvent, kEventParamMouseLocation, typeQDPoint,
|
||
NULL, sizeof(Point), NULL, &mouseLoc );
|
||
|
||
if ( err1 == noErr && err2 == noErr && err3 == noErr ) {
|
||
nsCocoaWindow* self = NS_REINTERPRET_CAST(nsCocoaWindow*, userData);
|
||
if ( self )
|
||
self->mMacEventHandler->Scroll ( axis, delta, mouseLoc );
|
||
}
|
||
return noErr;
|
||
|
||
} // ScrollEventHandler
|
||
|
||
|
||
pascal OSStatus
|
||
nsCocoaWindow :: WindowEventHandler ( EventHandlerCallRef inHandlerChain, EventRef inEvent, void* userData )
|
||
{
|
||
OSStatus retVal = noErr;
|
||
|
||
WindowRef myWind = NULL;
|
||
::GetEventParameter ( inEvent, kEventParamDirectObject, typeWindowRef, NULL, sizeof(myWind), NULL, &myWind );
|
||
if ( myWind ) {
|
||
UInt32 what = ::GetEventKind ( inEvent );
|
||
switch ( what ) {
|
||
|
||
case kEventWindowBoundsChanged:
|
||
{
|
||
// are we moving or resizing the window? we only care about resize.
|
||
UInt32 attributes = 0;
|
||
::GetEventParameter ( inEvent, kEventParamAttributes, typeUInt32, NULL, sizeof(attributes), NULL, &attributes );
|
||
if ( attributes & kWindowBoundsChangeSizeChanged ) {
|
||
Rect bounds;
|
||
::InvalWindowRect(myWind, ::GetWindowPortBounds(myWind, &bounds));
|
||
|
||
// resize the window and repaint
|
||
nsCocoaWindow* self = NS_REINTERPRET_CAST(nsCocoaWindow*, userData);
|
||
if ( self && !self->mResizeIsFromUs ) {
|
||
self->mMacEventHandler->ResizeEvent(myWind);
|
||
self->mMacEventHandler->UpdateEvent();
|
||
}
|
||
}
|
||
break;
|
||
}
|
||
|
||
case kEventWindowConstrain:
|
||
{
|
||
// Ignore this event if we're an invisible window, otherwise pass along the
|
||
// chain to ensure it's onscreen.
|
||
nsCocoaWindow* self = NS_REINTERPRET_CAST(nsCocoaWindow*, userData);
|
||
if ( self ) {
|
||
if ( self->mWindowType != eWindowType_invisible )
|
||
retVal = ::CallNextEventHandler( inHandlerChain, inEvent );
|
||
}
|
||
break;
|
||
}
|
||
|
||
default:
|
||
// do nothing...
|
||
break;
|
||
|
||
} // case of which event?
|
||
}
|
||
|
||
return retVal;
|
||
|
||
} // WindowEventHandler
|
||
|
||
#endif
|
||
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// Create a nsCocoaWindow using a native window provided by the application
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
NS_IMETHODIMP nsCocoaWindow::Create(nsNativeWidget aNativeParent,
|
||
const nsRect &aRect,
|
||
EVENT_CALLBACK aHandleEventFunction,
|
||
nsIDeviceContext *aContext,
|
||
nsIAppShell *aAppShell,
|
||
nsIToolkit *aToolkit,
|
||
nsWidgetInitData *aInitData)
|
||
{
|
||
return(StandardCreate(nsnull, aRect, aHandleEventFunction,
|
||
aContext, aAppShell, aToolkit, aInitData,
|
||
aNativeParent));
|
||
}
|
||
|
||
|
||
NS_IMETHODIMP nsCocoaWindow::Create(nsIWidget* aParent,
|
||
const nsRect &aRect,
|
||
EVENT_CALLBACK aHandleEventFunction,
|
||
nsIDeviceContext *aContext,
|
||
nsIAppShell *aAppShell,
|
||
nsIToolkit *aToolkit,
|
||
nsWidgetInitData *aInitData)
|
||
{
|
||
return(StandardCreate(aParent, aRect, aHandleEventFunction,
|
||
aContext, aAppShell, aToolkit, aInitData,
|
||
nsnull));
|
||
}
|
||
|
||
|
||
void*
|
||
nsCocoaWindow::GetNativeData(PRUint32 aDataType)
|
||
{
|
||
void* retVal = nsnull;
|
||
|
||
switch ( aDataType ) {
|
||
|
||
// to emulate how windows works, we always have to return a NSView
|
||
// for NS_NATIVE_WIDGET
|
||
case NS_NATIVE_WIDGET:
|
||
case NS_NATIVE_DISPLAY:
|
||
retVal = [mWindow contentView];
|
||
break;
|
||
|
||
case NS_NATIVE_WINDOW:
|
||
retVal = mWindow;
|
||
break;
|
||
|
||
case NS_NATIVE_GRAPHIC: // quickdraw port of top view (for now)
|
||
retVal = [[mWindow contentView] qdPort];
|
||
break;
|
||
|
||
#if 0
|
||
case NS_NATIVE_REGION:
|
||
retVal = (void*)mVisRegion;
|
||
break;
|
||
|
||
case NS_NATIVE_COLORMAP:
|
||
//<2F>TODO
|
||
break;
|
||
|
||
case NS_NATIVE_OFFSETX:
|
||
point.MoveTo(mBounds.x, mBounds.y);
|
||
LocalToWindowCoordinate(point);
|
||
retVal = (void*)point.x;
|
||
break;
|
||
|
||
case NS_NATIVE_OFFSETY:
|
||
point.MoveTo(mBounds.x, mBounds.y);
|
||
LocalToWindowCoordinate(point);
|
||
retVal = (void*)point.y;
|
||
break;
|
||
|
||
case NS_NATIVE_PLUGIN_PORT:
|
||
// this needs to be a combination of the port and the offsets.
|
||
if (mPluginPort == nsnull)
|
||
mPluginPort = new nsPluginPort;
|
||
|
||
point.MoveTo(mBounds.x, mBounds.y);
|
||
LocalToWindowCoordinate(point);
|
||
|
||
// for compatibility with 4.X, this origin is what you'd pass
|
||
// to SetOrigin.
|
||
mPluginPort->port = ::GetWindowPort(mWindowPtr);
|
||
mPluginPort->portx = -point.x;
|
||
mPluginPort->porty = -point.y;
|
||
|
||
retVal = (void*)mPluginPort;
|
||
break;
|
||
#endif
|
||
}
|
||
|
||
return retVal;
|
||
}
|
||
|
||
NS_IMETHODIMP
|
||
nsCocoaWindow::IsVisible(PRBool & aState)
|
||
{
|
||
aState = mVisible;
|
||
return NS_OK;
|
||
}
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// Hide or show this window
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
NS_IMETHODIMP nsCocoaWindow::Show(PRBool bState)
|
||
{
|
||
if ( bState )
|
||
[mWindow orderFront:NULL];
|
||
else
|
||
[mWindow orderOut:NULL];
|
||
|
||
mVisible = bState;
|
||
|
||
#if 0
|
||
// we need to make sure we call ::Show/HideWindow() to generate the
|
||
// necessary activate/deactivate events. Calling ::ShowHide() is
|
||
// not adequate, unless we don't want activation (popups). (pinkerton).
|
||
if ( bState && !mBounds.IsEmpty() ) {
|
||
if ( mAcceptsActivation )
|
||
::ShowWindow(mWindowPtr);
|
||
else {
|
||
::ShowHide(mWindowPtr, true);
|
||
::BringToFront(mWindowPtr); // competes with ComeToFront, but makes popups work
|
||
}
|
||
if (mZoomOnShow) {
|
||
SetSizeMode(nsSizeMode_Maximized);
|
||
mZoomOnShow = PR_FALSE;
|
||
}
|
||
ComeToFront();
|
||
}
|
||
else {
|
||
// when a toplevel window goes away, make sure we rollup any popups that may
|
||
// be lurking. We want to catch this here because we're guaranteed that
|
||
// we hide a window before we destroy it, and doing it here more closely
|
||
// approximates where we do the same thing on windows.
|
||
if ( mWindowType == eWindowType_toplevel ) {
|
||
if ( gRollupListener )
|
||
gRollupListener->Rollup();
|
||
NS_IF_RELEASE(gRollupListener);
|
||
NS_IF_RELEASE(gRollupWidget);
|
||
}
|
||
::HideWindow(mWindowPtr);
|
||
}
|
||
|
||
#endif
|
||
|
||
return NS_OK;
|
||
}
|
||
|
||
|
||
NS_IMETHODIMP nsCocoaWindow::Enable(PRBool aState)
|
||
{
|
||
return NS_OK;
|
||
}
|
||
|
||
|
||
NS_IMETHODIMP nsCocoaWindow::IsEnabled(PRBool *aState)
|
||
{
|
||
if (aState)
|
||
*aState = PR_TRUE;
|
||
return NS_OK;
|
||
}
|
||
|
||
|
||
/*
|
||
NS_METHOD nsWindow::Minimize(void)
|
||
{
|
||
return NS_OK;
|
||
}
|
||
|
||
NS_METHOD nsWindow::Maximize(void)
|
||
{
|
||
return NS_OK;
|
||
}
|
||
|
||
NS_METHOD nsWindow::Restore(void)
|
||
{
|
||
return NS_OK;
|
||
}
|
||
*/
|
||
|
||
NS_IMETHODIMP nsCocoaWindow::ConstrainPosition(PRBool aAllowSlop,
|
||
PRInt32 *aX, PRInt32 *aY)
|
||
{
|
||
#if 0
|
||
if (eWindowType_popup == mWindowType || !mWindowMadeHere)
|
||
return NS_OK;
|
||
|
||
// Sanity check against screen size
|
||
// make sure the window stays visible
|
||
|
||
// get the window bounds
|
||
Rect portBounds;
|
||
::GetWindowPortBounds(mWindowPtr, &portBounds);
|
||
short pos;
|
||
short windowWidth = portBounds.right - portBounds.left;
|
||
short windowHeight = portBounds.bottom - portBounds.top;
|
||
|
||
// now get our playing field. use the current screen, or failing that for any reason,
|
||
// the GrayRgn (which of course is arguably more correct but has drawbacks as well)
|
||
Rect screenRect;
|
||
nsCOMPtr<nsIScreenManager> screenmgr = do_GetService(sScreenManagerContractID);
|
||
if (screenmgr) {
|
||
nsCOMPtr<nsIScreen> screen;
|
||
PRInt32 left, top, width, height, fullHeight;
|
||
|
||
// zero size rects can happen during window creation, and confuse
|
||
// the screen manager
|
||
width = windowWidth > 0 ? windowWidth : 1;
|
||
height = windowHeight > 0 ? windowHeight : 1;
|
||
screenmgr->ScreenForRect(*aX, *aY, width, height,
|
||
getter_AddRefs(screen));
|
||
if (screen) {
|
||
screen->GetAvailRect(&left, &top, &width, &height);
|
||
screen->GetRect(&left, &top, &width, &fullHeight);
|
||
screenRect.left = left;
|
||
screenRect.right = left+width;
|
||
screenRect.top = top;
|
||
screenRect.bottom = top+height;
|
||
}
|
||
} else
|
||
::GetRegionBounds(::GetGrayRgn(), &screenRect);
|
||
|
||
pos = screenRect.left;
|
||
if (windowWidth > kWindowPositionSlop)
|
||
pos -= windowWidth - kWindowPositionSlop;
|
||
if (*aX < pos)
|
||
*aX = pos;
|
||
else if (*aX >= screenRect.right - kWindowPositionSlop)
|
||
*aX = screenRect.right - kWindowPositionSlop;
|
||
|
||
pos = screenRect.top;
|
||
if (windowHeight > kWindowPositionSlop)
|
||
pos -= windowHeight - kWindowPositionSlop;
|
||
if (*aY < pos)
|
||
*aY = pos;
|
||
else if (*aY >= screenRect.bottom - kWindowPositionSlop)
|
||
*aY = screenRect.bottom - kWindowPositionSlop;
|
||
|
||
#endif
|
||
return NS_OK;
|
||
}
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// Move this window
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
//-------------------------------------------------------------------------
|
||
// Move
|
||
//-------------------------------------------------------------------------
|
||
NS_IMETHODIMP nsCocoaWindow::Move(PRInt32 aX, PRInt32 aY)
|
||
{
|
||
if ( mWindow ) {
|
||
// if we're a popup, we have to convert from our parent widget's coord
|
||
// system to the global coord system first because the (x,y) we're given
|
||
// is in its coordinate system.
|
||
if ( mWindowType == eWindowType_popup ) {
|
||
nsRect localRect, globalRect;
|
||
localRect.x = aX;
|
||
localRect.y = aY;
|
||
if ( mOffsetParent ) {
|
||
mOffsetParent->WidgetToScreen(localRect,globalRect);
|
||
aX=globalRect.x;
|
||
aY=globalRect.y;
|
||
}
|
||
}
|
||
|
||
NSPoint coord = {aX, aY};
|
||
//coord = [mWindow convertBaseToScreen:coord];
|
||
//printf("moving to %d %d. screen coords %f %f\n", aX, aY, coord.x, coord.y);
|
||
|
||
//FIXME -- ensure it's on the screen. Cocoa automatically corrects for windows
|
||
// with title bars, but for other windows, we have to do this...
|
||
|
||
// the point we have assumes that the screen origin is the top-left. Well,
|
||
// it's not. Use the screen object to convert.
|
||
//FIXME -- doesn't work on monitors other than primary
|
||
NSRect screenRect = [[NSScreen mainScreen] frame];
|
||
coord.y = (screenRect.origin.y + screenRect.size.height) - coord.y;
|
||
//printf("final coords %f %f\n", coord.x, coord.y);
|
||
//printf("- window coords before %f %f\n", [mWindow frame].origin.x, [mWindow frame].origin.y);
|
||
[mWindow setFrameTopLeftPoint:coord];
|
||
//printf("- window coords after %f %f\n", [mWindow frame].origin.x, [mWindow frame].origin.y);
|
||
}
|
||
|
||
#if 0
|
||
StPortSetter setOurPortForLocalToGlobal ( mWindowPtr );
|
||
|
||
if (eWindowType_popup == mWindowType) {
|
||
PRInt32 xOffset=0,yOffset=0;
|
||
nsRect localRect,globalRect;
|
||
|
||
// convert to screen coordinates
|
||
localRect.x = aX;
|
||
localRect.y = aY;
|
||
localRect.width = 100;
|
||
localRect.height = 100;
|
||
|
||
if ( mOffsetParent ) {
|
||
mOffsetParent->WidgetToScreen(localRect,globalRect);
|
||
aX=globalRect.x;
|
||
aY=globalRect.y;
|
||
|
||
// there is a bug on OSX where if we call ::MoveWindow() with the same
|
||
// coordinates (within a pixel or two) as a window's current location, it will
|
||
// move to (0,0,-1,-1). The fix is to not move the window if we're already
|
||
// there. (radar# 2669004)
|
||
#if TARGET_CARBON
|
||
const PRInt32 kMoveThreshold = 2;
|
||
#else
|
||
const PRInt32 kMoveThreshold = 0;
|
||
#endif
|
||
Rect currBounds;
|
||
::GetWindowBounds ( mWindowPtr, kWindowGlobalPortRgn, &currBounds );
|
||
if ( abs(currBounds.left-aX) > kMoveThreshold || abs(currBounds.top-aY) > kMoveThreshold ) {
|
||
::MoveWindow(mWindowPtr, aX, aY, false);
|
||
|
||
Rect newBounds;
|
||
::GetWindowBounds ( mWindowPtr, kWindowGlobalPortRgn, &newBounds );
|
||
}
|
||
}
|
||
|
||
return NS_OK;
|
||
} else if (mWindowMadeHere) {
|
||
Rect portBounds;
|
||
::GetWindowPortBounds(mWindowPtr, &portBounds);
|
||
|
||
if (mIsDialog) {
|
||
aX += kDialogMarginWidth;
|
||
aY += kDialogTitleBarHeight;
|
||
} else {
|
||
aX += kWindowMarginWidth;
|
||
aY += kWindowTitleBarHeight;
|
||
}
|
||
|
||
nsCOMPtr<nsIScreenManager> screenmgr = do_GetService(sScreenManagerContractID);
|
||
if (screenmgr) {
|
||
nsCOMPtr<nsIScreen> screen;
|
||
PRInt32 left, top, width, height, fullTop;
|
||
// adjust for unset bounds, which confuses the screen manager
|
||
width = portBounds.right - portBounds.left;
|
||
height = portBounds.bottom - portBounds.top;
|
||
if (height <= 0) height = 1;
|
||
if (width <= 0) width = 1;
|
||
|
||
screenmgr->ScreenForRect(aX, aY, width, height,
|
||
getter_AddRefs(screen));
|
||
if (screen) {
|
||
screen->GetAvailRect(&left, &top, &width, &height);
|
||
screen->GetRect(&left, &fullTop, &width, &height);
|
||
aY += top-fullTop;
|
||
}
|
||
}
|
||
|
||
// move the window if it has not been moved yet
|
||
// (ie. if this function isn't called in response to a DragWindow event)
|
||
Point macPoint = topLeft(portBounds);
|
||
::LocalToGlobal(&macPoint);
|
||
if (macPoint.h != aX || macPoint.v != aY)
|
||
::MoveWindow(mWindowPtr, aX, aY, false);
|
||
|
||
// propagate the event in global coordinates
|
||
Inherited::Move(aX, aY);
|
||
|
||
// reset the coordinates to (0,0) because it's the top level widget
|
||
mBounds.x = 0;
|
||
mBounds.y = 0;
|
||
}
|
||
#endif
|
||
return NS_OK;
|
||
}
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// Position the window behind the given window
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
NS_METHOD nsCocoaWindow::PlaceBehind(nsIWidget *aWidget, PRBool aActivate)
|
||
{
|
||
#if 0
|
||
if (aWidget) {
|
||
WindowPtr behind = (WindowPtr)aWidget->GetNativeData(NS_NATIVE_DISPLAY);
|
||
::SendBehind(mWindowPtr, behind);
|
||
::HiliteWindow(mWindowPtr, FALSE);
|
||
} else {
|
||
if (::FrontWindow() != mWindowPtr)
|
||
::SelectWindow(mWindowPtr);
|
||
}
|
||
#endif
|
||
return NS_OK;
|
||
}
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// zoom/restore
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
NS_METHOD nsCocoaWindow::SetSizeMode(PRInt32 aMode)
|
||
{
|
||
#if 0
|
||
nsresult rv;
|
||
PRInt32 currentMode;
|
||
|
||
if (aMode == nsSizeMode_Minimized) // unlikely on the Mac
|
||
return NS_ERROR_UNEXPECTED;
|
||
|
||
// already done? it's bad to rezoom a window, so do nothing
|
||
rv = nsBaseWidget::GetSizeMode(¤tMode);
|
||
if (NS_SUCCEEDED(rv) && currentMode == aMode)
|
||
return NS_OK;
|
||
|
||
if (!mVisible) {
|
||
/* zooming on the Mac doesn't seem to work until the window is visible.
|
||
the rest of the app is structured to zoom before the window is visible
|
||
to avoid flashing. here's where we defeat that. */
|
||
if (aMode == nsSizeMode_Maximized)
|
||
mZoomOnShow = PR_TRUE;
|
||
} else {
|
||
Rect macRect;
|
||
rv = nsBaseWidget::SetSizeMode(aMode);
|
||
if (NS_SUCCEEDED(rv)) {
|
||
if (aMode == nsSizeMode_Maximized) {
|
||
CalculateAndSetZoomedSize();
|
||
::ZoomWindow(mWindowPtr, inZoomOut, ::FrontWindow() == mWindowPtr);
|
||
} else
|
||
::ZoomWindow(mWindowPtr, inZoomIn, ::FrontWindow() == mWindowPtr);
|
||
::GetWindowPortBounds(mWindowPtr, &macRect);
|
||
Resize(macRect.right - macRect.left, macRect.bottom - macRect.top, PR_FALSE);
|
||
}
|
||
}
|
||
#endif
|
||
|
||
return NS_OK;
|
||
}
|
||
|
||
void nsCocoaWindow::CalculateAndSetZoomedSize()
|
||
{
|
||
#if 0
|
||
StPortSetter setOurPort(mWindowPtr);
|
||
|
||
// calculate current window portbounds
|
||
Rect windRect;
|
||
::GetWindowPortBounds(mWindowPtr, &windRect);
|
||
::LocalToGlobal((Point *)&windRect.top);
|
||
::LocalToGlobal((Point *)&windRect.bottom);
|
||
|
||
// calculate window's borders on each side, these differ in Aqua / Platinum
|
||
short wTitleHeight;
|
||
short wLeftBorder;
|
||
short wRightBorder;
|
||
short wBottomBorder;
|
||
|
||
RgnHandle structRgn = ::NewRgn();
|
||
::GetWindowRegion(mWindowPtr, kWindowStructureRgn, structRgn);
|
||
Rect structRgnBounds;
|
||
::GetRegionBounds(structRgn, &structRgnBounds);
|
||
wTitleHeight = windRect.top - structRgnBounds.top;
|
||
wLeftBorder = windRect.left - structRgnBounds.left;
|
||
wRightBorder = structRgnBounds.right - windRect.right;
|
||
wBottomBorder = structRgnBounds.bottom - windRect.bottom;
|
||
|
||
::DisposeRgn(structRgn);
|
||
|
||
windRect.top -= wTitleHeight;
|
||
windRect.bottom += wBottomBorder;
|
||
windRect.right += wRightBorder;
|
||
windRect.left -= wLeftBorder;
|
||
|
||
// find which screen the window is (mostly) on and get its rect. GetAvailRect()
|
||
// handles subtracting out the menubar and the dock for us. Set the zoom rect
|
||
// to the screen rect, less some fudging and room for icons on the primary screen.
|
||
nsCOMPtr<nsIScreenManager> screenMgr = do_GetService(sScreenManagerContractID);
|
||
if ( screenMgr ) {
|
||
nsCOMPtr<nsIScreen> screen;
|
||
screenMgr->ScreenForRect ( windRect.left, windRect.top, windRect.right - windRect.left, windRect.bottom - windRect.top,
|
||
getter_AddRefs(screen) );
|
||
if ( screen ) {
|
||
nsRect newWindowRect;
|
||
screen->GetAvailRect ( &newWindowRect.x, &newWindowRect.y, &newWindowRect.width, &newWindowRect.height );
|
||
|
||
// leave room for icons on primary screen
|
||
nsCOMPtr<nsIScreen> primaryScreen;
|
||
screenMgr->GetPrimaryScreen ( getter_AddRefs(primaryScreen) );
|
||
if (screen == primaryScreen) {
|
||
int iconSpace = 96;
|
||
#if TARGET_CARBON
|
||
if(::OnMacOSX()) {
|
||
iconSpace = 128; //icons/grid is wider on Mac OS X
|
||
}
|
||
#endif
|
||
newWindowRect.width -= iconSpace;
|
||
}
|
||
|
||
Rect zoomRect;
|
||
::SetRect(&zoomRect,
|
||
newWindowRect.x + wLeftBorder,
|
||
newWindowRect.y + wTitleHeight,
|
||
newWindowRect.x + newWindowRect.width - wRightBorder,
|
||
newWindowRect.y + newWindowRect.height - wBottomBorder);
|
||
::SetWindowStandardState ( mWindowPtr, &zoomRect );
|
||
}
|
||
}
|
||
#endif
|
||
|
||
} // CalculateAndSetZoomedSize
|
||
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// Resize this window to a point given in global (screen) coordinates. This
|
||
// differs from simple Move(): that method makes JavaScript place windows
|
||
// like other browsers: it puts the top-left corner of the outer edge of the
|
||
// window border at the given coordinates, offset from the menubar.
|
||
// MoveToGlobalPoint expects the top-left corner of the portrect, which
|
||
// is inside the border, and is not offset by the menubar height.
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
void nsCocoaWindow::MoveToGlobalPoint(PRInt32 aX, PRInt32 aY)
|
||
{
|
||
#if 0
|
||
PRInt32 left, top, width, height, fullTop;
|
||
Rect portBounds;
|
||
|
||
StPortSetter doThatThingYouDo(mWindowPtr);
|
||
::GetWindowPortBounds(mWindowPtr, &portBounds);
|
||
|
||
width = portBounds.right - portBounds.left;
|
||
height = portBounds.bottom - portBounds.top;
|
||
::LocalToGlobal(&topLeft(portBounds));
|
||
|
||
nsCOMPtr<nsIScreenManager> screenmgr = do_GetService(sScreenManagerContractID);
|
||
if (screenmgr) {
|
||
nsCOMPtr<nsIScreen> screen;
|
||
//screenmgr->GetPrimaryScreen(getter_AddRefs(screen));
|
||
screenmgr->ScreenForRect(portBounds.left, portBounds.top, width, height,
|
||
getter_AddRefs(screen));
|
||
if (screen) {
|
||
screen->GetAvailRect(&left, &top, &width, &height);
|
||
screen->GetRect(&left, &fullTop, &width, &height);
|
||
aY -= top-fullTop;
|
||
}
|
||
}
|
||
|
||
if (mIsDialog) {
|
||
aX -= kDialogMarginWidth;
|
||
aY -= kDialogTitleBarHeight;
|
||
} else {
|
||
aX -= kWindowMarginWidth;
|
||
aY -= kWindowTitleBarHeight;
|
||
}
|
||
Move(aX, aY);
|
||
#endif
|
||
}
|
||
|
||
|
||
NS_IMETHODIMP nsCocoaWindow::Resize(PRInt32 aX, PRInt32 aY, PRInt32 aWidth, PRInt32 aHeight, PRBool aRepaint)
|
||
{
|
||
Move(aX, aY);
|
||
Resize(aWidth, aHeight, aRepaint);
|
||
return NS_OK;
|
||
}
|
||
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// Resize this window
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
NS_IMETHODIMP nsCocoaWindow::Resize(PRInt32 aWidth, PRInt32 aHeight, PRBool aRepaint)
|
||
{
|
||
printf("resizing to %d %d\n", aWidth, aHeight);
|
||
if ( mWindow ) {
|
||
NSRect newBounds = [mWindow frame];
|
||
newBounds.size.width = aWidth;
|
||
if ( mWindowType == eWindowType_popup )
|
||
newBounds.size.height = aHeight;
|
||
else
|
||
newBounds.size.height = aHeight + kTitleBarHeight; // add height of title bar
|
||
StartResizing();
|
||
[mWindow setFrame:newBounds display:NO];
|
||
StopResizing();
|
||
}
|
||
|
||
mBounds.width = aWidth;
|
||
mBounds.height = aHeight;
|
||
|
||
// tell gecko to update all the child widgets
|
||
ReportSizeEvent();
|
||
|
||
#if 0
|
||
if (mWindowMadeHere) {
|
||
// Sanity check against screen size
|
||
Rect screenRect;
|
||
::GetRegionBounds(::GetGrayRgn(), &screenRect);
|
||
|
||
// Need to use non-negative coordinates
|
||
PRInt32 screenWidth;
|
||
if(screenRect.left < 0)
|
||
screenWidth = screenRect.right - screenRect.left;
|
||
else
|
||
screenWidth = screenRect.right;
|
||
|
||
PRInt32 screenHeight;
|
||
if(screenRect.top < 0)
|
||
screenHeight = screenRect.bottom - screenRect.top;
|
||
else
|
||
screenHeight = screenRect.bottom;
|
||
|
||
if(aHeight > screenHeight)
|
||
aHeight = screenHeight;
|
||
|
||
if(aWidth > screenWidth)
|
||
aWidth = screenWidth;
|
||
|
||
Rect macRect;
|
||
::GetWindowPortBounds ( mWindowPtr, &macRect );
|
||
|
||
short w = macRect.right - macRect.left;
|
||
short h = macRect.bottom - macRect.top;
|
||
Boolean needReposition = (w == 1 && h == 1);
|
||
|
||
if ((w != aWidth) || (h != aHeight))
|
||
{
|
||
// make sure that we don't infinitely recurse if live-resize is on
|
||
mResizeIsFromUs = PR_TRUE;
|
||
::SizeWindow(mWindowPtr, aWidth, aHeight, aRepaint);
|
||
mResizeIsFromUs = PR_FALSE;
|
||
|
||
#if defined(XP_MACOSX)
|
||
// workaround for bug in MacOSX if windows start life as 1x1.
|
||
if (needReposition)
|
||
RepositionWindow(mWindowPtr, NULL, kWindowCascadeOnMainScreen);
|
||
#endif
|
||
}
|
||
}
|
||
#endif
|
||
//Inherited::Resize(aWidth, aHeight, aRepaint);
|
||
return NS_OK;
|
||
}
|
||
|
||
NS_IMETHODIMP nsCocoaWindow::GetScreenBounds(nsRect &aRect) {
|
||
#if 0
|
||
nsRect localBounds;
|
||
PRInt32 yAdjust = 0;
|
||
|
||
GetBounds(localBounds);
|
||
// nsCocoaWindow local bounds are always supposed to be local (0,0) but in the middle of a move
|
||
// can be global. This next adjustment assures they are in local coordinates, even then.
|
||
localBounds.MoveBy(-localBounds.x, -localBounds.y);
|
||
WidgetToScreen(localBounds, aRect);
|
||
|
||
nsCOMPtr<nsIScreenManager> screenmgr = do_GetService(sScreenManagerContractID);
|
||
if (screenmgr) {
|
||
nsCOMPtr<nsIScreen> screen;
|
||
//screenmgr->GetPrimaryScreen(getter_AddRefs(screen));
|
||
screenmgr->ScreenForRect(aRect.x, aRect.y, aRect.width, aRect.height,
|
||
getter_AddRefs(screen));
|
||
if (screen) {
|
||
PRInt32 left, top, width, height, fullTop;
|
||
screen->GetAvailRect(&left, &top, &width, &height);
|
||
screen->GetRect(&left, &fullTop, &width, &height);
|
||
yAdjust = top-fullTop;
|
||
}
|
||
}
|
||
|
||
if (mIsDialog)
|
||
aRect.MoveBy(-kDialogMarginWidth, -kDialogTitleBarHeight-yAdjust);
|
||
else
|
||
aRect.MoveBy(-kWindowMarginWidth, -kWindowTitleBarHeight-yAdjust);
|
||
|
||
#endif
|
||
return NS_OK;
|
||
}
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
PRBool nsCocoaWindow::OnPaint(nsPaintEvent &event)
|
||
{
|
||
return PR_TRUE; // don't dispatch the update event
|
||
}
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// Set this window's title
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
NS_IMETHODIMP nsCocoaWindow::SetTitle(const nsString& aTitle)
|
||
{
|
||
NSString* title = [NSString stringWithCharacters:aTitle.get() length:aTitle.Length()];
|
||
[mWindow setTitle:title];
|
||
|
||
return NS_OK;
|
||
}
|
||
|
||
|
||
//-------------------------------------------------------------------------
|
||
// Pass notification of some drag event to Gecko
|
||
//
|
||
// The drag manager has let us know that something related to a drag has
|
||
// occurred in this window. It could be any number of things, ranging from
|
||
// a drop, to a drag enter/leave, or a drag over event. The actual event
|
||
// is passed in |aMessage| and is passed along to our event hanlder so Gecko
|
||
// knows about it.
|
||
//-------------------------------------------------------------------------
|
||
PRBool nsCocoaWindow::DragEvent ( unsigned int aMessage, Point aMouseGlobal, UInt16 aKeyModifiers )
|
||
{
|
||
#if 0
|
||
PRBool retVal;
|
||
if (mMacEventHandler.get())
|
||
retVal = mMacEventHandler->DragEvent(aMessage, aMouseGlobal, aKeyModifiers);
|
||
else
|
||
retVal = PR_FALSE;
|
||
return retVal;
|
||
#endif
|
||
return PR_FALSE;
|
||
}
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// Like ::BringToFront, but constrains the window to its z-level
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
void nsCocoaWindow::ComeToFront() {
|
||
#if 0
|
||
nsZLevelEvent event;
|
||
|
||
event.point.x = mBounds.x;
|
||
event.point.y = mBounds.y;
|
||
event.time = PR_IntervalNow();
|
||
event.widget = this;
|
||
event.nativeMsg = nsnull;
|
||
event.eventStructType = NS_ZLEVEL_EVENT;
|
||
event.message = NS_SETZLEVEL;
|
||
|
||
event.mPlacement = nsWindowZTop;
|
||
event.mReqBelow = 0;
|
||
event.mImmediate = PR_TRUE;
|
||
event.mAdjusted = PR_FALSE;
|
||
|
||
DispatchWindowEvent(event);
|
||
#endif
|
||
}
|
||
|
||
|
||
NS_IMETHODIMP nsCocoaWindow::ResetInputState()
|
||
{
|
||
// return mMacEventHandler->ResetInputState();
|
||
return NS_OK;
|
||
}
|
||
|
||
void nsCocoaWindow::SetIsActive(PRBool aActive)
|
||
{
|
||
// mIsActive = aActive;
|
||
}
|
||
|
||
void nsCocoaWindow::IsActive(PRBool* aActive)
|
||
{
|
||
// *aActive = mIsActive;
|
||
}
|
||
|
||
|
||
//
|
||
// Return true if we are on Mac OS X, caching the result after the first call
|
||
//
|
||
static PRBool
|
||
OnMacOSX()
|
||
{
|
||
|
||
static PRBool gInitVer = PR_FALSE;
|
||
static PRBool gOnMacOSX = PR_FALSE;
|
||
if(! gInitVer) {
|
||
long version;
|
||
OSErr err = ::Gestalt(gestaltSystemVersion, &version);
|
||
gOnMacOSX = (err == noErr && version >= 0x00001000);
|
||
gInitVer = PR_TRUE;
|
||
}
|
||
return gOnMacOSX;
|
||
}
|
||
|
||
|
||
#if 0
|
||
|
||
//
|
||
// DispatchEvent
|
||
//
|
||
//
|
||
NS_IMETHODIMP
|
||
nsCocoaWindow::DispatchEvent ( void* anEvent, void* aView, PRBool *_retval )
|
||
{
|
||
*_retval = PR_FALSE;
|
||
|
||
#if 0
|
||
NSEvent* event = NS_REINTERPRET_CAST(NSEvent*, anEvent);
|
||
NS_ASSERTION(event, "null event");
|
||
|
||
ChildView* view = NS_REINTERPRET_CAST(ChildView*, aView);
|
||
|
||
nsMouseEvent geckoEvent;
|
||
geckoEvent.eventStructType = NS_MOUSE_EVENT;
|
||
|
||
geckoEvent.widget = this;
|
||
geckoEvent.nativeMsg = anEvent;
|
||
geckoEvent.time = PR_IntervalNow();
|
||
NSPoint mouseLoc = [event locationInWindow];
|
||
geckoEvent.refPoint.x = NS_STATIC_CAST(nscoord, mouseLoc.x);
|
||
geckoEvent.refPoint.y = NS_STATIC_CAST(nscoord, mouseLoc.y);
|
||
//printf("-- global mouse click at (%ld,%ld)\n", geckoEvent.refPoint.x, geckoEvent.refPoint.y );
|
||
if ( view ) {
|
||
geckoEvent.widget = [view widget];
|
||
|
||
// convert point to view coordinate system
|
||
NSPoint localPoint = [view convertPoint:mouseLoc fromView:nil];
|
||
geckoEvent.point.x = NS_STATIC_CAST(nscoord, localPoint.x);
|
||
geckoEvent.point.y = NS_STATIC_CAST(nscoord, localPoint.y);
|
||
//printf("-- local mouse click at (%ld,%ld) widget = %ld\n", geckoEvent.point.x, geckoEvent.point.y,
|
||
// geckoEvent.widget );
|
||
}
|
||
else
|
||
geckoEvent.point = geckoEvent.refPoint;
|
||
|
||
NSEventType type = [event type];
|
||
switch ( type ) {
|
||
case NSLeftMouseDown:
|
||
//printf("left mouse down\n");
|
||
geckoEvent.message = NS_MOUSE_LEFT_BUTTON_DOWN;
|
||
geckoEvent.clickCount = [event clickCount];
|
||
break;
|
||
case NSLeftMouseUp:
|
||
//printf("left mouse up\n");
|
||
break;
|
||
case NSMouseMoved:
|
||
printf("mouse move\n");
|
||
break;
|
||
case NSKeyDown:
|
||
printf("key down\n");
|
||
break;
|
||
case NSKeyUp:
|
||
printf("key up\n");
|
||
break;
|
||
case NSScrollWheel:
|
||
printf("scroll wheel\n");
|
||
break;
|
||
case NSLeftMouseDragged:
|
||
printf("drag\n");
|
||
break;
|
||
|
||
default:
|
||
printf("other\n");
|
||
break;
|
||
}
|
||
|
||
nsEventStatus status = nsEventStatus_eIgnore;
|
||
DispatchEvent ( &geckoEvent, status );
|
||
|
||
*_retval = PR_TRUE;
|
||
#endif
|
||
|
||
return NS_OK;
|
||
}
|
||
|
||
#endif
|
||
|
||
|
||
//-------------------------------------------------------------------------
|
||
//
|
||
// Invokes callback and ProcessEvent method on Event Listener object
|
||
//
|
||
//-------------------------------------------------------------------------
|
||
NS_IMETHODIMP
|
||
nsCocoaWindow::DispatchEvent(nsGUIEvent* event, nsEventStatus& aStatus)
|
||
{
|
||
aStatus = nsEventStatus_eIgnore;
|
||
|
||
nsIWidget* aWidget = event->widget;
|
||
NS_IF_ADDREF(aWidget);
|
||
|
||
if (nsnull != mMenuListener){
|
||
if(NS_MENU_EVENT == event->eventStructType)
|
||
aStatus = mMenuListener->MenuSelected( static_cast<nsMenuEvent&>(*event) );
|
||
}
|
||
if (mEventCallback)
|
||
aStatus = (*mEventCallback)(event);
|
||
|
||
// Dispatch to event listener if event was not consumed
|
||
if ((aStatus != nsEventStatus_eConsumeNoDefault) && (mEventListener != nsnull))
|
||
aStatus = mEventListener->ProcessEvent(*event);
|
||
|
||
NS_IF_RELEASE(aWidget);
|
||
|
||
return NS_OK;
|
||
}
|
||
|
||
|
||
void
|
||
nsCocoaWindow::ReportSizeEvent()
|
||
{
|
||
// nsEvent
|
||
nsSizeEvent sizeEvent;
|
||
sizeEvent.eventStructType = NS_SIZE_EVENT;
|
||
sizeEvent.message = NS_SIZE;
|
||
sizeEvent.point.x = 0;
|
||
sizeEvent.point.y = 0;
|
||
sizeEvent.time = PR_IntervalNow();
|
||
|
||
// nsGUIEvent
|
||
sizeEvent.widget = this;
|
||
sizeEvent.nativeMsg = nsnull;
|
||
|
||
// nsSizeEvent
|
||
sizeEvent.windowSize = &mBounds;
|
||
sizeEvent.mWinWidth = mBounds.width;
|
||
sizeEvent.mWinHeight = mBounds.height;
|
||
|
||
// dispatch event
|
||
nsEventStatus status = nsEventStatus_eIgnore;
|
||
DispatchEvent(&sizeEvent, status);
|
||
}
|
||
|
||
|
||
#if 0
|
||
|
||
PRBool
|
||
nsCocoaWindow::IsResizing ( ) const
|
||
{
|
||
return mIsResizing;
|
||
}
|
||
|
||
void StartResizing ( )
|
||
{
|
||
mIsResizing = PR_TRUE;
|
||
}
|
||
|
||
void StopResizing ( )
|
||
{
|
||
mIsResizing = PR_FALSE;
|
||
}
|
||
|
||
#endif
|
||
|
||
|
||
#pragma mark -
|
||
|
||
|
||
@implementation WindowDelegate
|
||
|
||
|
||
- (id)initWithGeckoWindow:(nsCocoaWindow*)geckoWind
|
||
{
|
||
[super init];
|
||
mGeckoWindow = geckoWind;
|
||
return self;
|
||
}
|
||
|
||
- (void)windowDidResize:(NSNotification *)aNotification
|
||
{
|
||
if ( !mGeckoWindow->IsResizing() ) {
|
||
// must remember to give Gecko top-left, not straight cocoa origin
|
||
// and that Gecko already compensates for the title bar, so we have to
|
||
// strip it out here.
|
||
NSRect frameRect = [[aNotification object] frame];
|
||
mGeckoWindow->Resize ( NS_STATIC_CAST(PRInt32,frameRect.size.width),
|
||
NS_STATIC_CAST(PRInt32,frameRect.size.height - nsCocoaWindow::kTitleBarHeight), PR_TRUE );
|
||
}
|
||
}
|
||
|
||
|
||
- (void)windowDidBecomeMain:(NSNotification *)aNotification
|
||
{
|
||
printf("got activation\n");
|
||
}
|
||
|
||
|
||
- (void)windowDidResignMain:(NSNotification *)aNotification
|
||
{
|
||
printf("got deactivate\n");
|
||
}
|
||
|
||
|
||
- (void)windowDidBecomeKey:(NSNotification *)aNotification
|
||
{
|
||
printf("we're key window\n");
|
||
}
|
||
|
||
|
||
- (void)windowDidResignKey:(NSNotification *)aNotification
|
||
{
|
||
printf("we're not the key window\n");
|
||
}
|
||
|
||
|
||
- (void)windowDidMove:(NSNotification *)aNotification
|
||
{
|
||
}
|
||
|
||
@end
|