1998-05-21 02:02:14 +04:00
|
|
|
/* -*- Mode: c++; tab-width: 2; indent-tabs-mode: nil; -*- */
|
2012-05-21 15:12:37 +04:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
1998-05-21 02:02:14 +04:00
|
|
|
|
2011-05-18 15:57:08 +04:00
|
|
|
#include "mozilla/ipc/RPCChannel.h"
|
1998-05-21 02:02:14 +04:00
|
|
|
#include "nsAppShell.h"
|
2000-01-07 03:34:34 +03:00
|
|
|
#include "nsToolkit.h"
|
2006-05-10 21:30:15 +04:00
|
|
|
#include "nsThreadUtils.h"
|
2009-10-07 01:42:45 +04:00
|
|
|
#include "WinTaskbar.h"
|
2012-03-22 04:59:12 +04:00
|
|
|
#include "WinMouseScrollHandler.h"
|
|
|
|
#include "nsWindowDefs.h"
|
2009-12-18 08:54:02 +03:00
|
|
|
#include "nsString.h"
|
2013-02-25 08:00:07 +04:00
|
|
|
#include "WinIMEHandler.h"
|
2011-06-24 18:20:03 +04:00
|
|
|
#include "mozilla/widget/AudioSession.h"
|
2011-10-12 21:52:26 +04:00
|
|
|
#include "mozilla/HangMonitor.h"
|
2009-12-18 08:54:02 +03:00
|
|
|
|
2010-11-29 19:25:16 +03:00
|
|
|
const PRUnichar* kAppShellEventId = L"nsAppShell:EventID";
|
|
|
|
const PRUnichar* kTaskbarButtonEventId = L"TaskbarButtonCreated";
|
|
|
|
|
2006-05-10 21:30:15 +04:00
|
|
|
static UINT sMsgId;
|
1998-09-15 00:40:49 +04:00
|
|
|
|
2012-11-02 15:54:44 +04:00
|
|
|
UINT sTaskbarButtonCreatedMsg;
|
2009-10-06 06:26:54 +04:00
|
|
|
|
|
|
|
/* static */
|
|
|
|
UINT nsAppShell::GetTaskbarButtonCreatedMessage() {
|
|
|
|
return sTaskbarButtonCreatedMsg;
|
|
|
|
}
|
|
|
|
|
2010-11-22 03:29:47 +03:00
|
|
|
namespace mozilla {
|
|
|
|
namespace crashreporter {
|
|
|
|
void LSPAnnotate();
|
|
|
|
} // namespace crashreporter
|
|
|
|
} // namespace mozilla
|
|
|
|
|
|
|
|
using mozilla::crashreporter::LSPAnnotate;
|
|
|
|
|
1998-05-21 02:02:14 +04:00
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
|
2011-09-29 10:19:26 +04:00
|
|
|
static bool PeekUIMessage(MSG* aMsg)
|
2002-08-14 03:43:01 +04:00
|
|
|
{
|
2012-03-22 04:59:12 +04:00
|
|
|
// For avoiding deadlock between our process and plugin process by
|
|
|
|
// mouse wheel messages, we're handling actually when we receive one of
|
|
|
|
// following internal messages which is posted by native mouse wheel message
|
|
|
|
// handler. Any other events, especially native modifier key events, should
|
|
|
|
// not be handled between native message and posted internal message because
|
|
|
|
// it may make different modifier key state or mouse cursor position between
|
|
|
|
// them.
|
|
|
|
if (mozilla::widget::MouseScrollHandler::IsWaitingInternalMessage() &&
|
|
|
|
::PeekMessageW(aMsg, NULL, MOZ_WM_MOUSEWHEEL_FIRST,
|
|
|
|
MOZ_WM_MOUSEWHEEL_LAST, PM_REMOVE)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-02-12 04:56:12 +03:00
|
|
|
MSG keyMsg, imeMsg, mouseMsg, *pMsg = 0;
|
2011-09-29 10:19:26 +04:00
|
|
|
bool haveKeyMsg, haveIMEMsg, haveMouseMsg;
|
2011-02-12 04:56:12 +03:00
|
|
|
|
|
|
|
haveKeyMsg = ::PeekMessageW(&keyMsg, NULL, WM_KEYFIRST, WM_IME_KEYLAST, PM_NOREMOVE);
|
|
|
|
haveIMEMsg = ::PeekMessageW(&imeMsg, NULL, NS_WM_IMEFIRST, NS_WM_IMELAST, PM_NOREMOVE);
|
|
|
|
haveMouseMsg = ::PeekMessageW(&mouseMsg, NULL, WM_MOUSEFIRST, WM_MOUSELAST, PM_NOREMOVE);
|
|
|
|
|
|
|
|
if (haveKeyMsg) {
|
|
|
|
pMsg = &keyMsg;
|
|
|
|
}
|
|
|
|
if (haveIMEMsg && (!pMsg || imeMsg.time < pMsg->time)) {
|
|
|
|
pMsg = &imeMsg;
|
|
|
|
}
|
|
|
|
|
2013-02-25 08:00:07 +04:00
|
|
|
if (pMsg && !mozilla::widget::IMEHandler::CanOptimizeKeyAndIMEMessages()) {
|
2011-10-02 06:16:19 +04:00
|
|
|
return false;
|
2011-02-12 04:56:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (haveMouseMsg && (!pMsg || mouseMsg.time < pMsg->time)) {
|
|
|
|
pMsg = &mouseMsg;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!pMsg) {
|
2011-10-02 06:16:19 +04:00
|
|
|
return false;
|
2002-08-14 03:43:01 +04:00
|
|
|
}
|
|
|
|
|
2011-02-12 04:56:12 +03:00
|
|
|
return ::PeekMessageW(aMsg, NULL, pMsg->message, pMsg->message, PM_REMOVE);
|
2002-08-14 03:43:01 +04:00
|
|
|
}
|
|
|
|
|
2006-05-10 21:30:15 +04:00
|
|
|
/*static*/ LRESULT CALLBACK
|
|
|
|
nsAppShell::EventWindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
1998-05-21 02:02:14 +04:00
|
|
|
{
|
2006-05-10 21:30:15 +04:00
|
|
|
if (uMsg == sMsgId) {
|
2007-07-08 11:08:04 +04:00
|
|
|
nsAppShell *as = reinterpret_cast<nsAppShell *>(lParam);
|
2006-05-10 21:30:15 +04:00
|
|
|
as->NativeEventCallback();
|
|
|
|
NS_RELEASE(as);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return DefWindowProc(hwnd, uMsg, wParam, lParam);
|
|
|
|
}
|
1999-10-23 07:09:30 +04:00
|
|
|
|
2006-05-10 21:30:15 +04:00
|
|
|
nsAppShell::~nsAppShell()
|
|
|
|
{
|
|
|
|
if (mEventWnd) {
|
|
|
|
// DestroyWindow doesn't do anything when called from a non UI thread.
|
|
|
|
// Since mEventWnd was created on the UI thread, it must be destroyed on
|
|
|
|
// the UI thread.
|
|
|
|
SendMessage(mEventWnd, WM_CLOSE, 0, 0);
|
|
|
|
}
|
1998-05-21 02:02:14 +04:00
|
|
|
}
|
|
|
|
|
2006-05-10 21:30:15 +04:00
|
|
|
nsresult
|
|
|
|
nsAppShell::Init()
|
|
|
|
{
|
2010-11-22 03:29:47 +03:00
|
|
|
#ifdef MOZ_CRASHREPORTER
|
|
|
|
LSPAnnotate();
|
|
|
|
#endif
|
|
|
|
|
2013-02-12 01:56:59 +04:00
|
|
|
mLastNativeEventScheduled = TimeStamp::NowLoRes();
|
2011-08-09 18:48:10 +04:00
|
|
|
|
2006-05-10 21:30:15 +04:00
|
|
|
if (!sMsgId)
|
2010-11-29 19:25:16 +03:00
|
|
|
sMsgId = RegisterWindowMessageW(kAppShellEventId);
|
2006-05-10 21:30:15 +04:00
|
|
|
|
2010-11-29 19:25:16 +03:00
|
|
|
sTaskbarButtonCreatedMsg = ::RegisterWindowMessageW(kTaskbarButtonEventId);
|
2009-10-06 06:26:54 +04:00
|
|
|
NS_ASSERTION(sTaskbarButtonCreatedMsg, "Could not register taskbar button creation message");
|
|
|
|
|
2008-08-30 01:26:56 +04:00
|
|
|
WNDCLASSW wc;
|
2006-05-10 21:30:15 +04:00
|
|
|
HINSTANCE module = GetModuleHandle(NULL);
|
|
|
|
|
2008-08-30 01:26:56 +04:00
|
|
|
const PRUnichar *const kWindowClass = L"nsAppShell:EventWindowClass";
|
|
|
|
if (!GetClassInfoW(module, kWindowClass, &wc)) {
|
2006-05-10 21:30:15 +04:00
|
|
|
wc.style = 0;
|
|
|
|
wc.lpfnWndProc = EventWindowProc;
|
|
|
|
wc.cbClsExtra = 0;
|
|
|
|
wc.cbWndExtra = 0;
|
|
|
|
wc.hInstance = module;
|
|
|
|
wc.hIcon = NULL;
|
|
|
|
wc.hCursor = NULL;
|
|
|
|
wc.hbrBackground = (HBRUSH) NULL;
|
2008-08-30 01:26:56 +04:00
|
|
|
wc.lpszMenuName = (LPCWSTR) NULL;
|
2006-05-10 21:30:15 +04:00
|
|
|
wc.lpszClassName = kWindowClass;
|
2008-08-30 01:26:56 +04:00
|
|
|
RegisterClassW(&wc);
|
2006-05-10 21:30:15 +04:00
|
|
|
}
|
1999-10-22 01:34:57 +04:00
|
|
|
|
2008-08-30 01:26:56 +04:00
|
|
|
mEventWnd = CreateWindowW(kWindowClass, L"nsAppShell:EventWindow",
|
2006-05-10 21:30:15 +04:00
|
|
|
0, 0, 0, 10, 10, NULL, NULL, module, NULL);
|
|
|
|
NS_ENSURE_STATE(mEventWnd);
|
1999-10-22 01:34:57 +04:00
|
|
|
|
2006-05-10 21:30:15 +04:00
|
|
|
return nsBaseAppShell::Init();
|
|
|
|
}
|
1999-10-22 01:34:57 +04:00
|
|
|
|
2009-12-18 08:54:02 +03:00
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsAppShell::Run(void)
|
|
|
|
{
|
2011-06-24 18:20:03 +04:00
|
|
|
// Ignore failure; failing to start the application is not exactly an
|
|
|
|
// appropriate response to failing to start an audio session.
|
|
|
|
mozilla::widget::StartAudioSession();
|
2009-12-18 08:54:02 +03:00
|
|
|
|
2010-02-24 04:39:14 +03:00
|
|
|
nsresult rv = nsBaseAppShell::Run();
|
|
|
|
|
2011-06-24 18:20:03 +04:00
|
|
|
mozilla::widget::StopAudioSession();
|
|
|
|
|
2010-02-24 04:39:14 +03:00
|
|
|
return rv;
|
2009-12-18 08:54:02 +03:00
|
|
|
}
|
|
|
|
|
2012-11-02 15:54:44 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsAppShell::Exit(void)
|
|
|
|
{
|
|
|
|
return nsBaseAppShell::Exit();
|
|
|
|
}
|
|
|
|
|
2010-11-29 19:25:16 +03:00
|
|
|
void
|
|
|
|
nsAppShell::DoProcessMoreGeckoEvents()
|
|
|
|
{
|
|
|
|
// Called by nsBaseAppShell's NativeEventCallback() after it has finished
|
|
|
|
// processing pending gecko events and there are still gecko events pending
|
|
|
|
// for the thread. (This can happen if NS_ProcessPendingEvents reached it's
|
|
|
|
// starvation timeout limit.) The default behavior in nsBaseAppShell is to
|
|
|
|
// call ScheduleNativeEventCallback to post a follow up native event callback
|
|
|
|
// message. This triggers an additional call to NativeEventCallback for more
|
|
|
|
// gecko event processing.
|
|
|
|
|
|
|
|
// There's a deadlock risk here with certain internal Windows modal loops. In
|
|
|
|
// our dispatch code, we prioritize messages so that input is handled first.
|
|
|
|
// However Windows modal dispatch loops often prioritize posted messages. If
|
|
|
|
// we find ourselves in a tight gecko timer loop where NS_ProcessPendingEvents
|
|
|
|
// takes longer than the timer duration, NS_HasPendingEvents(thread) will
|
|
|
|
// always be true. ScheduleNativeEventCallback will be called on every
|
|
|
|
// NativeEventCallback callback, and in a Windows modal dispatch loop, the
|
|
|
|
// callback message will be processed first -> input gets starved, dead lock.
|
|
|
|
|
|
|
|
// To avoid, don't post native callback messages from NativeEventCallback
|
|
|
|
// when we're in a modal loop. This gets us back into the Windows modal
|
|
|
|
// dispatch loop dispatching input messages. Once we drop out of the modal
|
|
|
|
// loop, we use mNativeCallbackPending to fire off a final NativeEventCallback
|
|
|
|
// if we need it, which insures NS_ProcessPendingEvents gets called and all
|
|
|
|
// gecko events get processed.
|
|
|
|
if (mEventloopNestingLevel < 2) {
|
2012-07-30 18:20:58 +04:00
|
|
|
OnDispatchedEvent(nullptr);
|
2011-10-02 06:16:19 +04:00
|
|
|
mNativeCallbackPending = false;
|
2010-11-29 19:25:16 +03:00
|
|
|
} else {
|
2011-10-02 06:16:19 +04:00
|
|
|
mNativeCallbackPending = true;
|
2010-11-29 19:25:16 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-05-10 21:30:15 +04:00
|
|
|
void
|
|
|
|
nsAppShell::ScheduleNativeEventCallback()
|
1999-02-10 19:20:29 +03:00
|
|
|
{
|
2010-11-29 19:25:16 +03:00
|
|
|
// Post a message to the hidden message window
|
|
|
|
NS_ADDREF_THIS(); // will be released when the event is processed
|
2011-08-09 18:48:10 +04:00
|
|
|
// Time stamp this event so we can detect cases where the event gets
|
|
|
|
// dropping in sub classes / modal loops we do not control.
|
2013-02-12 01:56:59 +04:00
|
|
|
mLastNativeEventScheduled = TimeStamp::NowLoRes();
|
2007-07-08 11:08:04 +04:00
|
|
|
::PostMessage(mEventWnd, sMsgId, 0, reinterpret_cast<LPARAM>(this));
|
2006-05-10 21:30:15 +04:00
|
|
|
}
|
2000-01-22 00:56:09 +03:00
|
|
|
|
2011-09-29 10:19:26 +04:00
|
|
|
bool
|
|
|
|
nsAppShell::ProcessNextNativeEvent(bool mayWait)
|
2006-05-10 21:30:15 +04:00
|
|
|
{
|
2011-05-18 15:57:08 +04:00
|
|
|
// Notify ipc we are spinning a (possibly nested) gecko event loop.
|
|
|
|
mozilla::ipc::RPCChannel::NotifyGeckoEventDispatch();
|
|
|
|
|
2011-09-29 10:19:26 +04:00
|
|
|
bool gotMessage = false;
|
2002-04-14 07:37:42 +04:00
|
|
|
|
2000-01-22 00:56:09 +03:00
|
|
|
do {
|
2006-05-10 21:30:15 +04:00
|
|
|
MSG msg;
|
2012-05-25 17:22:19 +04:00
|
|
|
bool uiMessage = PeekUIMessage(&msg);
|
|
|
|
|
2011-02-12 04:56:12 +03:00
|
|
|
// Give priority to keyboard and mouse messages.
|
2012-05-25 17:22:19 +04:00
|
|
|
if (uiMessage ||
|
|
|
|
PeekMessageW(&msg, NULL, 0, 0, PM_REMOVE)) {
|
2011-10-02 06:16:19 +04:00
|
|
|
gotMessage = true;
|
2006-07-13 02:43:15 +04:00
|
|
|
if (msg.message == WM_QUIT) {
|
2008-08-09 22:49:42 +04:00
|
|
|
::PostQuitMessage(msg.wParam);
|
2006-07-13 02:43:15 +04:00
|
|
|
Exit();
|
|
|
|
} else {
|
2012-05-25 17:22:19 +04:00
|
|
|
// If we had UI activity we would be processing it now so we know we
|
|
|
|
// have either kUIActivity or kActivityNoUIAVail.
|
|
|
|
mozilla::HangMonitor::NotifyActivity(
|
|
|
|
uiMessage ? mozilla::HangMonitor::kUIActivity :
|
|
|
|
mozilla::HangMonitor::kActivityNoUIAVail);
|
2006-07-13 02:43:15 +04:00
|
|
|
::TranslateMessage(&msg);
|
|
|
|
::DispatchMessageW(&msg);
|
|
|
|
}
|
2006-05-10 21:30:15 +04:00
|
|
|
} else if (mayWait) {
|
|
|
|
// Block and wait for any posted application message
|
2011-10-12 21:52:26 +04:00
|
|
|
mozilla::HangMonitor::Suspend();
|
2006-05-10 21:30:15 +04:00
|
|
|
::WaitMessage();
|
2000-01-22 00:56:09 +03:00
|
|
|
}
|
2006-05-10 21:30:15 +04:00
|
|
|
} while (!gotMessage && mayWait);
|
2000-01-22 00:56:09 +03:00
|
|
|
|
2010-11-29 19:25:16 +03:00
|
|
|
// See DoProcessNextNativeEvent, mEventloopNestingLevel will be
|
|
|
|
// one when a modal loop unwinds.
|
|
|
|
if (mNativeCallbackPending && mEventloopNestingLevel == 1)
|
|
|
|
DoProcessMoreGeckoEvents();
|
|
|
|
|
2011-08-09 18:48:10 +04:00
|
|
|
// Check for starved native callbacks. If we haven't processed one
|
|
|
|
// of these events in NATIVE_EVENT_STARVATION_LIMIT, fire one off.
|
2013-02-12 01:56:59 +04:00
|
|
|
static const mozilla::TimeDuration nativeEventStarvationLimit =
|
|
|
|
mozilla::TimeDuration::FromSeconds(NATIVE_EVENT_STARVATION_LIMIT);
|
|
|
|
|
|
|
|
if ((TimeStamp::NowLoRes() - mLastNativeEventScheduled) >
|
|
|
|
nativeEventStarvationLimit) {
|
2011-08-09 18:48:10 +04:00
|
|
|
ScheduleNativeEventCallback();
|
|
|
|
}
|
|
|
|
|
2006-05-10 21:30:15 +04:00
|
|
|
return gotMessage;
|
1999-02-10 19:20:29 +03:00
|
|
|
}
|