2010-03-26 11:39:39 -07:00
|
|
|
/* -*- Mode: C++; c-basic-offset: 2; indent-tabs-mode: nil; tab-width: 8; -*- */
|
2010-06-24 15:20:42 -07:00
|
|
|
/* vim: set sw=2 ts=8 et tw=80 : */
|
2009-09-09 15:00:14 -07:00
|
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla 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/MPL/
|
|
|
|
*
|
|
|
|
* 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 Content App.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
|
|
|
* The Mozilla Foundation.
|
|
|
|
* Portions created by the Initial Developer are Copyright (C) 2009
|
|
|
|
* 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 MPL, 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 MPL, the GPL or the LGPL.
|
|
|
|
*
|
|
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
|
2009-06-30 13:39:22 -07:00
|
|
|
#include "TabParent.h"
|
|
|
|
|
2010-08-20 16:24:41 -07:00
|
|
|
#include "mozilla/dom/ContentParent.h"
|
2009-10-29 10:58:31 -07:00
|
|
|
#include "mozilla/ipc/DocumentRendererParent.h"
|
2010-08-20 16:24:41 -07:00
|
|
|
#include "mozilla/layout/RenderFrameParent.h"
|
2010-10-20 10:12:32 -07:00
|
|
|
#include "mozilla/docshell/OfflineCacheUpdateParent.h"
|
2009-06-30 13:39:22 -07:00
|
|
|
|
|
|
|
#include "nsIURI.h"
|
2009-11-05 10:14:22 -08:00
|
|
|
#include "nsFocusManager.h"
|
|
|
|
#include "nsCOMPtr.h"
|
|
|
|
#include "nsServiceManagerUtils.h"
|
|
|
|
#include "nsIDOMElement.h"
|
2009-11-17 06:22:23 -08:00
|
|
|
#include "nsEventDispatcher.h"
|
|
|
|
#include "nsIDOMEventTarget.h"
|
2010-05-17 04:25:22 -07:00
|
|
|
#include "nsIWindowWatcher.h"
|
|
|
|
#include "nsIDOMWindow.h"
|
2010-08-13 01:06:40 -07:00
|
|
|
#include "nsIIdentityInfo.h"
|
2010-05-17 04:25:22 -07:00
|
|
|
#include "nsPIDOMWindow.h"
|
|
|
|
#include "TabChild.h"
|
2009-11-17 06:22:23 -08:00
|
|
|
#include "nsIDOMEvent.h"
|
|
|
|
#include "nsIPrivateDOMEvent.h"
|
2009-12-31 17:35:55 -08:00
|
|
|
#include "nsFrameLoader.h"
|
2010-03-26 11:39:39 -07:00
|
|
|
#include "nsNetUtil.h"
|
2010-03-29 13:29:45 -07:00
|
|
|
#include "nsContentUtils.h"
|
2010-09-13 12:49:27 -07:00
|
|
|
#include "nsContentPermissionHelper.h"
|
2010-05-13 10:44:51 -07:00
|
|
|
#include "nsIDOMNSHTMLFrameElement.h"
|
2010-05-17 04:25:22 -07:00
|
|
|
#include "nsIDialogCreator.h"
|
|
|
|
#include "nsThreadUtils.h"
|
2010-08-13 01:06:40 -07:00
|
|
|
#include "nsSerializationHelper.h"
|
2010-06-24 15:20:42 -07:00
|
|
|
#include "nsIPromptFactory.h"
|
|
|
|
#include "nsIContent.h"
|
2010-07-19 11:33:33 -07:00
|
|
|
#include "mozilla/unused.h"
|
2010-12-02 17:24:04 -08:00
|
|
|
#include "nsDebug.h"
|
2010-07-19 11:33:33 -07:00
|
|
|
|
2010-08-20 16:24:41 -07:00
|
|
|
using namespace mozilla::dom;
|
|
|
|
using namespace mozilla::ipc;
|
|
|
|
using namespace mozilla::layout;
|
2009-06-30 13:39:22 -07:00
|
|
|
|
2010-03-26 11:39:39 -07:00
|
|
|
// The flags passed by the webProgress notifications are 16 bits shifted
|
|
|
|
// from the ones registered by webProgressListeners.
|
|
|
|
#define NOTIFY_FLAG_SHIFT 16
|
|
|
|
|
2009-06-30 13:39:22 -07:00
|
|
|
namespace mozilla {
|
2009-08-12 09:18:08 -07:00
|
|
|
namespace dom {
|
2009-06-30 13:39:22 -07:00
|
|
|
|
2010-09-23 20:28:15 -07:00
|
|
|
TabParent *TabParent::mIMETabParent = nsnull;
|
|
|
|
|
2010-11-21 11:21:56 -08:00
|
|
|
NS_IMPL_ISUPPORTS3(TabParent, nsITabParent, nsIAuthPromptProvider, nsISecureBrowserUI)
|
2010-03-26 11:39:39 -07:00
|
|
|
|
2009-08-12 09:18:08 -07:00
|
|
|
TabParent::TabParent()
|
2010-12-05 05:15:36 -08:00
|
|
|
: mIMEComposing(PR_FALSE)
|
|
|
|
, mIMECompositionEnding(PR_FALSE)
|
2011-01-11 13:34:31 -08:00
|
|
|
, mIMESeqno(0)
|
2010-12-09 10:57:05 -08:00
|
|
|
, mDPI(0)
|
2009-06-30 13:39:22 -07:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
TabParent::~TabParent()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2010-12-09 10:57:05 -08:00
|
|
|
void
|
|
|
|
TabParent::SetOwnerElement(nsIDOMElement* aElement)
|
|
|
|
{
|
|
|
|
mFrameElement = aElement;
|
|
|
|
|
|
|
|
// Cache the DPI of the screen, since we may lose the element/widget later
|
|
|
|
if (aElement) {
|
|
|
|
nsCOMPtr<nsIWidget> widget = GetWidget();
|
|
|
|
NS_ABORT_IF_FALSE(widget, "Non-null OwnerElement must provide a widget!");
|
|
|
|
mDPI = widget->GetDPI();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-05 20:54:47 -08:00
|
|
|
void
|
|
|
|
TabParent::Destroy()
|
|
|
|
{
|
|
|
|
// If this fails, it's most likely due to a content-process crash,
|
|
|
|
// and auto-cleanup will kick in. Otherwise, the child side will
|
|
|
|
// destroy itself and send back __delete__().
|
|
|
|
unused << SendDestroy();
|
|
|
|
|
|
|
|
for (size_t i = 0; i < ManagedPRenderFrameParent().Length(); ++i) {
|
|
|
|
RenderFrameParent* rfp =
|
|
|
|
static_cast<RenderFrameParent*>(ManagedPRenderFrameParent()[i]);
|
|
|
|
rfp->Destroy();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-04-11 17:34:02 -07:00
|
|
|
void
|
|
|
|
TabParent::ActorDestroy(ActorDestroyReason why)
|
|
|
|
{
|
2011-01-21 08:48:56 -08:00
|
|
|
if (mIMETabParent == this)
|
|
|
|
mIMETabParent = nsnull;
|
2010-08-05 15:11:23 -07:00
|
|
|
nsRefPtr<nsFrameLoader> frameLoader = GetFrameLoader();
|
|
|
|
if (frameLoader) {
|
|
|
|
frameLoader->DestroyChild();
|
2010-04-11 17:34:02 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-11-05 10:14:22 -08:00
|
|
|
bool
|
2010-07-19 11:33:33 -07:00
|
|
|
TabParent::RecvMoveFocus(const bool& aForward)
|
2009-11-05 10:14:22 -08:00
|
|
|
{
|
|
|
|
nsCOMPtr<nsIFocusManager> fm = do_GetService(FOCUSMANAGER_CONTRACTID);
|
|
|
|
if (fm) {
|
|
|
|
nsCOMPtr<nsIDOMElement> dummy;
|
2010-07-19 11:33:33 -07:00
|
|
|
PRUint32 type = aForward ? PRUint32(nsIFocusManager::MOVEFOCUS_FORWARD)
|
|
|
|
: PRUint32(nsIFocusManager::MOVEFOCUS_BACKWARD);
|
2009-11-05 10:14:22 -08:00
|
|
|
fm->MoveFocus(nsnull, mFrameElement, type, nsIFocusManager::FLAG_BYKEY,
|
|
|
|
getter_AddRefs(dummy));
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-11-17 06:22:23 -08:00
|
|
|
bool
|
2010-07-19 11:33:33 -07:00
|
|
|
TabParent::RecvEvent(const RemoteDOMEvent& aEvent)
|
2009-11-17 06:22:23 -08:00
|
|
|
{
|
|
|
|
nsCOMPtr<nsIDOMEvent> event = do_QueryInterface(aEvent.mEvent);
|
|
|
|
NS_ENSURE_TRUE(event, true);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMEventTarget> target = do_QueryInterface(mFrameElement);
|
|
|
|
NS_ENSURE_TRUE(target, true);
|
|
|
|
|
|
|
|
PRBool dummy;
|
|
|
|
target->DispatchEvent(event, &dummy);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-12-31 17:35:55 -08:00
|
|
|
bool
|
2010-07-19 11:33:33 -07:00
|
|
|
TabParent::AnswerCreateWindow(PBrowserParent** retval)
|
2009-12-31 17:35:55 -08:00
|
|
|
{
|
|
|
|
if (!mBrowserDOMWindow) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get a new rendering area from the browserDOMWin. We don't want
|
|
|
|
// to be starting any loads here, so get it with a null URI.
|
|
|
|
nsCOMPtr<nsIFrameLoaderOwner> frameLoaderOwner;
|
|
|
|
mBrowserDOMWindow->OpenURIInFrame(nsnull, nsnull,
|
|
|
|
nsIBrowserDOMWindow::OPEN_NEWTAB,
|
|
|
|
nsIBrowserDOMWindow::OPEN_NEW,
|
|
|
|
getter_AddRefs(frameLoaderOwner));
|
|
|
|
if (!frameLoaderOwner) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsRefPtr<nsFrameLoader> frameLoader = frameLoaderOwner->GetFrameLoader();
|
|
|
|
if (!frameLoader) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-07-19 11:33:33 -07:00
|
|
|
*retval = frameLoader->GetRemoteBrowser();
|
2009-12-31 17:35:55 -08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-06-30 13:39:22 -07:00
|
|
|
void
|
|
|
|
TabParent::LoadURL(nsIURI* aURI)
|
|
|
|
{
|
|
|
|
nsCString spec;
|
|
|
|
aURI->GetSpec(spec);
|
|
|
|
|
2010-07-19 11:33:33 -07:00
|
|
|
unused << SendLoadURL(spec);
|
2009-06-30 13:39:22 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2010-08-20 16:24:40 -07:00
|
|
|
TabParent::Show(const nsIntSize& size)
|
2009-06-30 13:39:22 -07:00
|
|
|
{
|
2010-08-20 16:24:40 -07:00
|
|
|
// sigh
|
|
|
|
unused << SendShow(size);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
TabParent::Move(const nsIntSize& size)
|
|
|
|
{
|
|
|
|
unused << SendMove(size);
|
2009-06-30 13:39:22 -07:00
|
|
|
}
|
|
|
|
|
2009-11-05 10:14:22 -08:00
|
|
|
void
|
|
|
|
TabParent::Activate()
|
|
|
|
{
|
2010-07-19 11:33:33 -07:00
|
|
|
unused << SendActivate();
|
2009-11-05 10:14:22 -08:00
|
|
|
}
|
|
|
|
|
2011-06-17 17:08:32 -07:00
|
|
|
void
|
|
|
|
TabParent::Deactivate()
|
|
|
|
{
|
|
|
|
unused << SendDeactivate();
|
|
|
|
}
|
|
|
|
|
2010-08-13 01:06:40 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
TabParent::Init(nsIDOMWindow *window)
|
|
|
|
{
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
TabParent::GetState(PRUint32 *aState)
|
|
|
|
{
|
|
|
|
NS_ENSURE_ARG(aState);
|
2010-09-17 19:11:05 -07:00
|
|
|
NS_WARNING("SecurityState not valid here");
|
2010-11-21 11:21:56 -08:00
|
|
|
*aState = 0;
|
2010-08-13 01:06:40 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
TabParent::GetTooltipText(nsAString & aTooltipText)
|
|
|
|
{
|
2010-11-21 11:21:56 -08:00
|
|
|
aTooltipText.Truncate();
|
2010-08-13 01:06:40 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-10-26 15:20:53 -07:00
|
|
|
PDocumentRendererParent*
|
|
|
|
TabParent::AllocPDocumentRenderer(const nsRect& documentRect,
|
2010-10-26 15:20:53 -07:00
|
|
|
const gfxMatrix& transform,
|
2010-10-26 15:20:53 -07:00
|
|
|
const nsString& bgcolor,
|
|
|
|
const PRUint32& renderFlags,
|
2010-10-26 15:20:53 -07:00
|
|
|
const bool& flushLayout,
|
|
|
|
const nsIntSize& renderSize)
|
2009-10-29 10:58:31 -07:00
|
|
|
{
|
|
|
|
return new DocumentRendererParent();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2009-12-03 00:16:14 -08:00
|
|
|
TabParent::DeallocPDocumentRenderer(PDocumentRendererParent* actor)
|
2009-10-29 10:58:31 -07:00
|
|
|
{
|
2009-12-03 00:16:14 -08:00
|
|
|
delete actor;
|
2009-10-29 10:58:31 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-09-09 22:00:08 -07:00
|
|
|
PContentPermissionRequestParent*
|
|
|
|
TabParent::AllocPContentPermissionRequest(const nsCString& type, const IPC::URI& uri)
|
2010-05-13 10:44:51 -07:00
|
|
|
{
|
2010-09-13 12:49:27 -07:00
|
|
|
return new ContentPermissionRequestParent(type, mFrameElement, uri);
|
2010-05-13 10:44:51 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2010-09-09 22:00:08 -07:00
|
|
|
TabParent::DeallocPContentPermissionRequest(PContentPermissionRequestParent* actor)
|
2010-05-13 10:44:51 -07:00
|
|
|
{
|
|
|
|
delete actor;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-11-05 10:21:09 -08:00
|
|
|
void
|
|
|
|
TabParent::SendMouseEvent(const nsAString& aType, float aX, float aY,
|
|
|
|
PRInt32 aButton, PRInt32 aClickCount,
|
|
|
|
PRInt32 aModifiers, PRBool aIgnoreRootScrollFrame)
|
|
|
|
{
|
2010-07-19 11:33:33 -07:00
|
|
|
unused << PBrowserParent::SendMouseEvent(nsString(aType), aX, aY,
|
|
|
|
aButton, aClickCount,
|
|
|
|
aModifiers, aIgnoreRootScrollFrame);
|
2009-11-05 10:21:09 -08:00
|
|
|
}
|
|
|
|
|
2010-03-18 23:52:18 -07:00
|
|
|
void
|
|
|
|
TabParent::SendKeyEvent(const nsAString& aType,
|
|
|
|
PRInt32 aKeyCode,
|
|
|
|
PRInt32 aCharCode,
|
|
|
|
PRInt32 aModifiers,
|
|
|
|
PRBool aPreventDefault)
|
|
|
|
{
|
2010-07-19 11:33:33 -07:00
|
|
|
unused << PBrowserParent::SendKeyEvent(nsString(aType), aKeyCode, aCharCode,
|
|
|
|
aModifiers, aPreventDefault);
|
2010-03-18 23:52:18 -07:00
|
|
|
}
|
|
|
|
|
2010-02-20 09:05:20 -08:00
|
|
|
bool
|
2010-07-19 11:33:33 -07:00
|
|
|
TabParent::RecvSyncMessage(const nsString& aMessage,
|
|
|
|
const nsString& aJSON,
|
2010-11-08 18:49:00 -08:00
|
|
|
InfallibleTArray<nsString>* aJSONRetVal)
|
2010-02-20 09:05:20 -08:00
|
|
|
{
|
2010-06-17 14:10:14 -07:00
|
|
|
return ReceiveMessage(aMessage, PR_TRUE, aJSON, aJSONRetVal);
|
2010-02-20 09:05:20 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2010-07-19 11:33:33 -07:00
|
|
|
TabParent::RecvAsyncMessage(const nsString& aMessage,
|
|
|
|
const nsString& aJSON)
|
2010-03-29 13:29:45 -07:00
|
|
|
{
|
|
|
|
return ReceiveMessage(aMessage, PR_FALSE, aJSON, nsnull);
|
|
|
|
}
|
|
|
|
|
2010-09-23 20:28:15 -07:00
|
|
|
bool
|
|
|
|
TabParent::RecvNotifyIMEFocus(const PRBool& aFocus,
|
2010-10-01 07:17:37 -07:00
|
|
|
nsIMEUpdatePreference* aPreference,
|
|
|
|
PRUint32* aSeqno)
|
2010-09-23 20:28:15 -07:00
|
|
|
{
|
|
|
|
nsCOMPtr<nsIWidget> widget = GetWidget();
|
|
|
|
if (!widget)
|
|
|
|
return true;
|
|
|
|
|
2010-10-01 07:17:37 -07:00
|
|
|
*aSeqno = mIMESeqno;
|
2010-09-23 20:28:15 -07:00
|
|
|
mIMETabParent = aFocus ? this : nsnull;
|
|
|
|
mIMESelectionAnchor = 0;
|
|
|
|
mIMESelectionFocus = 0;
|
2010-09-23 20:28:15 -07:00
|
|
|
nsresult rv = widget->OnIMEFocusChange(aFocus);
|
|
|
|
|
|
|
|
if (aFocus) {
|
|
|
|
if (NS_SUCCEEDED(rv) && rv != NS_SUCCESS_IME_NO_UPDATES) {
|
|
|
|
*aPreference = widget->GetIMEUpdatePreference();
|
|
|
|
} else {
|
|
|
|
aPreference->mWantUpdates = PR_FALSE;
|
|
|
|
aPreference->mWantHints = PR_FALSE;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
mIMECacheText.Truncate(0);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
TabParent::RecvNotifyIMETextChange(const PRUint32& aStart,
|
|
|
|
const PRUint32& aEnd,
|
|
|
|
const PRUint32& aNewEnd)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIWidget> widget = GetWidget();
|
|
|
|
if (!widget)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
widget->OnIMETextChange(aStart, aEnd, aNewEnd);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2010-10-01 07:17:37 -07:00
|
|
|
TabParent::RecvNotifyIMESelection(const PRUint32& aSeqno,
|
|
|
|
const PRUint32& aAnchor,
|
2010-09-23 20:28:15 -07:00
|
|
|
const PRUint32& aFocus)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIWidget> widget = GetWidget();
|
|
|
|
if (!widget)
|
|
|
|
return true;
|
|
|
|
|
2010-10-01 07:17:37 -07:00
|
|
|
if (aSeqno == mIMESeqno) {
|
|
|
|
mIMESelectionAnchor = aAnchor;
|
|
|
|
mIMESelectionFocus = aFocus;
|
|
|
|
widget->OnIMESelectionChange();
|
|
|
|
}
|
2010-09-23 20:28:15 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
TabParent::RecvNotifyIMETextHint(const nsString& aText)
|
|
|
|
{
|
|
|
|
// Replace our cache with new text
|
|
|
|
mIMECacheText = aText;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-09-23 20:28:15 -07:00
|
|
|
/**
|
|
|
|
* Try to answer query event using cached text.
|
|
|
|
*
|
|
|
|
* For NS_QUERY_SELECTED_TEXT, fail if the cache doesn't contain the whole
|
|
|
|
* selected range. (This shouldn't happen because PuppetWidget should have
|
|
|
|
* already sent the whole selection.)
|
|
|
|
*
|
|
|
|
* For NS_QUERY_TEXT_CONTENT, fail only if the cache doesn't overlap with
|
|
|
|
* the queried range. Note the difference from above. We use
|
|
|
|
* this behavior because a normal NS_QUERY_TEXT_CONTENT event is allowed to
|
|
|
|
* have out-of-bounds offsets, so that widget can request content without
|
|
|
|
* knowing the exact length of text. It's up to widget to handle cases when
|
|
|
|
* the returned offset/length are different from the queried offset/length.
|
|
|
|
*/
|
|
|
|
bool
|
|
|
|
TabParent::HandleQueryContentEvent(nsQueryContentEvent& aEvent)
|
|
|
|
{
|
|
|
|
aEvent.mSucceeded = PR_FALSE;
|
|
|
|
aEvent.mWasAsync = PR_FALSE;
|
|
|
|
aEvent.mReply.mFocusedWidget = nsCOMPtr<nsIWidget>(GetWidget()).get();
|
|
|
|
|
|
|
|
switch (aEvent.message)
|
|
|
|
{
|
|
|
|
case NS_QUERY_SELECTED_TEXT:
|
|
|
|
{
|
2011-06-02 05:56:50 -07:00
|
|
|
aEvent.mReply.mOffset = NS_MIN(mIMESelectionAnchor, mIMESelectionFocus);
|
2010-09-23 20:28:15 -07:00
|
|
|
if (mIMESelectionAnchor == mIMESelectionFocus) {
|
|
|
|
aEvent.mReply.mString.Truncate(0);
|
|
|
|
} else {
|
|
|
|
if (mIMESelectionAnchor > mIMECacheText.Length() ||
|
|
|
|
mIMESelectionFocus > mIMECacheText.Length()) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
PRUint32 selLen = mIMESelectionAnchor > mIMESelectionFocus ?
|
|
|
|
mIMESelectionAnchor - mIMESelectionFocus :
|
|
|
|
mIMESelectionFocus - mIMESelectionAnchor;
|
|
|
|
aEvent.mReply.mString = Substring(mIMECacheText,
|
|
|
|
aEvent.mReply.mOffset,
|
|
|
|
selLen);
|
|
|
|
}
|
|
|
|
aEvent.mReply.mReversed = mIMESelectionFocus < mIMESelectionAnchor;
|
|
|
|
aEvent.mReply.mHasSelection = PR_TRUE;
|
|
|
|
aEvent.mSucceeded = PR_TRUE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case NS_QUERY_TEXT_CONTENT:
|
|
|
|
{
|
|
|
|
PRUint32 inputOffset = aEvent.mInput.mOffset,
|
|
|
|
inputEnd = inputOffset + aEvent.mInput.mLength;
|
|
|
|
|
|
|
|
if (inputEnd > mIMECacheText.Length()) {
|
|
|
|
inputEnd = mIMECacheText.Length();
|
|
|
|
}
|
|
|
|
if (inputEnd < inputOffset) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
aEvent.mReply.mOffset = inputOffset;
|
|
|
|
aEvent.mReply.mString = Substring(mIMECacheText,
|
|
|
|
inputOffset,
|
|
|
|
inputEnd - inputOffset);
|
|
|
|
aEvent.mSucceeded = PR_TRUE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-09-23 20:34:26 -07:00
|
|
|
bool
|
2010-10-01 07:17:37 -07:00
|
|
|
TabParent::SendCompositionEvent(nsCompositionEvent& event)
|
2010-09-23 20:34:26 -07:00
|
|
|
{
|
|
|
|
mIMEComposing = event.message == NS_COMPOSITION_START;
|
2011-06-02 05:56:50 -07:00
|
|
|
mIMECompositionStart = NS_MIN(mIMESelectionAnchor, mIMESelectionFocus);
|
2010-09-23 20:34:26 -07:00
|
|
|
if (mIMECompositionEnding)
|
|
|
|
return true;
|
2010-10-01 07:17:37 -07:00
|
|
|
event.seqno = ++mIMESeqno;
|
2010-09-23 20:34:26 -07:00
|
|
|
return PBrowserParent::SendCompositionEvent(event);
|
|
|
|
}
|
|
|
|
|
2010-09-23 20:28:15 -07:00
|
|
|
/**
|
|
|
|
* During ResetInputState or CancelComposition, widget usually sends a
|
|
|
|
* NS_TEXT_TEXT event to finalize or clear the composition, respectively
|
|
|
|
*
|
|
|
|
* Because the event will not reach content in time, we intercept it
|
|
|
|
* here and pass the text as the EndIMEComposition return value
|
|
|
|
*/
|
|
|
|
bool
|
2010-10-01 07:17:37 -07:00
|
|
|
TabParent::SendTextEvent(nsTextEvent& event)
|
2010-09-23 20:28:15 -07:00
|
|
|
{
|
|
|
|
if (mIMECompositionEnding) {
|
|
|
|
mIMECompositionText = event.theText;
|
|
|
|
return true;
|
|
|
|
}
|
2010-09-23 20:34:26 -07:00
|
|
|
|
|
|
|
// We must be able to simulate the selection because
|
|
|
|
// we might not receive selection updates in time
|
|
|
|
if (!mIMEComposing) {
|
2011-06-02 05:56:50 -07:00
|
|
|
mIMECompositionStart = NS_MIN(mIMESelectionAnchor, mIMESelectionFocus);
|
2010-09-23 20:34:26 -07:00
|
|
|
}
|
|
|
|
mIMESelectionAnchor = mIMESelectionFocus =
|
|
|
|
mIMECompositionStart + event.theText.Length();
|
|
|
|
|
2010-10-01 07:17:37 -07:00
|
|
|
event.seqno = ++mIMESeqno;
|
2010-09-23 20:28:15 -07:00
|
|
|
return PBrowserParent::SendTextEvent(event);
|
|
|
|
}
|
|
|
|
|
2010-09-23 20:34:26 -07:00
|
|
|
bool
|
2010-10-01 07:17:37 -07:00
|
|
|
TabParent::SendSelectionEvent(nsSelectionEvent& event)
|
2010-09-23 20:34:26 -07:00
|
|
|
{
|
|
|
|
mIMESelectionAnchor = event.mOffset + (event.mReversed ? event.mLength : 0);
|
|
|
|
mIMESelectionFocus = event.mOffset + (!event.mReversed ? event.mLength : 0);
|
2010-10-01 07:17:37 -07:00
|
|
|
event.seqno = ++mIMESeqno;
|
2010-09-23 20:34:26 -07:00
|
|
|
return PBrowserParent::SendSelectionEvent(event);
|
|
|
|
}
|
|
|
|
|
2010-09-23 20:28:15 -07:00
|
|
|
bool
|
|
|
|
TabParent::RecvEndIMEComposition(const PRBool& aCancel,
|
|
|
|
nsString* aComposition)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIWidget> widget = GetWidget();
|
|
|
|
if (!widget)
|
|
|
|
return true;
|
|
|
|
|
2010-09-23 20:28:15 -07:00
|
|
|
mIMECompositionEnding = PR_TRUE;
|
|
|
|
|
2010-09-23 20:28:15 -07:00
|
|
|
if (aCancel) {
|
|
|
|
widget->CancelIMEComposition();
|
|
|
|
} else {
|
|
|
|
widget->ResetInputState();
|
|
|
|
}
|
2010-09-23 20:28:15 -07:00
|
|
|
|
|
|
|
mIMECompositionEnding = PR_FALSE;
|
|
|
|
*aComposition = mIMECompositionText;
|
|
|
|
mIMECompositionText.Truncate(0);
|
2010-09-23 20:28:15 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
TabParent::RecvGetIMEEnabled(PRUint32* aValue)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIWidget> widget = GetWidget();
|
2011-06-29 14:24:47 -07:00
|
|
|
if (!widget) {
|
|
|
|
*aValue = nsIWidget::IME_STATUS_DISABLED;
|
2010-11-22 22:48:45 -08:00
|
|
|
return true;
|
2011-06-29 14:24:47 -07:00
|
|
|
}
|
2010-11-22 22:48:45 -08:00
|
|
|
|
|
|
|
IMEContext context;
|
2011-03-25 08:03:35 -07:00
|
|
|
widget->GetInputMode(context);
|
|
|
|
*aValue = context.mStatus;
|
2010-09-23 20:28:15 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2011-04-20 05:47:40 -07:00
|
|
|
TabParent::RecvSetInputMode(const PRUint32& aValue, const nsString& aType, const nsString& aAction, const PRUint32& aReason)
|
2010-09-23 20:28:15 -07:00
|
|
|
{
|
2011-05-20 11:44:09 -07:00
|
|
|
// mIMETabParent (which is actually static) tracks which if any TabParent has IMEFocus
|
|
|
|
// When the input mode is set to anything but IME_STATUS_NONE, mIMETabParent should be set to this
|
|
|
|
mIMETabParent = aValue & nsIContent::IME_STATUS_MASK_ENABLED ? this : nsnull;
|
2010-09-23 20:28:15 -07:00
|
|
|
nsCOMPtr<nsIWidget> widget = GetWidget();
|
2010-11-22 22:48:03 -08:00
|
|
|
if (!widget || !AllowContentIME())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
IMEContext context;
|
|
|
|
context.mStatus = aValue;
|
|
|
|
context.mHTMLInputType.Assign(aType);
|
2010-11-23 18:12:53 -08:00
|
|
|
context.mActionHint.Assign(aAction);
|
2011-04-20 05:47:40 -07:00
|
|
|
context.mReason = aReason;
|
2011-03-25 08:03:35 -07:00
|
|
|
widget->SetInputMode(context);
|
2010-11-22 22:48:03 -08:00
|
|
|
|
|
|
|
nsCOMPtr<nsIObserverService> observerService = mozilla::services::GetObserverService();
|
|
|
|
if (!observerService)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
nsAutoString state;
|
|
|
|
state.AppendInt(aValue);
|
|
|
|
observerService->NotifyObservers(nsnull, "ime-enabled-state-changed", state.get());
|
2010-10-18 09:37:00 -07:00
|
|
|
|
2010-09-23 20:28:15 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
TabParent::RecvGetIMEOpenState(PRBool* aValue)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIWidget> widget = GetWidget();
|
|
|
|
if (widget)
|
|
|
|
widget->GetIMEOpenState(aValue);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
TabParent::RecvSetIMEOpenState(const PRBool& aValue)
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIWidget> widget = GetWidget();
|
2010-10-19 01:21:31 -07:00
|
|
|
if (widget && AllowContentIME())
|
2010-09-23 20:28:15 -07:00
|
|
|
widget->SetIMEOpenState(aValue);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-12-02 17:24:04 -08:00
|
|
|
bool
|
|
|
|
TabParent::RecvGetDPI(float* aValue)
|
|
|
|
{
|
2010-12-09 10:57:05 -08:00
|
|
|
NS_ABORT_IF_FALSE(mDPI > 0,
|
|
|
|
"Must not ask for DPI before OwnerElement is received!");
|
|
|
|
*aValue = mDPI;
|
2010-12-02 17:24:04 -08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-03-29 13:29:45 -07:00
|
|
|
bool
|
|
|
|
TabParent::ReceiveMessage(const nsString& aMessage,
|
|
|
|
PRBool aSync,
|
|
|
|
const nsString& aJSON,
|
2010-11-08 18:49:00 -08:00
|
|
|
InfallibleTArray<nsString>* aJSONRetVal)
|
2010-02-20 09:05:20 -08:00
|
|
|
{
|
2010-08-05 15:11:23 -07:00
|
|
|
nsRefPtr<nsFrameLoader> frameLoader = GetFrameLoader();
|
|
|
|
if (frameLoader && frameLoader->GetFrameMessageManager()) {
|
2011-06-16 11:21:08 -07:00
|
|
|
nsRefPtr<nsFrameMessageManager> manager =
|
|
|
|
frameLoader->GetFrameMessageManager();
|
2011-03-24 13:00:45 -07:00
|
|
|
JSContext* ctx = manager->GetJSContext();
|
|
|
|
JSAutoRequest ar(ctx);
|
|
|
|
PRUint32 len = 0; //TODO: obtain a real value in bug 572685
|
|
|
|
// Because we want JS messages to have always the same properties,
|
|
|
|
// create array even if len == 0.
|
|
|
|
JSObject* objectsArray = JS_NewArrayObject(ctx, len, NULL);
|
|
|
|
if (!objectsArray) {
|
|
|
|
return false;
|
2010-02-20 09:05:20 -08:00
|
|
|
}
|
2010-08-05 15:11:23 -07:00
|
|
|
|
|
|
|
manager->ReceiveMessage(mFrameElement,
|
|
|
|
aMessage,
|
|
|
|
aSync,
|
|
|
|
aJSON,
|
|
|
|
objectsArray,
|
|
|
|
aJSONRetVal);
|
2010-02-20 09:05:20 -08:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-06-24 15:20:42 -07:00
|
|
|
// nsIAuthPromptProvider
|
|
|
|
|
|
|
|
// This method is largely copied from nsDocShell::GetAuthPrompt
|
|
|
|
NS_IMETHODIMP
|
|
|
|
TabParent::GetAuthPrompt(PRUint32 aPromptReason, const nsIID& iid,
|
|
|
|
void** aResult)
|
|
|
|
{
|
|
|
|
// we're either allowing auth, or it's a proxy request
|
|
|
|
nsresult rv;
|
|
|
|
nsCOMPtr<nsIPromptFactory> wwatch =
|
|
|
|
do_GetService(NS_WINDOWWATCHER_CONTRACTID, &rv);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMWindow> window;
|
|
|
|
nsCOMPtr<nsIContent> frame = do_QueryInterface(mFrameElement);
|
|
|
|
if (frame)
|
|
|
|
window = do_QueryInterface(frame->GetOwnerDoc()->GetWindow());
|
|
|
|
|
|
|
|
// Get an auth prompter for our window so that the parenting
|
|
|
|
// of the dialogs works as it should when using tabs.
|
|
|
|
return wwatch->GetPrompt(window, iid,
|
|
|
|
reinterpret_cast<void**>(aResult));
|
|
|
|
}
|
|
|
|
|
2010-05-17 04:25:22 -07:00
|
|
|
PContentDialogParent*
|
|
|
|
TabParent::AllocPContentDialog(const PRUint32& aType,
|
|
|
|
const nsCString& aName,
|
|
|
|
const nsCString& aFeatures,
|
2010-11-08 18:49:00 -08:00
|
|
|
const InfallibleTArray<int>& aIntParams,
|
|
|
|
const InfallibleTArray<nsString>& aStringParams)
|
2010-05-17 04:25:22 -07:00
|
|
|
{
|
|
|
|
ContentDialogParent* parent = new ContentDialogParent();
|
|
|
|
nsCOMPtr<nsIDialogParamBlock> params =
|
|
|
|
do_CreateInstance(NS_DIALOGPARAMBLOCK_CONTRACTID);
|
|
|
|
TabChild::ArraysToParams(aIntParams, aStringParams, params);
|
|
|
|
mDelayedDialogs.AppendElement(new DelayedDialogData(parent, aType, aName,
|
|
|
|
aFeatures, params));
|
|
|
|
nsRefPtr<nsIRunnable> ev =
|
|
|
|
NS_NewRunnableMethod(this, &TabParent::HandleDelayedDialogs);
|
|
|
|
NS_DispatchToCurrentThread(ev);
|
|
|
|
return parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
TabParent::HandleDelayedDialogs()
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIWindowWatcher> ww = do_GetService(NS_WINDOWWATCHER_CONTRACTID);
|
|
|
|
nsCOMPtr<nsIDOMWindow> window;
|
|
|
|
nsCOMPtr<nsIContent> frame = do_QueryInterface(mFrameElement);
|
|
|
|
if (frame) {
|
|
|
|
window = do_QueryInterface(frame->GetOwnerDoc()->GetWindow());
|
|
|
|
}
|
|
|
|
nsCOMPtr<nsIDialogCreator> dialogCreator = do_QueryInterface(mBrowserDOMWindow);
|
|
|
|
while (!ShouldDelayDialogs() && mDelayedDialogs.Length()) {
|
|
|
|
PRUint32 index = mDelayedDialogs.Length() - 1;
|
|
|
|
DelayedDialogData* data = mDelayedDialogs[index];
|
|
|
|
mDelayedDialogs.RemoveElementAt(index);
|
|
|
|
nsCOMPtr<nsIDialogParamBlock> params;
|
|
|
|
params.swap(data->mParams);
|
|
|
|
PContentDialogParent* dialog = data->mDialog;
|
|
|
|
if (dialogCreator) {
|
|
|
|
dialogCreator->OpenDialog(data->mType,
|
|
|
|
data->mName, data->mFeatures,
|
|
|
|
params, mFrameElement);
|
|
|
|
} else if (ww) {
|
|
|
|
nsCAutoString url;
|
|
|
|
if (data->mType) {
|
|
|
|
if (data->mType == nsIDialogCreator::SELECT_DIALOG) {
|
|
|
|
url.Assign("chrome://global/content/selectDialog.xul");
|
|
|
|
} else if (data->mType == nsIDialogCreator::GENERIC_DIALOG) {
|
|
|
|
url.Assign("chrome://global/content/commonDialog.xul");
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<nsISupports> arguments(do_QueryInterface(params));
|
|
|
|
nsCOMPtr<nsIDOMWindow> dialog;
|
|
|
|
ww->OpenWindow(window, url.get(), data->mName.get(),
|
|
|
|
data->mFeatures.get(), arguments, getter_AddRefs(dialog));
|
|
|
|
} else {
|
|
|
|
NS_WARNING("unknown dialog types aren't automatically supported in E10s yet!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
delete data;
|
|
|
|
if (dialog) {
|
2010-11-08 18:49:00 -08:00
|
|
|
InfallibleTArray<PRInt32> intParams;
|
|
|
|
InfallibleTArray<nsString> stringParams;
|
2010-05-17 04:25:22 -07:00
|
|
|
TabChild::ParamsToArrays(params, intParams, stringParams);
|
2010-07-19 11:33:33 -07:00
|
|
|
unused << PContentDialogParent::Send__delete__(dialog,
|
|
|
|
intParams, stringParams);
|
2010-05-17 04:25:22 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (ShouldDelayDialogs() && mDelayedDialogs.Length()) {
|
|
|
|
nsContentUtils::DispatchTrustedEvent(frame->GetOwnerDoc(), frame,
|
|
|
|
NS_LITERAL_STRING("MozDelayedModalDialog"),
|
|
|
|
PR_TRUE, PR_TRUE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-20 16:24:41 -07:00
|
|
|
PRenderFrameParent*
|
|
|
|
TabParent::AllocPRenderFrame()
|
|
|
|
{
|
|
|
|
nsRefPtr<nsFrameLoader> frameLoader = GetFrameLoader();
|
2010-10-26 10:51:08 -07:00
|
|
|
NS_WARN_IF_FALSE(frameLoader, "'message sent to unknown actor ID' coming up");
|
|
|
|
return frameLoader ? new RenderFrameParent(frameLoader) : nsnull;
|
2010-08-20 16:24:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
TabParent::DeallocPRenderFrame(PRenderFrameParent* aFrame)
|
|
|
|
{
|
|
|
|
delete aFrame;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-10-20 10:12:32 -07:00
|
|
|
mozilla::docshell::POfflineCacheUpdateParent*
|
|
|
|
TabParent::AllocPOfflineCacheUpdate(const URI& aManifestURI,
|
|
|
|
const URI& aDocumentURI,
|
|
|
|
const nsCString& aClientID,
|
|
|
|
const bool& stickDocument)
|
|
|
|
{
|
|
|
|
nsRefPtr<mozilla::docshell::OfflineCacheUpdateParent> update =
|
|
|
|
new mozilla::docshell::OfflineCacheUpdateParent();
|
|
|
|
|
|
|
|
nsresult rv = update->Schedule(aManifestURI, aDocumentURI, aClientID,
|
|
|
|
stickDocument);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return nsnull;
|
|
|
|
|
|
|
|
POfflineCacheUpdateParent* result = update.get();
|
|
|
|
update.forget();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
TabParent::DeallocPOfflineCacheUpdate(mozilla::docshell::POfflineCacheUpdateParent* actor)
|
|
|
|
{
|
|
|
|
mozilla::docshell::OfflineCacheUpdateParent* update =
|
|
|
|
static_cast<mozilla::docshell::OfflineCacheUpdateParent*>(actor);
|
|
|
|
|
|
|
|
update->Release();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-05-17 04:25:22 -07:00
|
|
|
PRBool
|
|
|
|
TabParent::ShouldDelayDialogs()
|
|
|
|
{
|
2010-08-05 15:11:23 -07:00
|
|
|
nsRefPtr<nsFrameLoader> frameLoader = GetFrameLoader();
|
2010-05-17 04:25:22 -07:00
|
|
|
NS_ENSURE_TRUE(frameLoader, PR_TRUE);
|
|
|
|
PRBool delay = PR_FALSE;
|
|
|
|
frameLoader->GetDelayRemoteDialogs(&delay);
|
|
|
|
return delay;
|
|
|
|
}
|
|
|
|
|
2010-10-19 01:21:31 -07:00
|
|
|
PRBool
|
|
|
|
TabParent::AllowContentIME()
|
|
|
|
{
|
|
|
|
nsFocusManager* fm = nsFocusManager::GetFocusManager();
|
|
|
|
NS_ENSURE_TRUE(fm, PR_FALSE);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIContent> focusedContent = fm->GetFocusedContent();
|
|
|
|
if (focusedContent && focusedContent->IsEditable())
|
|
|
|
return PR_FALSE;
|
|
|
|
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-08-05 15:11:23 -07:00
|
|
|
already_AddRefed<nsFrameLoader>
|
|
|
|
TabParent::GetFrameLoader() const
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIFrameLoaderOwner> frameLoaderOwner = do_QueryInterface(mFrameElement);
|
|
|
|
return frameLoaderOwner ? frameLoaderOwner->GetFrameLoader() : nsnull;
|
|
|
|
}
|
|
|
|
|
2010-09-23 20:28:15 -07:00
|
|
|
already_AddRefed<nsIWidget>
|
|
|
|
TabParent::GetWidget() const
|
|
|
|
{
|
|
|
|
nsCOMPtr<nsIContent> content = do_QueryInterface(mFrameElement);
|
|
|
|
if (!content)
|
|
|
|
return nsnull;
|
|
|
|
|
|
|
|
nsIFrame *frame = content->GetPrimaryFrame();
|
|
|
|
if (!frame)
|
|
|
|
return nsnull;
|
|
|
|
|
|
|
|
return nsCOMPtr<nsIWidget>(frame->GetNearestWidget()).forget();
|
|
|
|
}
|
|
|
|
|
2009-06-30 13:39:22 -07:00
|
|
|
} // namespace tabs
|
|
|
|
} // namespace mozilla
|