Mozilla/mozilla/widget/src/mac/nsAppShell.cpp
pavlov%netscape.com 7f3fe63a8b remove GetNativeData since it isn't used r=shaver (for the entire checkin)
git-svn-id: svn://10.0.0.236/trunk@49555 18797224-902f-48f8-a5cc-f745e15eee43
1999-10-01 08:33:41 +00:00

316 lines
9.0 KiB
C++

/* -*- Mode: c++; tab-width: 2; indent-tabs-mode: nil; -*- */
/*
* The contents of this file are subject to the Netscape Public License
* Version 1.0 (the "NPL"); you may not use this file except in
* compliance with the NPL. You may obtain a copy of the NPL at
* http://www.mozilla.org/NPL/
*
* Software distributed under the NPL is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
* for the specific language governing rights and limitations under the
* NPL.
*
* The Initial Developer of this code under the NPL is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
* Reserved.
*/
//
// nsAppShell
//
// This file contains the default implementation of the application shell. Clients
// may either use this implementation or write their own. If you write your
// own, you must create a message sink to route events to. (The message sink
// interface may change, so this comment must be updated accordingly.)
//
#include "nsAppShell.h"
#include "nsIAppShell.h"
#include "nsIEventQueueService.h"
#include "nsIServiceManager.h"
#include "nsIWidget.h"
#include "nsMacMessageSink.h"
#include "nsMacMessagePump.h"
#include "nsToolKit.h"
#include <Quickdraw.h>
#include <Fonts.h>
#include <TextEdit.h>
#include <Dialogs.h>
#include <Traps.h>
#include <Events.h>
#include <Menus.h>
#include <stdlib.h>
#include "macstdlibextras.h"
PRBool nsAppShell::mInitializedToolbox = PR_FALSE;
//-------------------------------------------------------------------------
//
// nsISupports implementation macro
//
//-------------------------------------------------------------------------
NS_DEFINE_IID(kIAppShellIID, NS_IAPPSHELL_IID);
NS_DEFINE_CID(kEventQueueServiceCID, NS_EVENTQUEUESERVICE_CID);
NS_IMPL_ISUPPORTS(nsAppShell,kIAppShellIID);
NS_IMETHODIMP nsAppShell::SetDispatchListener(nsDispatchListener* aDispatchListener)
{
mDispatchListener = aDispatchListener;
return NS_OK;
}
//-------------------------------------------------------------------------
//
// Create the application shell
//
//-------------------------------------------------------------------------
NS_IMETHODIMP nsAppShell::Create(int* argc, char ** argv)
{
mToolKit = auto_ptr<nsToolkit>( new nsToolkit() );
mMacSink = new nsMacMessageSink();
mMacPump = auto_ptr<nsMacMessagePump>( new nsMacMessagePump(mToolKit.get(), mMacSink) );
return NS_OK;
}
//-------------------------------------------------------------------------
//
// Enter a message handler loop
//
//-------------------------------------------------------------------------
nsresult nsAppShell::Run()
{
if (!mMacPump.get())
return NS_ERROR_NOT_INITIALIZED;
mMacPump->StartRunning();
mMacPump->DoMessagePump();
//if (mDispatchListener)
//mDispatchListener->AfterDispatch();
if (mExitCalled) // hack: see below
{
mRefCnt --;
if (mRefCnt == 0)
delete this;
}
return NS_OK;
}
//-------------------------------------------------------------------------
//
// Exit appshell
//
//-------------------------------------------------------------------------
NS_IMETHODIMP nsAppShell::Exit()
{
if (mMacPump.get())
{
Spindown();
mExitCalled = PR_TRUE;
mRefCnt ++; // hack: since the applications are likely to delete us
// after calling this method (see nsViewerApp::Exit()),
// we temporarily bump the refCnt to let the message pump
// exit properly. The object will delete itself afterwards.
}
return NS_OK;
}
//-------------------------------------------------------------------------
//
// Prepare to process events
//
//-------------------------------------------------------------------------
NS_IMETHODIMP nsAppShell::Spinup()
{
if (mMacPump.get())
{
mMacPump->StartRunning();
return NS_OK;
}
return NS_ERROR_NOT_INITIALIZED;
}
//-------------------------------------------------------------------------
//
// Stop being prepared to process events.
//
//-------------------------------------------------------------------------
NS_IMETHODIMP nsAppShell::Spindown()
{
if (mMacPump.get())
mMacPump->StopRunning();
return NS_OK;
}
//-------------------------------------------------------------------------
//
// nsAppShell constructor
//
//-------------------------------------------------------------------------
nsAppShell::nsAppShell()
{
// The toolbox initialization code has moved to NSStdLib (InitializeToolbox)
if (!mInitializedToolbox)
{
InitializeMacToolbox();
mInitializedToolbox = PR_TRUE;
}
mRefCnt = 0;
mExitCalled = PR_FALSE;
mMacSink = 0;
}
//-------------------------------------------------------------------------
//
// nsAppShell destructor
//
//-------------------------------------------------------------------------
nsAppShell::~nsAppShell()
{
}
//-------------------------------------------------------------------------
//
// PushThreadEventQueue - begin processing events from a new queue
//
//-------------------------------------------------------------------------
NS_METHOD nsAppShell::PushThreadEventQueue()
{
nsresult rv;
// push a nested event queue for event processing from netlib
// onto our UI thread queue stack.
NS_WITH_SERVICE(nsIEventQueueService, eQueueService, kEventQueueServiceCID, &rv);
if (NS_SUCCEEDED(rv))
rv = eQueueService->PushThreadEventQueue();
else
NS_ERROR("Appshell unable to obtain eventqueue service.");
return rv;
}
//-------------------------------------------------------------------------
//
// PopThreadEventQueue - stop processing on a previously pushed event queue
//
//-------------------------------------------------------------------------
NS_METHOD nsAppShell::PopThreadEventQueue()
{
nsresult rv;
NS_WITH_SERVICE(nsIEventQueueService, eQueueService, kEventQueueServiceCID, &rv);
if (NS_SUCCEEDED(rv))
rv = eQueueService->PopThreadEventQueue();
else
NS_ERROR("Appshell unable to obtain eventqueue service.");
return rv;
}
NS_METHOD
nsAppShell::GetNativeEvent(PRBool &aRealEvent, void *&aEvent)
{
static EventRecord theEvent; // icky icky static (can't really do any better)
if (!mMacPump.get())
return NS_ERROR_NOT_INITIALIZED;
aRealEvent = mMacPump->GetEvent(theEvent);
aEvent = &theEvent;
return NS_OK;
}
//-------------------------------------------------------------------------
//
// determine whether the given event is suitable for a modal window
//
//-------------------------------------------------------------------------
NS_METHOD
nsAppShell::EventIsForModalWindow(PRBool aRealEvent, void *aEvent,
nsIWidget *aWidget,
PRBool *aForWindow)
{
*aForWindow = PR_FALSE;
EventRecord *theEvent = (EventRecord *) aEvent;
if ( aRealEvent == PR_TRUE && theEvent->what != nullEvent ) {
WindowPtr window = (WindowPtr) aWidget->GetNativeData(NS_NATIVE_DISPLAY);
WindowPtr eventWindow = nsnull;
PRInt16 where = ::FindWindow ( theEvent->where, &eventWindow );
switch ( theEvent->what ) {
// is it a mouse event?
case mouseDown:
case mouseUp:
// is it in the given window?
// (note we also let some events questionable for modal dialogs pass through.
// but it makes sense that the draggability et.al. of a modal window should
// be controlled by whether the window has a drag bar).
if ( window == eventWindow &&
( where == inContent || where == inDrag || where == inGrow ||
where == inGoAway || where == inZoomIn || where == inZoomOut ))
*aForWindow = PR_TRUE;
break;
case keyDown:
case keyUp:
case autoKey:
if ( window == eventWindow )
*aForWindow = PR_TRUE;
break;
case diskEvt:
// I think dialogs might want to support floppy insertion, and it doesn't
// interfere with modality...
case updateEvt:
// always let update events through, because if we don't handle them, we're
// doomed!
case activateEvt:
// certainly we have to let the obvious activate events through. hopefully
// our consumption of other events will keep any unwanted activate events
// from even getting this far
*aForWindow = PR_TRUE;
break;
case osEvt:
// check for mouseMoved or suspend/resume events. We especially need to
// let suspend/resume events through in order to make sure the clipboard is
// converted correctly.
unsigned char eventType = (theEvent->message >> 24) & 0x00ff;
if (eventType == mouseMovedMessage) {
// I'm guessing we don't want to let these through unless the mouse is
// in the modal dialog so we don't see rollover feedback in windows behind
// the dialog.
if ( where == inContent && window == eventWindow )
*aForWindow = PR_TRUE;
}
if ( eventType == suspendResumeMessage )
*aForWindow = PR_TRUE;
break;
} // case of which event type
} else
*aForWindow = PR_TRUE;
return NS_OK;
} // EventIsForModalWindow
NS_METHOD
nsAppShell::DispatchNativeEvent(PRBool aRealEvent, void *aEvent)
{
if (!mMacPump.get())
return NS_ERROR_NOT_INITIALIZED;
mMacPump->DispatchEvent(aRealEvent, (EventRecord *) aEvent);
return NS_OK;
}