mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
7b7af2c67c
--HG-- rename : layout/generic/nsFrameFrame.cpp => layout/generic/nsSubDocumentFrame.cpp
854 lines
25 KiB
C++
854 lines
25 KiB
C++
/* -*- Mode: C++; c-basic-offset: 2; indent-tabs-mode: nil; tab-width: 8; -*- */
|
|
/* vim: set sw=2 ts=8 et tw=80 : */
|
|
/* ***** 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 ***** */
|
|
|
|
#include "mozilla/dom/ExternalHelperAppParent.h"
|
|
#include "TabParent.h"
|
|
|
|
#include "mozilla/dom/ContentParent.h"
|
|
#include "mozilla/ipc/DocumentRendererParent.h"
|
|
#include "mozilla/ipc/DocumentRendererShmemParent.h"
|
|
#include "mozilla/ipc/DocumentRendererNativeIDParent.h"
|
|
#include "mozilla/layout/RenderFrameParent.h"
|
|
|
|
#include "nsIURI.h"
|
|
#include "nsFocusManager.h"
|
|
#include "nsCOMPtr.h"
|
|
#include "nsServiceManagerUtils.h"
|
|
#include "nsIDOMElement.h"
|
|
#include "nsEventDispatcher.h"
|
|
#include "nsIDOMEventTarget.h"
|
|
#include "nsIWindowWatcher.h"
|
|
#include "nsIDOMWindow.h"
|
|
#include "nsIIdentityInfo.h"
|
|
#include "nsPIDOMWindow.h"
|
|
#include "TabChild.h"
|
|
#include "nsIDOMEvent.h"
|
|
#include "nsIPrivateDOMEvent.h"
|
|
#include "nsIWebProgressListener2.h"
|
|
#include "nsFrameLoader.h"
|
|
#include "nsNetUtil.h"
|
|
#include "jsarray.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsGeolocationOOP.h"
|
|
#include "nsIDOMNSHTMLFrameElement.h"
|
|
#include "nsIDialogCreator.h"
|
|
#include "nsThreadUtils.h"
|
|
#include "nsSerializationHelper.h"
|
|
#include "nsIPromptFactory.h"
|
|
#include "nsIContent.h"
|
|
#include "mozilla/unused.h"
|
|
|
|
#ifdef ANDROID
|
|
#include "AndroidBridge.h"
|
|
using namespace mozilla;
|
|
#endif
|
|
|
|
using namespace mozilla::dom;
|
|
using namespace mozilla::ipc;
|
|
using namespace mozilla::layout;
|
|
|
|
// The flags passed by the webProgress notifications are 16 bits shifted
|
|
// from the ones registered by webProgressListeners.
|
|
#define NOTIFY_FLAG_SHIFT 16
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
NS_IMPL_ISUPPORTS5(TabParent, nsITabParent, nsIWebProgress, nsIAuthPromptProvider, nsISSLStatusProvider, nsISecureBrowserUI)
|
|
|
|
TabParent::TabParent()
|
|
: mSecurityState(nsIWebProgressListener::STATE_IS_INSECURE)
|
|
{
|
|
}
|
|
|
|
TabParent::~TabParent()
|
|
{
|
|
}
|
|
|
|
void
|
|
TabParent::ActorDestroy(ActorDestroyReason why)
|
|
{
|
|
nsRefPtr<nsFrameLoader> frameLoader = GetFrameLoader();
|
|
if (frameLoader) {
|
|
frameLoader->DestroyChild();
|
|
}
|
|
}
|
|
|
|
bool
|
|
TabParent::RecvMoveFocus(const bool& aForward)
|
|
{
|
|
nsCOMPtr<nsIFocusManager> fm = do_GetService(FOCUSMANAGER_CONTRACTID);
|
|
if (fm) {
|
|
nsCOMPtr<nsIDOMElement> dummy;
|
|
PRUint32 type = aForward ? PRUint32(nsIFocusManager::MOVEFOCUS_FORWARD)
|
|
: PRUint32(nsIFocusManager::MOVEFOCUS_BACKWARD);
|
|
fm->MoveFocus(nsnull, mFrameElement, type, nsIFocusManager::FLAG_BYKEY,
|
|
getter_AddRefs(dummy));
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
TabParent::RecvEvent(const RemoteDOMEvent& aEvent)
|
|
{
|
|
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;
|
|
}
|
|
|
|
bool
|
|
TabParent::RecvNotifyProgressChange(const PRInt64& aProgress,
|
|
const PRInt64& aProgressMax,
|
|
const PRInt64& aTotalProgress,
|
|
const PRInt64& aMaxTotalProgress)
|
|
{
|
|
/*
|
|
* First notify any listeners of the new progress info...
|
|
*
|
|
* Operate the elements from back to front so that if items get
|
|
* get removed from the list it won't affect our iteration
|
|
*/
|
|
nsCOMPtr<nsIWebProgressListener> listener;
|
|
PRUint32 count = mListenerInfoList.Length();
|
|
|
|
while (count-- > 0) {
|
|
TabParentListenerInfo *info = &mListenerInfoList[count];
|
|
if (!(info->mNotifyMask & nsIWebProgress::NOTIFY_PROGRESS)) {
|
|
continue;
|
|
}
|
|
|
|
listener = do_QueryReferent(info->mWeakListener);
|
|
if (!listener) {
|
|
// the listener went away. gracefully pull it out of the list.
|
|
mListenerInfoList.RemoveElementAt(count);
|
|
continue;
|
|
}
|
|
|
|
nsCOMPtr<nsIWebProgressListener2> listener2 =
|
|
do_QueryReferent(info->mWeakListener);
|
|
if (listener2) {
|
|
listener2->OnProgressChange64(this, nsnull, aProgress, aProgressMax,
|
|
aTotalProgress, aMaxTotalProgress);
|
|
} else {
|
|
listener->OnProgressChange(this, nsnull, PRInt32(aProgress),
|
|
PRInt32(aProgressMax),
|
|
PRInt32(aTotalProgress),
|
|
PRInt32(aMaxTotalProgress));
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
TabParent::RecvNotifyStateChange(const PRUint32& aStateFlags,
|
|
const nsresult& aStatus)
|
|
{
|
|
/*
|
|
* First notify any listeners of the new state info...
|
|
*
|
|
* Operate the elements from back to front so that if items get
|
|
* get removed from the list it won't affect our iteration
|
|
*/
|
|
nsCOMPtr<nsIWebProgressListener> listener;
|
|
PRUint32 count = mListenerInfoList.Length();
|
|
|
|
while (count-- > 0) {
|
|
TabParentListenerInfo *info = &mListenerInfoList[count];
|
|
|
|
// The flags used in listener registration are shifted over
|
|
// 16 bits from the ones sent in the notification, so we shift
|
|
// to see if the listener is interested in this change.
|
|
// Note that the flags are not changed in the notification we
|
|
// send along. Flags are defined in nsIWebProgressListener and
|
|
// nsIWebProgress.
|
|
// See nsDocLoader for another example of this.
|
|
if (!(info->mNotifyMask & (aStateFlags >> NOTIFY_FLAG_SHIFT))) {
|
|
continue;
|
|
}
|
|
|
|
listener = do_QueryReferent(info->mWeakListener);
|
|
if (!listener) {
|
|
// the listener went away. gracefully pull it out of the list.
|
|
mListenerInfoList.RemoveElementAt(count);
|
|
continue;
|
|
}
|
|
|
|
listener->OnStateChange(this, nsnull, aStateFlags, aStatus);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
TabParent::RecvNotifyLocationChange(const nsCString& aUri)
|
|
{
|
|
nsCOMPtr<nsIURI> uri;
|
|
nsresult rv = NS_NewURI(getter_AddRefs(uri), aUri);
|
|
if (NS_FAILED(rv)) {
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
* First notify any listeners of the new state info...
|
|
*
|
|
* Operate the elements from back to front so that if items get
|
|
* get removed from the list it won't affect our iteration
|
|
*/
|
|
nsCOMPtr<nsIWebProgressListener> listener;
|
|
PRUint32 count = mListenerInfoList.Length();
|
|
|
|
while (count-- > 0) {
|
|
TabParentListenerInfo *info = &mListenerInfoList[count];
|
|
if (!(info->mNotifyMask & nsIWebProgress::NOTIFY_LOCATION)) {
|
|
continue;
|
|
}
|
|
|
|
listener = do_QueryReferent(info->mWeakListener);
|
|
if (!listener) {
|
|
// the listener went away. gracefully pull it out of the list.
|
|
mListenerInfoList.RemoveElementAt(count);
|
|
continue;
|
|
}
|
|
|
|
listener->OnLocationChange(this, nsnull, uri);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
TabParent::RecvNotifyStatusChange(const nsresult& status,
|
|
const nsString& message)
|
|
{
|
|
/*
|
|
* First notify any listeners of the new state info...
|
|
*
|
|
* Operate the elements from back to front so that if items get
|
|
* get removed from the list it won't affect our iteration
|
|
*/
|
|
nsCOMPtr<nsIWebProgressListener> listener;
|
|
PRUint32 count = mListenerInfoList.Length();
|
|
|
|
while (count-- > 0) {
|
|
TabParentListenerInfo *info = &mListenerInfoList[count];
|
|
if (!(info->mNotifyMask & nsIWebProgress::NOTIFY_STATUS)) {
|
|
continue;
|
|
}
|
|
|
|
listener = do_QueryReferent(info->mWeakListener);
|
|
if (!listener) {
|
|
// the listener went away. gracefully pull it out of the list.
|
|
mListenerInfoList.RemoveElementAt(count);
|
|
continue;
|
|
}
|
|
|
|
listener->OnStatusChange(this, nsnull, status, message.BeginReading());
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
TabParent::RecvNotifySecurityChange(const PRUint32& aState,
|
|
const PRBool& aUseSSLStatusObject,
|
|
const nsString& aTooltip,
|
|
const nsCString& aSecInfoAsString)
|
|
{
|
|
/*
|
|
* First notify any listeners of the new state info...
|
|
*
|
|
* Operate the elements from back to front so that if items get
|
|
* get removed from the list it won't affect our iteration
|
|
*/
|
|
|
|
mSecurityState = aState;
|
|
mSecurityTooltipText = aTooltip;
|
|
|
|
if (!aSecInfoAsString.IsEmpty()) {
|
|
nsCOMPtr<nsISupports> secInfoSupports;
|
|
nsresult rv = NS_DeserializeObject(aSecInfoAsString, getter_AddRefs(secInfoSupports));
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
nsCOMPtr<nsIIdentityInfo> idInfo = do_QueryInterface(secInfoSupports);
|
|
if (idInfo) {
|
|
PRBool isEV;
|
|
if (NS_SUCCEEDED(idInfo->GetIsExtendedValidation(&isEV)) && isEV)
|
|
mSecurityState |= nsIWebProgressListener::STATE_IDENTITY_EV_TOPLEVEL;
|
|
}
|
|
}
|
|
|
|
mSecurityStatusObject = nsnull;
|
|
if (aUseSSLStatusObject)
|
|
{
|
|
nsCOMPtr<nsISSLStatusProvider> sslStatusProvider =
|
|
do_QueryInterface(secInfoSupports);
|
|
if (sslStatusProvider)
|
|
sslStatusProvider->GetSSLStatus(getter_AddRefs(mSecurityStatusObject));
|
|
}
|
|
}
|
|
|
|
nsCOMPtr<nsIWebProgressListener> listener;
|
|
PRUint32 count = mListenerInfoList.Length();
|
|
|
|
while (count-- > 0) {
|
|
TabParentListenerInfo *info = &mListenerInfoList[count];
|
|
if (!(info->mNotifyMask & nsIWebProgress::NOTIFY_SECURITY)) {
|
|
continue;
|
|
}
|
|
|
|
listener = do_QueryReferent(info->mWeakListener);
|
|
if (!listener) {
|
|
// the listener went away. gracefully pull it out of the list.
|
|
mListenerInfoList.RemoveElementAt(count);
|
|
continue;
|
|
}
|
|
|
|
listener->OnSecurityChange(this, nsnull, mSecurityState);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
TabParent::RecvRefreshAttempted(const nsCString& aURI, const PRInt32& aMillis,
|
|
const bool& aSameURI, bool* refreshAllowed)
|
|
{
|
|
nsCOMPtr<nsIURI> uri;
|
|
nsresult rv = NS_NewURI(getter_AddRefs(uri), aURI);
|
|
if (NS_FAILED(rv)) {
|
|
return false;
|
|
}
|
|
/*
|
|
* First notify any listeners of the new state info...
|
|
*
|
|
* Operate the elements from back to front so that if items get
|
|
* get removed from the list it won't affect our iteration
|
|
*/
|
|
|
|
nsCOMPtr<nsIWebProgressListener> listener;
|
|
PRUint32 count = mListenerInfoList.Length();
|
|
|
|
*refreshAllowed = true;
|
|
while (count-- > 0) {
|
|
TabParentListenerInfo *info = &mListenerInfoList[count];
|
|
if (!(info->mNotifyMask & nsIWebProgress::NOTIFY_REFRESH)) {
|
|
continue;
|
|
}
|
|
|
|
listener = do_QueryReferent(info->mWeakListener);
|
|
if (!listener) {
|
|
// the listener went away. gracefully pull it out of the list.
|
|
mListenerInfoList.RemoveElementAt(count);
|
|
continue;
|
|
}
|
|
|
|
nsCOMPtr<nsIWebProgressListener2> listener2 =
|
|
do_QueryReferent(info->mWeakListener);
|
|
if (!listener2) {
|
|
continue;
|
|
}
|
|
|
|
// some listeners don't seem to set this at all...
|
|
PRBool allowed = true;
|
|
listener2->OnRefreshAttempted(this, uri,
|
|
aMillis, aSameURI, &allowed);
|
|
*refreshAllowed = allowed && *refreshAllowed;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
TabParent::AnswerCreateWindow(PBrowserParent** retval)
|
|
{
|
|
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;
|
|
}
|
|
|
|
*retval = frameLoader->GetRemoteBrowser();
|
|
return true;
|
|
}
|
|
|
|
void
|
|
TabParent::LoadURL(nsIURI* aURI)
|
|
{
|
|
nsCString spec;
|
|
aURI->GetSpec(spec);
|
|
|
|
unused << SendLoadURL(spec);
|
|
}
|
|
|
|
void
|
|
TabParent::Show(const nsIntSize& size)
|
|
{
|
|
// sigh
|
|
unused << SendShow(size);
|
|
}
|
|
|
|
void
|
|
TabParent::Move(const nsIntSize& size)
|
|
{
|
|
unused << SendMove(size);
|
|
}
|
|
|
|
void
|
|
TabParent::Activate()
|
|
{
|
|
unused << SendActivate();
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
TabParent::Init(nsIDOMWindow *window)
|
|
{
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
TabParent::GetState(PRUint32 *aState)
|
|
{
|
|
NS_ENSURE_ARG(aState);
|
|
*aState = mSecurityState;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
TabParent::GetTooltipText(nsAString & aTooltipText)
|
|
{
|
|
aTooltipText = mSecurityTooltipText;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
TabParent::GetSSLStatus(nsISupports ** aStatus)
|
|
{
|
|
NS_IF_ADDREF(*aStatus = mSecurityStatusObject);
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
mozilla::ipc::PDocumentRendererParent*
|
|
TabParent::AllocPDocumentRenderer(const PRInt32& x,
|
|
const PRInt32& y, const PRInt32& w, const PRInt32& h, const nsString& bgcolor,
|
|
const PRUint32& flags, const bool& flush)
|
|
{
|
|
return new DocumentRendererParent();
|
|
}
|
|
|
|
bool
|
|
TabParent::DeallocPDocumentRenderer(PDocumentRendererParent* actor)
|
|
{
|
|
delete actor;
|
|
return true;
|
|
}
|
|
|
|
mozilla::ipc::PDocumentRendererShmemParent*
|
|
TabParent::AllocPDocumentRendererShmem(const PRInt32& x,
|
|
const PRInt32& y, const PRInt32& w, const PRInt32& h, const nsString& bgcolor,
|
|
const PRUint32& flags, const bool& flush, const gfxMatrix& aMatrix,
|
|
Shmem& buf)
|
|
{
|
|
return new DocumentRendererShmemParent();
|
|
}
|
|
|
|
bool
|
|
TabParent::DeallocPDocumentRendererShmem(PDocumentRendererShmemParent* actor)
|
|
{
|
|
delete actor;
|
|
return true;
|
|
}
|
|
|
|
mozilla::ipc::PDocumentRendererNativeIDParent*
|
|
TabParent::AllocPDocumentRendererNativeID(const PRInt32& x,
|
|
const PRInt32& y, const PRInt32& w, const PRInt32& h, const nsString& bgcolor,
|
|
const PRUint32& flags, const bool& flush, const gfxMatrix& aMatrix,
|
|
const PRUint32& nativeID)
|
|
{
|
|
return new DocumentRendererNativeIDParent();
|
|
}
|
|
|
|
bool
|
|
TabParent::DeallocPDocumentRendererNativeID(PDocumentRendererNativeIDParent* actor)
|
|
{
|
|
delete actor;
|
|
return true;
|
|
}
|
|
|
|
PGeolocationRequestParent*
|
|
TabParent::AllocPGeolocationRequest(const IPC::URI& uri)
|
|
{
|
|
return new GeolocationRequestParent(mFrameElement, uri);
|
|
}
|
|
|
|
bool
|
|
TabParent::DeallocPGeolocationRequest(PGeolocationRequestParent* actor)
|
|
{
|
|
delete actor;
|
|
return true;
|
|
}
|
|
|
|
void
|
|
TabParent::SendMouseEvent(const nsAString& aType, float aX, float aY,
|
|
PRInt32 aButton, PRInt32 aClickCount,
|
|
PRInt32 aModifiers, PRBool aIgnoreRootScrollFrame)
|
|
{
|
|
unused << PBrowserParent::SendMouseEvent(nsString(aType), aX, aY,
|
|
aButton, aClickCount,
|
|
aModifiers, aIgnoreRootScrollFrame);
|
|
}
|
|
|
|
void
|
|
TabParent::SendKeyEvent(const nsAString& aType,
|
|
PRInt32 aKeyCode,
|
|
PRInt32 aCharCode,
|
|
PRInt32 aModifiers,
|
|
PRBool aPreventDefault)
|
|
{
|
|
unused << PBrowserParent::SendKeyEvent(nsString(aType), aKeyCode, aCharCode,
|
|
aModifiers, aPreventDefault);
|
|
}
|
|
|
|
bool
|
|
TabParent::RecvSyncMessage(const nsString& aMessage,
|
|
const nsString& aJSON,
|
|
nsTArray<nsString>* aJSONRetVal)
|
|
{
|
|
return ReceiveMessage(aMessage, PR_TRUE, aJSON, aJSONRetVal);
|
|
}
|
|
|
|
bool
|
|
TabParent::RecvAsyncMessage(const nsString& aMessage,
|
|
const nsString& aJSON)
|
|
{
|
|
return ReceiveMessage(aMessage, PR_FALSE, aJSON, nsnull);
|
|
}
|
|
|
|
bool
|
|
TabParent::RecvQueryContentResult(const nsQueryContentEvent& event)
|
|
{
|
|
#ifdef ANDROID
|
|
if (!event.mSucceeded) {
|
|
AndroidBridge::Bridge()->ReturnIMEQueryResult(nsnull, 0, 0, 0);
|
|
return true;
|
|
}
|
|
|
|
switch (event.message) {
|
|
case NS_QUERY_TEXT_CONTENT:
|
|
AndroidBridge::Bridge()->ReturnIMEQueryResult(
|
|
event.mReply.mString.get(), event.mReply.mString.Length(), 0, 0);
|
|
break;
|
|
case NS_QUERY_SELECTED_TEXT:
|
|
AndroidBridge::Bridge()->ReturnIMEQueryResult(
|
|
event.mReply.mString.get(),
|
|
event.mReply.mString.Length(),
|
|
event.GetSelectionStart(),
|
|
event.GetSelectionEnd() - event.GetSelectionStart());
|
|
break;
|
|
}
|
|
#endif
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
TabParent::ReceiveMessage(const nsString& aMessage,
|
|
PRBool aSync,
|
|
const nsString& aJSON,
|
|
nsTArray<nsString>* aJSONRetVal)
|
|
{
|
|
nsRefPtr<nsFrameLoader> frameLoader = GetFrameLoader();
|
|
if (frameLoader && frameLoader->GetFrameMessageManager()) {
|
|
nsFrameMessageManager* manager = frameLoader->GetFrameMessageManager();
|
|
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;
|
|
}
|
|
|
|
manager->ReceiveMessage(mFrameElement,
|
|
aMessage,
|
|
aSync,
|
|
aJSON,
|
|
objectsArray,
|
|
aJSONRetVal);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// nsIWebProgress
|
|
nsresult
|
|
TabParent::AddProgressListener(nsIWebProgressListener* aListener,
|
|
PRUint32 aNotifyMask)
|
|
{
|
|
if (GetListenerInfo(aListener)) {
|
|
// The listener is already registered!
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
nsWeakPtr listener = do_GetWeakReference(aListener);
|
|
if (!listener) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
TabParentListenerInfo info(listener, aNotifyMask);
|
|
|
|
if (!mListenerInfoList.AppendElement(info))
|
|
return NS_ERROR_FAILURE;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
TabParent::RemoveProgressListener(nsIWebProgressListener *aListener)
|
|
{
|
|
nsAutoPtr<TabParentListenerInfo> info(GetListenerInfo(aListener));
|
|
|
|
return info && mListenerInfoList.RemoveElement(*info) ?
|
|
NS_OK : NS_ERROR_FAILURE;
|
|
}
|
|
|
|
TabParentListenerInfo *
|
|
TabParent::GetListenerInfo(nsIWebProgressListener *aListener)
|
|
{
|
|
PRUint32 i, count;
|
|
TabParentListenerInfo *info;
|
|
|
|
nsCOMPtr<nsISupports> listener1 = do_QueryInterface(aListener);
|
|
count = mListenerInfoList.Length();
|
|
for (i = 0; i < count; ++i) {
|
|
info = &mListenerInfoList[i];
|
|
|
|
if (info) {
|
|
nsCOMPtr<nsISupports> listener2 = do_QueryReferent(info->mWeakListener);
|
|
if (listener1 == listener2) {
|
|
return info;
|
|
}
|
|
}
|
|
}
|
|
return nsnull;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
TabParent::GetDOMWindow(nsIDOMWindow **aResult)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
TabParent::GetIsLoadingDocument(PRBool *aIsLoadingDocument)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|
|
|
|
// 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));
|
|
}
|
|
|
|
PContentDialogParent*
|
|
TabParent::AllocPContentDialog(const PRUint32& aType,
|
|
const nsCString& aName,
|
|
const nsCString& aFeatures,
|
|
const nsTArray<int>& aIntParams,
|
|
const nsTArray<nsString>& aStringParams)
|
|
{
|
|
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) {
|
|
nsTArray<PRInt32> intParams;
|
|
nsTArray<nsString> stringParams;
|
|
TabChild::ParamsToArrays(params, intParams, stringParams);
|
|
unused << PContentDialogParent::Send__delete__(dialog,
|
|
intParams, stringParams);
|
|
}
|
|
}
|
|
if (ShouldDelayDialogs() && mDelayedDialogs.Length()) {
|
|
nsContentUtils::DispatchTrustedEvent(frame->GetOwnerDoc(), frame,
|
|
NS_LITERAL_STRING("MozDelayedModalDialog"),
|
|
PR_TRUE, PR_TRUE);
|
|
}
|
|
}
|
|
|
|
PRenderFrameParent*
|
|
TabParent::AllocPRenderFrame()
|
|
{
|
|
nsRefPtr<nsFrameLoader> frameLoader = GetFrameLoader();
|
|
return new RenderFrameParent(frameLoader);
|
|
}
|
|
|
|
bool
|
|
TabParent::DeallocPRenderFrame(PRenderFrameParent* aFrame)
|
|
{
|
|
delete aFrame;
|
|
return true;
|
|
}
|
|
|
|
PRBool
|
|
TabParent::ShouldDelayDialogs()
|
|
{
|
|
nsRefPtr<nsFrameLoader> frameLoader = GetFrameLoader();
|
|
NS_ENSURE_TRUE(frameLoader, PR_TRUE);
|
|
PRBool delay = PR_FALSE;
|
|
frameLoader->GetDelayRemoteDialogs(&delay);
|
|
return delay;
|
|
}
|
|
|
|
already_AddRefed<nsFrameLoader>
|
|
TabParent::GetFrameLoader() const
|
|
{
|
|
nsCOMPtr<nsIFrameLoaderOwner> frameLoaderOwner = do_QueryInterface(mFrameElement);
|
|
return frameLoaderOwner ? frameLoaderOwner->GetFrameLoader() : nsnull;
|
|
}
|
|
|
|
PExternalHelperAppParent*
|
|
TabParent::AllocPExternalHelperApp(const IPC::URI& uri,
|
|
const nsCString& aMimeContentType,
|
|
const bool& aForceSave,
|
|
const PRInt64& aContentLength)
|
|
{
|
|
ExternalHelperAppParent *parent = new ExternalHelperAppParent(uri, aContentLength);
|
|
parent->AddRef();
|
|
parent->Init(this, aMimeContentType, aForceSave);
|
|
return parent;
|
|
}
|
|
|
|
bool
|
|
TabParent::DeallocPExternalHelperApp(PExternalHelperAppParent* aService)
|
|
{
|
|
ExternalHelperAppParent *parent = static_cast<ExternalHelperAppParent *>(aService);
|
|
parent->Release();
|
|
return true;
|
|
}
|
|
|
|
} // namespace tabs
|
|
} // namespace mozilla
|