gecko-dev/dom/src/base/nsGlobalWindow.cpp

5177 lines
148 KiB
C++
Raw Normal View History

/* -*- 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
1998-07-16 01:16:47 +00:00
*
* 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/
1998-07-16 01:16:47 +00:00
*
* 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.
1998-07-16 01:16:47 +00:00
*
* 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):
* Travis Bogard <travis@netscape.com>
* Brendan Eich <brendan@mozilla.org>
* David Hyatt (hyatt@netscape.com)
* Dan Rosen <dr@netscape.com>
* Vidur Apparao <vidur@netscape.com>
* Johnny Stenback <jst@netscape.com>
*
*
* 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 ***** */
1998-07-16 01:16:47 +00:00
// Local Includes
#include "nsGlobalWindow.h"
#include "nsScreen.h"
#include "nsHistory.h"
#include "nsBarProps.h"
// Helper Classes
#include "nsCOMPtr.h"
#include "nsXPIDLString.h"
#include "nsJSUtils.h"
1998-07-16 01:16:47 +00:00
#include "prmem.h"
#include "jsdbgapi.h" // for JS_ClearWatchPointsForObject
#include "nsReadableUtils.h"
#include "nsDOMClassInfo.h"
// Other Classes
#include "nsIEventListenerManager.h"
#include "nsEscape.h"
#include "nsStyleCoord.h"
#include "nsMimeTypeArray.h"
#include "nsNetUtil.h"
#include "nsPluginArray.h"
#include "nsIPluginHost.h"
#include "nsIJVMManager.h"
#include "nsContentCID.h"
// Interfaces Needed
#include "nsIWidget.h"
#include "nsIBaseWindow.h"
#include "nsICharsetConverterManager.h"
#include "nsICodebasePrincipal.h"
#include "nsIContent.h"
#include "nsIContentViewerFile.h"
#include "nsIContentViewerEdit.h"
#include "nsIDocShell.h"
#include "nsIDocShellLoadInfo.h"
#include "nsIDocShellTreeItem.h"
#include "nsIDocShellTreeNode.h"
#include "nsIDocCharset.h"
#include "nsIDocument.h"
#include "nsIDOMCrypto.h"
#include "nsIDOMDocument.h"
#include "nsIDOMDocumentView.h"
#include "nsIDOMPkcs11.h"
#include "nsIEmbeddingSiteWindow2.h"
#include "nsIEventQueueService.h"
#include "nsIEventStateManager.h"
#include "nsIHttpProtocolHandler.h"
#include "nsIJSContextStack.h"
#include "nsIJSRuntimeService.h"
#include "nsIMarkupDocumentViewer.h"
#include "nsIPref.h"
#include "nsIPresShell.h"
#include "nsIPrivateDOMEvent.h"
#include "nsIProgrammingLanguage.h"
#include "nsIAuthPrompt.h"
#include "nsIServiceManager.h"
#include "nsIScriptGlobalObjectOwner.h"
#include "nsIScriptSecurityManager.h"
#include "nsIScrollableView.h"
#include "nsISelectionController.h"
#include "nsISelection.h"
#include "nsIFrameSelection.h"
#include "nsISidebar.h" // XXX for sidebar HACK, see bug 20721
#include "nsIPrompt.h"
#include "nsIStyleContext.h"
#include "nsIWebNavigation.h"
#include "nsIWebBrowser.h"
#include "nsIWebBrowserChrome.h"
#include "nsIWebBrowserFind.h" // For window.find()
#include "nsIWebShell.h"
#include "nsIWindowMediator.h" // For window.find()
2000-05-17 06:38:43 +00:00
#include "nsIComputedDOMStyle.h"
#include "nsIEntropyCollector.h"
#include "nsDOMCID.h"
#include "nsDOMError.h"
#include "nsIWindowWatcher.h"
#include "nsPIWindowWatcher.h"
#include "nsIContentViewer.h"
#include "nsISupportsPrimitives.h"
#include "nsDOMClassInfo.h"
#include "nsIJSNativeInitializer.h"
#include "nsIPrintSettings.h"
2000-11-04 08:21:20 +00:00
#include "nsWindowRoot.h"
2000-09-22 05:02:20 +00:00
// XXX An unfortunate dependency exists here (two XUL files).
#include "nsIDOMXULDocument.h"
#include "nsIDOMXULCommandDispatcher.h"
2000-09-22 05:02:20 +00:00
2000-09-08 06:59:14 +00:00
#include "nsIBindingManager.h"
2000-09-22 05:02:20 +00:00
#include "nsIXBLService.h"
static nsIEntropyCollector* gEntropyCollector = nsnull;
static PRInt32 gRefCnt = 0;
nsIXPConnect *GlobalWindowImpl::sXPConnect = nsnull;
// CIDs
static NS_DEFINE_IID(kEventQueueServiceCID, NS_EVENTQUEUESERVICE_CID);
static NS_DEFINE_CID(kPrefServiceCID, NS_PREF_CID);
static NS_DEFINE_CID(kJVMServiceCID, NS_JVMMANAGER_CID);
static NS_DEFINE_CID(kHTTPHandlerCID, NS_HTTPPROTOCOLHANDLER_CID);
static NS_DEFINE_CID(kXULControllersCID, NS_XULCONTROLLERS_CID);
static NS_DEFINE_CID(kCharsetConverterManagerCID,
NS_ICHARSETCONVERTERMANAGER_CID);
static NS_DEFINE_IID(kISupportsIID, NS_ISUPPORTS_IID);
static NS_DEFINE_CID(kWindowMediatorCID, NS_WINDOWMEDIATOR_CID); // For window.find()
static const char *sWindowWatcherContractID = "@mozilla.org/embedcomp/window-watcher;1";
static const char *sJSStackContractID = "@mozilla.org/js/xpc/ContextStack;1";
static const char * const kCryptoContractID = NS_CRYPTO_CONTRACTID;
static const char * const kPkcs11ContractID = NS_PKCS11_CONTRACTID;
//*****************************************************************************
//*** GlobalWindowImpl: Object Management
//*****************************************************************************
1998-07-16 01:16:47 +00:00
GlobalWindowImpl::GlobalWindowImpl() :
mJSObject(nsnull), mNavigator(nsnull), mScreen(nsnull), mHistory(nsnull),
mFrames(nsnull), mLocation(nsnull), mMenubar(nsnull), mToolbar(nsnull),
mLocationbar(nsnull), mPersonalbar(nsnull), mStatusbar(nsnull),
mScrollbars(nsnull), mTimeouts(nsnull), mTimeoutInsertionPoint(&mTimeouts),
mRunningTimeout(nsnull), mTimeoutPublicIdCounter(1), mTimeoutFiringDepth(0),
mTimeoutsWereCleared(PR_FALSE), mFirstDocumentLoad(PR_TRUE),
mIsScopeClear(PR_TRUE), mIsDocumentLoaded(PR_FALSE), mGlobalObjectOwner(nsnull),
mDocShell(nsnull), mMutationBits(0), mChromeEventHandler(nsnull)
{
NS_INIT_REFCNT();
// We could have failed the first time through trying
// to create the entropy collector, so we should
// try to get one until we succeed.
if (gRefCnt++ == 0 || !gEntropyCollector) {
nsCOMPtr<nsIEntropyCollector> enCol =
do_GetService(NS_ENTROPYCOLLECTOR_CONTRACTID);
if (enCol) {
gEntropyCollector = enCol;
NS_ADDREF(gEntropyCollector);
}
}
if (!sXPConnect) {
nsServiceManager::GetService(nsIXPConnect::GetCID(),
nsIXPConnect::GetIID(),
(nsISupports **)&sXPConnect);
}
1998-07-16 01:16:47 +00:00
}
GlobalWindowImpl::~GlobalWindowImpl()
{
if (!--gRefCnt) {
NS_IF_RELEASE(gEntropyCollector);
}
mDocument = nsnull; // Forces Release
CleanUp();
}
// static
void
GlobalWindowImpl::ShutDown()
{
NS_IF_RELEASE(sXPConnect);
#ifdef DEBUG_jst
printf ("---- Leaked %d GlobalWindowImpl's\n", gRefCnt);
#endif
}
void GlobalWindowImpl::CleanUp()
{
NS_IF_RELEASE(mNavigator);
NS_IF_RELEASE(mScreen);
NS_IF_RELEASE(mHistory);
NS_IF_RELEASE(mMenubar);
NS_IF_RELEASE(mToolbar);
NS_IF_RELEASE(mLocationbar);
NS_IF_RELEASE(mPersonalbar);
NS_IF_RELEASE(mStatusbar);
NS_IF_RELEASE(mScrollbars);
NS_IF_RELEASE(mLocation);
NS_IF_RELEASE(mFrames);
mOpener = nsnull; // Forces Release
mControllers = nsnull; // Forces Release
mContext = nsnull; // Forces Release
mChromeEventHandler = nsnull; // Forces Release
}
//*****************************************************************************
// GlobalWindowImpl::nsISupports
//*****************************************************************************
1998-07-16 01:16:47 +00:00
// QueryInterface implementation for GlobalWindowImpl
NS_INTERFACE_MAP_BEGIN(GlobalWindowImpl)
NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIScriptGlobalObject)
NS_INTERFACE_MAP_ENTRY(nsIDOMWindowInternal)
NS_INTERFACE_MAP_ENTRY(nsIDOMWindow)
NS_INTERFACE_MAP_ENTRY(nsIDOMJSWindow)
NS_INTERFACE_MAP_ENTRY(nsIScriptGlobalObject)
NS_INTERFACE_MAP_ENTRY(nsIScriptObjectPrincipal)
NS_INTERFACE_MAP_ENTRY(nsIDOMEventReceiver)
NS_INTERFACE_MAP_ENTRY(nsIDOMEventTarget)
NS_INTERFACE_MAP_ENTRY(nsPIDOMWindow)
NS_INTERFACE_MAP_ENTRY(nsIDOMViewCSS)
NS_INTERFACE_MAP_ENTRY(nsIDOMAbstractView)
NS_INTERFACE_MAP_ENTRY(nsISupportsWeakReference)
NS_INTERFACE_MAP_ENTRY(nsIInterfaceRequestor)
NS_DOM_INTERFACE_MAP_ENTRY_CLASSINFO(Window)
NS_INTERFACE_MAP_END
1998-07-16 01:16:47 +00:00
NS_IMPL_ADDREF(GlobalWindowImpl)
NS_IMPL_RELEASE(GlobalWindowImpl)
1998-07-16 01:16:47 +00:00
//*****************************************************************************
// GlobalWindowImpl::nsIScriptGlobalObject
//*****************************************************************************
NS_IMETHODIMP GlobalWindowImpl::SetContext(nsIScriptContext* aContext)
1999-05-14 03:10:51 +00:00
{
// if setting the context to null, then we won't get to clean up the
// named reference, so do it now
if (!aContext) {
NS_WARNING("Possibly early removal of script object, see bug #41608");
} else {
JSContext *cx = (JSContext *)aContext->GetNativeContext();
mJSObject = ::JS_GetGlobalObject(cx);
}
mContext = aContext;
if (mContext) {
nsCOMPtr<nsIDOMWindowInternal> parent;
GetParentInternal(getter_AddRefs(parent));
if (parent) {
// This window is a [i]frame, don't bother GC'ing when the
// frame's context is destroyed since a GC will happen when the
// frameset or host document is destroyed anyway.
mContext->SetGCOnDestruction(PR_FALSE);
}
}
return NS_OK;
1999-05-14 03:10:51 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::GetContext(nsIScriptContext ** aContext)
1998-07-16 01:16:47 +00:00
{
*aContext = mContext;
NS_IF_ADDREF(*aContext);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::SetNewDocument(nsIDOMDocument* aDocument,
PRBool removeEventListeners)
{
if (!aDocument) {
if (mDocument) {
// Cache the old principal now that the document is being removed.
nsCOMPtr<nsIDocument> doc(do_QueryInterface(mDocument));
NS_ENSURE_TRUE(doc, NS_ERROR_FAILURE);
doc->GetPrincipal(getter_AddRefs(mDocumentPrincipal));
}
} else {
// let go of the old cached principal
mDocumentPrincipal = nsnull;
}
// Always clear watchpoints, to deal with two cases:
// 1. The first document for this window is loading, and a miscreant has
// preset watchpoints on the window object in order to attack the new
// document's privileged information.
// 2. A document loaded and used watchpoints on its own window, leaving
// them set until the next document loads. We must clean up window
// watchpoints here.
// Watchpoints set on document and subordinate objects are all cleared
// when those sub-window objects are finalized, after JS_ClearScope and
// a GC run that finds them to be garbage.
if (mContext && mJSObject)
::JS_ClearWatchPointsForObject((JSContext *)mContext->GetNativeContext(),
mJSObject);
if (mFirstDocumentLoad) {
2000-09-22 05:02:20 +00:00
if (aDocument) {
mFirstDocumentLoad = PR_FALSE;
2000-09-22 05:02:20 +00:00
}
mDocument = aDocument;
2000-09-22 05:02:20 +00:00
if (mDocument) {
// Get our private root. If it is equal to us, then we
// need to attach our global key bindings that handle
// browser scrolling and other browser commands.
nsCOMPtr<nsIDOMWindowInternal> internal;
GetPrivateRoot(getter_AddRefs(internal));
2000-11-04 08:21:20 +00:00
nsCOMPtr<nsIDOMWindowInternal> us(do_QueryInterface(NS_STATIC_CAST(nsIDOMWindow*,this)));
2000-09-22 05:02:20 +00:00
if (internal == us) {
nsresult rv;
nsCOMPtr<nsIXBLService> xblService =
do_GetService("@mozilla.org/xbl;1", &rv);
2000-09-22 05:02:20 +00:00
if (xblService) {
2000-11-04 08:21:20 +00:00
nsCOMPtr<nsIDOMEventReceiver> rec(do_QueryInterface(mChromeEventHandler));
2000-09-22 05:02:20 +00:00
xblService->AttachGlobalKeyHandler(rec);
// for now, the only way to get drag/drop is to use the XUL
// wrapper. There are just too many things that need to be
// added to hookup DnD with XBL (pinkerton)
//xblService->AttachGlobalDragHandler(rec);
2000-09-22 05:02:20 +00:00
}
}
}
return NS_OK;
}
/* No mDocShell means we've already been partially closed down.
When that happens, setting status isn't a big requirement,
so don't. (Doesn't happen under normal circumstances, but
bug 49615 describes a case.) */
/* We only want to do this when we're setting a new document rather
than going away. See bug 61840. */
if (mDocShell && aDocument) {
SetStatus(nsString());
SetDefaultStatus(nsString());
}
if (mDocument) {
nsCOMPtr<nsIDocument> doc(do_QueryInterface(mDocument));
nsCOMPtr<nsIURI> docURL;
// If we had a document in this window the document most likely
// made our scope "unclear"
mIsScopeClear = PR_FALSE;
if (doc) {
doc->GetDocumentURL(getter_AddRefs(docURL));
doc = nsnull; // Forces release now
}
if (removeEventListeners && mListenerManager) {
mListenerManager->RemoveAllListeners(PR_FALSE);
mListenerManager = nsnull;
}
if (docURL) {
nsXPIDLCString url;
docURL->GetSpec(getter_Copies(url));
//about:blank URL's do not have ClearScope called on page change.
if (nsCRT::strcmp(url.get(), "about:blank") != 0) {
ClearAllTimeouts();
if (mSidebar) {
mSidebar->SetWindow(nsnull);
mSidebar = nsnull;
}
1998-07-16 01:16:47 +00:00
if (mContext && mJSObject) {
// if (mContext && mJSObject && aDocument) {
// not doing this unless there's a new document prevents a closed window's
// JS properties from going away (that's good) and causes everything,
// and I mean everything, to be leaked (that's bad)
::JS_ClearScope((JSContext *)mContext->GetNativeContext(), mJSObject);
mIsScopeClear = PR_TRUE;
}
}
}
//XXX Should this be outside the about:blank clearscope exception?
mDocument = nsnull; // Forces Release
}
if (mContext && aDocument) {
// Add an extra ref in case we release mContext during GC.
nsCOMPtr<nsIScriptContext> kungFuDeathGrip = mContext;
kungFuDeathGrip->GC();
}
mDocument = aDocument;
if (mDocument && mContext && mIsScopeClear) {
mContext->InitContext(this);
}
2000-11-27 07:55:20 +00:00
// Clear our mutation bitfield.
mMutationBits = 0;
return NS_OK;
}
1999-06-09 19:04:18 +00:00
NS_IMETHODIMP GlobalWindowImpl::SetDocShell(nsIDocShell* aDocShell)
{
2000-09-22 05:02:20 +00:00
if (aDocShell == mDocShell)
return NS_OK;
/* SetDocShell(nsnull) means the window is being torn down. Set the
"closed" JS property, Drop our reference to the script context,
allowing it to be deleted later, and hand off our reference
to the script object (held via a named JS root) to the context
so it will be unrooted later. Meanwhile, keep our weak reference
to the script object so it can be retrieved later, as the JS glue
is wont to do. */
if (!aDocShell && mContext) {
ClearAllTimeouts();
if (mJSObject) {
// Indicate that the window is now closed. Since we've
// cleared scope, we have to explicitly set a property.
jsval val = BOOLEAN_TO_JSVAL(JS_TRUE);
::JS_SetProperty((JSContext *)mContext->GetNativeContext(),
mJSObject, "closed", &val);
}
mContext = nsnull; // force release now
mControllers = nsnull; // force release now
mChromeEventHandler = nsnull; // force release now
}
mDocShell = aDocShell; // Weak Reference
if (mLocation)
mLocation->SetDocShell(aDocShell);
if (mNavigator)
mNavigator->SetDocShell(aDocShell);
if (mHistory)
mHistory->SetDocShell(aDocShell);
if (mFrames)
mFrames->SetDocShell(aDocShell);
if (mScreen)
mScreen->SetDocShell(aDocShell);
if (mDocShell) {
// tell our member elements about the new browserwindow
if (mMenubar) {
nsCOMPtr<nsIWebBrowserChrome> browserChrome;
GetWebBrowserChrome(getter_AddRefs(browserChrome));
mMenubar->SetWebBrowserChrome(browserChrome);
}
// Get our enclosing chrome shell and retrieve its global window impl, so
// that we can do some forwarding to the chrome document.
2000-11-04 08:21:20 +00:00
mDocShell->GetChromeEventHandler(getter_AddRefs(mChromeEventHandler));
if (!mChromeEventHandler) {
// We have no chrome event handler. If we have a parent,
// get our chrome event handler from the parent. If
// we don't have a parent, then we need to make a new
// window root object that will function as a chrome event
// handler and receive all events that occur anywhere inside
// our window.
nsCOMPtr<nsIDOMWindow> parentWindow;
GetParent(getter_AddRefs(parentWindow));
if (parentWindow.get() != NS_STATIC_CAST(nsIDOMWindow*,this)) {
nsCOMPtr<nsPIDOMWindow> piWindow(do_QueryInterface(parentWindow));
nsCOMPtr<nsIChromeEventHandler> handler;
piWindow->GetChromeEventHandler(getter_AddRefs(mChromeEventHandler));
}
else NS_NewWindowRoot(this, getter_AddRefs(mChromeEventHandler));
}
}
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetDocShell(nsIDocShell ** aDocShell)
1999-02-10 16:25:11 +00:00
{
*aDocShell = mDocShell;
NS_IF_ADDREF(*aDocShell);
return NS_OK;
1999-02-10 16:25:11 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::SetOpenerWindow(nsIDOMWindowInternal* aOpener)
{
mOpener = aOpener;
return NS_OK;
}
NS_IMETHODIMP
GlobalWindowImpl::SetGlobalObjectOwner(nsIScriptGlobalObjectOwner* aOwner)
{
mGlobalObjectOwner = aOwner; // Note this is supposed to be a weak ref.
return NS_OK;
}
NS_IMETHODIMP
GlobalWindowImpl::GetGlobalObjectOwner(nsIScriptGlobalObjectOwner ** aOwner)
{
NS_ENSURE_ARG_POINTER(aOwner);
*aOwner = mGlobalObjectOwner;
NS_IF_ADDREF(*aOwner);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::HandleDOMEvent(nsIPresContext* aPresContext,
nsEvent* aEvent,
nsIDOMEvent** aDOMEvent,
PRUint32 aFlags,
nsEventStatus* aEventStatus)
1998-07-16 01:16:47 +00:00
{
nsresult ret = NS_OK;
PRBool externalDOMEvent = PR_FALSE;
nsIDOMEvent *domEvent = nsnull;
static PRUint32 count = 0;
1998-07-16 01:16:47 +00:00
/* mChromeEventHandler and mContext go dangling in the middle of this
function under some circumstances (events that destroy the window)
without this addref. */
nsCOMPtr<nsIChromeEventHandler> kungFuDeathGrip1(mChromeEventHandler);
nsCOMPtr<nsIScriptContext> kungFuDeathGrip2(mContext);
/* If this is a mouse event, use the struct to provide entropy for
* the system.
*/
if (gEntropyCollector &&
(NS_EVENT_FLAG_BUBBLE != aFlags) &&
(aEvent->message == NS_MOUSE_MOVE)) {
//I'd like to not come in here if there is a mChromeEventHandler
//present, but there is always one when the message is
//NS_MOUSE_MOVE.
//
//Chances are this counter will overflow during the life of the
//process, but that's OK for our case. Means we get a little
//more entropy.
if (count++ % 100 == 0) {
//Since the high bits seem to be zero's most of the time,
//let's only take the lowest half of the point structure.
PRInt16 myCoord[4];
myCoord[0] = aEvent->point.x;
myCoord[1] = aEvent->point.y;
myCoord[2] = aEvent->refPoint.x;
myCoord[3] = aEvent->refPoint.y;
gEntropyCollector->RandomUpdate((void*)myCoord, sizeof(myCoord));
gEntropyCollector->RandomUpdate((void*)&aEvent->time, sizeof(PRUint32));
}
}
if (NS_EVENT_FLAG_INIT & aFlags) {
if (aDOMEvent) {
if (*aDOMEvent) {
externalDOMEvent = PR_TRUE;
}
}
else {
aDOMEvent = &domEvent;
}
aEvent->flags = aFlags;
aFlags &= ~(NS_EVENT_FLAG_CANT_BUBBLE | NS_EVENT_FLAG_CANT_CANCEL);
2000-09-08 06:59:14 +00:00
// Execute bindingdetached handlers before we tear ourselves
// down.
if (aEvent->message == NS_PAGE_UNLOAD && mDocument) {
nsCOMPtr<nsIDocument> doc(do_QueryInterface(mDocument));
nsCOMPtr<nsIBindingManager> bindingManager;
doc->GetBindingManager(getter_AddRefs(bindingManager));
if (bindingManager)
bindingManager->ExecuteDetachedHandlers();
}
}
if (aEvent->message == NS_PAGE_UNLOAD) {
mIsDocumentLoaded = PR_FALSE;
}
// Capturing stage
if ((NS_EVENT_FLAG_BUBBLE != aFlags) && mChromeEventHandler) {
// Check chrome document capture here.
// XXX The chrome can not handle this, see bug 51211
if (aEvent->message != NS_IMAGE_LOAD) {
mChromeEventHandler->HandleChromeEvent(aPresContext, aEvent, aDOMEvent,
NS_EVENT_FLAG_CAPTURE,
aEventStatus);
}
}
// Local handling stage
if (mListenerManager && !(aEvent->flags & NS_EVENT_FLAG_STOP_DISPATCH) &&
!((NS_EVENT_FLAG_BUBBLE & aFlags) &&
(NS_EVENT_FLAG_CANT_BUBBLE & aEvent->flags))) {
aEvent->flags |= aFlags;
mListenerManager->HandleEvent(aPresContext, aEvent, aDOMEvent, this,
aFlags, aEventStatus);
aEvent->flags &= ~aFlags;
}
if (aEvent->message == NS_PAGE_LOAD)
mIsDocumentLoaded = PR_TRUE;
// Bubbling stage
if ((NS_EVENT_FLAG_CAPTURE != aFlags) && mChromeEventHandler) {
// Bubble to a chrome document if it exists
// XXX Need a way to know if an event should really bubble or not.
// For now filter out load and unload, since they cause problems.
if ((aEvent->message != NS_PAGE_LOAD) &&
(aEvent->message != NS_PAGE_UNLOAD) &&
(aEvent->message != NS_IMAGE_LOAD) &&
(aEvent->message != NS_FOCUS_CONTENT) &&
(aEvent->message != NS_BLUR_CONTENT)) {
mChromeEventHandler->HandleChromeEvent(aPresContext, aEvent,
aDOMEvent, NS_EVENT_FLAG_BUBBLE,
aEventStatus);
}
}
if (NS_EVENT_FLAG_INIT & aFlags) {
// We're leaving the DOM event loop so if we created an event,
// release here.
if (*aDOMEvent && !externalDOMEvent) {
nsrefcnt rc;
NS_RELEASE2(*aDOMEvent, rc);
if (rc) {
// Okay, so someone in the DOM loop (a listener, JS object) still has
// a ref to the DOM Event but the internal data hasn't been malloc'd.
// Force a copy of the data here so the DOM Event is still valid.
nsCOMPtr<nsIPrivateDOMEvent>
privateEvent(do_QueryInterface(*aDOMEvent));
if (privateEvent)
privateEvent->DuplicatePrivateData();
}
aDOMEvent = nsnull;
}
}
return ret;
1998-07-16 01:16:47 +00:00
}
JSObject *
GlobalWindowImpl::GetGlobalJSObject()
{
return mJSObject;
}
NS_IMETHODIMP
GlobalWindowImpl::OnFinalize(JSObject *aJSObject)
{
if (aJSObject == mJSObject) {
mJSObject = nsnull;
} else if (mJSObject) {
NS_ERROR("Huh? XPConnect created more than one wrapper for this global!");
} else {
NS_WARNING("Weird, we're finalized with a null mJSObject?");
}
return NS_OK;
}
NS_IMETHODIMP
GlobalWindowImpl::SetScriptsEnabled(PRBool aEnabled)
{
if (aEnabled) {
// Scripts are enabled (again?) on this context, run timeouts that
// fired on this context while scripts were disabled.
RunTimeout(nsnull);
}
return NS_OK;
}
//*****************************************************************************
// GlobalWindowImpl::nsIScriptObjectPrincipal
//*****************************************************************************
NS_IMETHODIMP GlobalWindowImpl::GetPrincipal(nsIPrincipal** result)
1998-07-16 01:16:47 +00:00
{
NS_ENSURE_ARG_POINTER(result);
if (mDocument) {
// If we have a document, get the principal from the document
nsCOMPtr<nsIDocument> doc(do_QueryInterface(mDocument));
NS_ENSURE_TRUE(doc, NS_ERROR_FAILURE);
return doc->GetPrincipal(result);
}
if (mDocumentPrincipal) {
*result = mDocumentPrincipal;
NS_ADDREF(*result);
return NS_OK;
}
// If we don't have a principal and we don't have a document we
// ask the parent window for the principal. This can happen when
// loading a frameset that has a <frame src="javascript:xxx">, in
// that case the global window is used in JS before we've loaded
// a document into the window.
nsCOMPtr<nsIDOMWindowInternal> parent;
GetParentInternal(getter_AddRefs(parent));
if (parent) {
nsCOMPtr<nsIScriptObjectPrincipal> objPrincipal(do_QueryInterface(parent));
if (objPrincipal) {
return objPrincipal->GetPrincipal(result);
}
}
return NS_ERROR_FAILURE;
}
1998-07-16 01:16:47 +00:00
//*****************************************************************************
// GlobalWindowImpl::nsIDOMWindow
//*****************************************************************************
NS_IMETHODIMP GlobalWindowImpl::GetDocument(nsIDOMDocument** aDocument)
{
/* lazily instantiate an about:blank document if necessary, and if we have
what it takes to do so. Note that domdoc here is the same thing as
our mDocument, but we don't have to explicitly set the member variable
because the docshell has already called SetNewDocument(). */
if (!mDocument && mDocShell)
nsCOMPtr<nsIDOMDocument> domdoc(do_GetInterface(mDocShell));
*aDocument = mDocument;
NS_IF_ADDREF(*aDocument);
return NS_OK;
1998-07-16 01:16:47 +00:00
}
//*****************************************************************************
// GlobalWindowImpl::nsIDOMWindowInternal
//*****************************************************************************
NS_IMETHODIMP GlobalWindowImpl::GetWindow(nsIDOMWindowInternal** aWindow)
1998-07-16 01:16:47 +00:00
{
*aWindow = NS_STATIC_CAST(nsIDOMWindowInternal *, this);
NS_ADDREF(*aWindow);
return NS_OK;
}
1998-07-16 01:16:47 +00:00
NS_IMETHODIMP GlobalWindowImpl::GetSelf(nsIDOMWindowInternal** aWindow)
{
*aWindow = NS_STATIC_CAST(nsIDOMWindowInternal *, this);
NS_ADDREF(*aWindow);
return NS_OK;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::GetNavigator(nsIDOMNavigator** aNavigator)
1998-07-16 01:16:47 +00:00
{
if (!mNavigator) {
mNavigator = new NavigatorImpl(mDocShell);
NS_ENSURE_TRUE(mNavigator, NS_ERROR_OUT_OF_MEMORY);
NS_ADDREF(mNavigator);
}
1998-07-16 01:16:47 +00:00
*aNavigator = mNavigator;
NS_ADDREF(*aNavigator);
1998-07-16 01:16:47 +00:00
return NS_OK;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::GetScreen(nsIDOMScreen** aScreen)
{
if (!mScreen && mDocShell) {
mScreen = new ScreenImpl(mDocShell);
NS_ENSURE_TRUE(mScreen, NS_ERROR_OUT_OF_MEMORY);
NS_ADDREF(mScreen);
}
*aScreen = mScreen;
NS_ADDREF(*aScreen);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetHistory(nsIDOMHistory** aHistory)
{
if (!mHistory && mDocShell) {
mHistory = new HistoryImpl(mDocShell);
NS_ENSURE_TRUE(mHistory, NS_ERROR_OUT_OF_MEMORY);
NS_ADDREF(mHistory);
}
*aHistory = mHistory;
NS_ADDREF(*aHistory);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetParent(nsIDOMWindow** aParent)
1999-06-09 19:04:18 +00:00
{
NS_ENSURE_ARG_POINTER(aParent);
*aParent = nsnull;
if (!mDocShell)
return NS_OK;
1999-06-09 19:04:18 +00:00
nsCOMPtr<nsIDocShellTreeItem> docShellAsItem(do_QueryInterface(mDocShell));
NS_ENSURE_TRUE(docShellAsItem, NS_ERROR_FAILURE);
nsCOMPtr<nsIDocShellTreeItem> parent;
docShellAsItem->GetSameTypeParent(getter_AddRefs(parent));
if (parent) {
nsCOMPtr<nsIScriptGlobalObject> globalObject(do_GetInterface(parent));
NS_ENSURE_SUCCESS(CallQueryInterface(globalObject.get(), aParent),
NS_ERROR_FAILURE);
}
else {
*aParent = NS_STATIC_CAST(nsIDOMWindowInternal *, this);
NS_ADDREF(*aParent);
}
return NS_OK;
1999-06-09 19:04:18 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::GetTop(nsIDOMWindow** aTop)
1999-06-09 19:04:18 +00:00
{
nsresult ret = NS_OK;
*aTop = nsnull;
if (mDocShell) {
nsCOMPtr<nsIDocShellTreeItem> docShellAsItem(do_QueryInterface(mDocShell));
nsCOMPtr<nsIDocShellTreeItem> root;
docShellAsItem->GetSameTypeRootTreeItem(getter_AddRefs(root));
if (root) {
nsCOMPtr<nsIScriptGlobalObject> globalObject(do_GetInterface(root));
CallQueryInterface(globalObject.get(), aTop);
}
}
1999-06-09 19:04:18 +00:00
return ret;
1999-06-09 19:04:18 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::GetContent(nsIDOMWindow** aContent)
1999-06-09 19:04:18 +00:00
{
*aContent = nsnull;
nsCOMPtr<nsIDocShellTreeOwner> treeOwner;
GetTreeOwner(getter_AddRefs(treeOwner));
NS_ENSURE_TRUE(treeOwner, NS_ERROR_FAILURE);
nsCOMPtr<nsIDocShellTreeItem> primaryContent;
treeOwner->GetPrimaryContentShell(getter_AddRefs(primaryContent));
nsCOMPtr<nsIDOMWindowInternal> domWindow(do_GetInterface(primaryContent));
*aContent = domWindow;
NS_IF_ADDREF(*aContent);
1999-06-09 19:04:18 +00:00
return NS_OK;
}
2000-02-16 07:13:59 +00:00
// XXX for sidebar HACK, see bug 20721
NS_IMETHODIMP GlobalWindowImpl::GetSidebar(nsISidebar** aSidebar)
{
nsresult rv = NS_OK;
if (!mSidebar) {
mSidebar = do_CreateInstance(NS_SIDEBAR_CONTRACTID, &rv);
2000-02-16 07:13:59 +00:00
if (mSidebar) {
nsIDOMWindowInternal *win = NS_STATIC_CAST(nsIDOMWindowInternal *, this);
2000-02-16 07:13:59 +00:00
/* no addref */
mSidebar->SetWindow(win);
}
}
*aSidebar = mSidebar;
NS_IF_ADDREF(*aSidebar);
return rv;
}
NS_IMETHODIMP GlobalWindowImpl::GetPrompter(nsIPrompt** aPrompt)
{
if (!mDocShell)
return NS_ERROR_FAILURE;
nsCOMPtr<nsIPrompt> prompter(do_GetInterface(mDocShell));
NS_ENSURE_TRUE(prompter, NS_ERROR_NO_INTERFACE);
NS_ADDREF(*aPrompt = prompter);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetMenubar(nsIDOMBarProp** aMenubar)
1999-06-09 19:04:18 +00:00
{
if (!mMenubar) {
mMenubar = new MenubarPropImpl();
if (mMenubar) {
NS_ADDREF(mMenubar);
nsCOMPtr<nsIWebBrowserChrome> browserChrome;
if (mDocShell &&
NS_SUCCEEDED(GetWebBrowserChrome(getter_AddRefs(browserChrome)))) {
mMenubar-> SetWebBrowserChrome(browserChrome);
1999-06-19 22:15:14 +00:00
}
}
}
1999-06-09 19:04:18 +00:00
*aMenubar = mMenubar;
NS_IF_ADDREF(mMenubar);
return NS_OK;
1999-06-09 19:04:18 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::GetToolbar(nsIDOMBarProp** aToolbar)
1999-06-09 19:04:18 +00:00
{
if (!mToolbar) {
mToolbar = new ToolbarPropImpl();
if (mToolbar) {
NS_ADDREF(mToolbar);
nsCOMPtr<nsIWebBrowserChrome> browserChrome;
if (mDocShell &&
NS_SUCCEEDED(GetWebBrowserChrome(getter_AddRefs(browserChrome)))) {
mToolbar->SetWebBrowserChrome(browserChrome);
1999-06-19 22:15:14 +00:00
}
}
}
1999-06-09 19:04:18 +00:00
*aToolbar = mToolbar;
NS_IF_ADDREF(mToolbar);
return NS_OK;
1999-06-09 19:04:18 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::GetLocationbar(nsIDOMBarProp** aLocationbar)
1999-06-09 19:04:18 +00:00
{
if (!mLocationbar) {
mLocationbar = new LocationbarPropImpl();
if (mLocationbar) {
NS_ADDREF(mLocationbar);
nsCOMPtr<nsIWebBrowserChrome> browserChrome;
if (mDocShell &&
NS_SUCCEEDED(GetWebBrowserChrome(getter_AddRefs(browserChrome)))) {
mLocationbar->SetWebBrowserChrome(browserChrome);
1999-06-19 22:15:14 +00:00
}
}
}
1999-06-09 19:04:18 +00:00
*aLocationbar = mLocationbar;
NS_IF_ADDREF(mLocationbar);
return NS_OK;
1999-06-09 19:04:18 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::GetPersonalbar(nsIDOMBarProp** aPersonalbar)
1999-06-09 19:04:18 +00:00
{
if (!mPersonalbar) {
mPersonalbar = new PersonalbarPropImpl();
if (mPersonalbar) {
NS_ADDREF(mPersonalbar);
nsCOMPtr<nsIWebBrowserChrome> browserChrome;
if (mDocShell &&
NS_SUCCEEDED(GetWebBrowserChrome(getter_AddRefs(browserChrome)))) {
mPersonalbar->SetWebBrowserChrome(browserChrome);
}
}
}
*aPersonalbar = mPersonalbar;
NS_IF_ADDREF(mPersonalbar);
return NS_OK;
1999-06-09 19:04:18 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::GetStatusbar(nsIDOMBarProp** aStatusbar)
{
if (!mStatusbar) {
mStatusbar = new StatusbarPropImpl();
if (mStatusbar) {
NS_ADDREF(mStatusbar);
nsCOMPtr<nsIWebBrowserChrome> browserChrome;
if (mDocShell &&
NS_SUCCEEDED(GetWebBrowserChrome(getter_AddRefs(browserChrome)))) {
mStatusbar->SetWebBrowserChrome(browserChrome);
}
}
}
*aStatusbar = mStatusbar;
NS_IF_ADDREF(mStatusbar);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetScrollbars(nsIDOMBarProp** aScrollbars)
{
if (!mScrollbars) {
mScrollbars = new ScrollbarsPropImpl();
if (mScrollbars) {
NS_ADDREF(mScrollbars);
nsCOMPtr<nsIWebBrowserChrome> browserChrome;
if (mDocShell &&
NS_SUCCEEDED(GetWebBrowserChrome(getter_AddRefs(browserChrome)))) {
mScrollbars->SetWebBrowserChrome(browserChrome);
}
}
}
*aScrollbars = mScrollbars;
NS_IF_ADDREF(mScrollbars);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetDirectories(nsIDOMBarProp** aDirectories)
{
return GetPersonalbar(aDirectories);
}
NS_IMETHODIMP GlobalWindowImpl::GetClosed(PRBool* aClosed)
{
*aClosed = !mDocShell;
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetFrames(nsIDOMWindowCollection** aFrames)
{
if (!mFrames && mDocShell) {
mFrames = new nsDOMWindowList(mDocShell);
NS_ENSURE_TRUE(mFrames, NS_ERROR_OUT_OF_MEMORY);
NS_ADDREF(mFrames);
}
*aFrames = NS_STATIC_CAST(nsIDOMWindowCollection *, mFrames);
NS_IF_ADDREF(mFrames);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetCrypto(nsIDOMCrypto** aCrypto)
{
nsresult rv;
if (!mCrypto) {
mCrypto = do_CreateInstance(kCryptoContractID, &rv);
}
*aCrypto = mCrypto;
NS_IF_ADDREF(*aCrypto);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetPkcs11(nsIDOMPkcs11** aPkcs11)
{
nsresult rv;
if (!mPkcs11) {
mPkcs11 = do_CreateInstance(kPkcs11ContractID, &rv);
}
*aPkcs11 = mPkcs11;
NS_IF_ADDREF(*aPkcs11);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetControllers(nsIControllers** aResult)
{
if (!mControllers) {
mControllers = do_CreateInstance(kXULControllersCID);
NS_ENSURE_TRUE(mControllers, NS_ERROR_FAILURE);
#ifdef DOM_CONTROLLER
// Add in the default controller
nsDOMWindowController *domController = new nsDOMWindowController(this);
if (domController) {
nsCOMPtr<nsIController> controller(domController);
mControllers->AppendController(controller);
}
#endif // DOM_CONTROLLER
}
*aResult = mControllers;
NS_ADDREF(*aResult);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetOpener(nsIDOMWindowInternal** aOpener)
1998-08-13 04:34:53 +00:00
{
*aOpener = mOpener;
NS_IF_ADDREF(*aOpener);
return NS_OK;
1998-08-13 04:34:53 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::SetOpener(nsIDOMWindowInternal* aOpener)
{
mOpener = aOpener;
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetStatus(nsAWritableString& aStatus)
{
aStatus = mStatus;
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::SetStatus(const nsAReadableString& aStatus)
1999-06-26 21:48:29 +00:00
{
mStatus = aStatus;
1999-06-26 21:48:29 +00:00
nsCOMPtr<nsIWebBrowserChrome> browserChrome;
GetWebBrowserChrome(getter_AddRefs(browserChrome));
if(browserChrome)
browserChrome->SetStatus(nsIWebBrowserChrome::STATUS_SCRIPT,
PromiseFlatString(aStatus).get());
1999-06-26 21:48:29 +00:00
return NS_OK;
1999-06-26 21:48:29 +00:00
}
NS_IMETHODIMP
GlobalWindowImpl::GetDefaultStatus(nsAWritableString& aDefaultStatus)
{
aDefaultStatus = mDefaultStatus;
return NS_OK;
}
NS_IMETHODIMP
GlobalWindowImpl::SetDefaultStatus(const nsAReadableString& aDefaultStatus)
{
mDefaultStatus = aDefaultStatus;
nsCOMPtr<nsIWebBrowserChrome> browserChrome;
GetWebBrowserChrome(getter_AddRefs(browserChrome));
if(browserChrome)
browserChrome->SetStatus(nsIWebBrowserChrome::STATUS_SCRIPT_DEFAULT,
PromiseFlatString(aDefaultStatus).get());
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetName(nsAWritableString& aName)
{
nsXPIDLString name;
nsCOMPtr<nsIDocShellTreeItem> docShellAsItem(do_QueryInterface(mDocShell));
if (docShellAsItem)
docShellAsItem->GetName(getter_Copies(name));
aName.Assign(name);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::SetName(const nsAReadableString& aName)
{
nsresult result = NS_OK;
nsCOMPtr<nsIDocShellTreeItem> docShellAsItem(do_QueryInterface(mDocShell));
if (docShellAsItem)
2001-04-02 19:40:52 +00:00
result = docShellAsItem->SetName(PromiseFlatString(aName).get());
return result;
}
NS_IMETHODIMP
GlobalWindowImpl::GetTitle(nsAWritableString& aTitle)
{
aTitle = mTitle;
return NS_OK;
}
NS_IMETHODIMP
GlobalWindowImpl::SetTitle(const nsAReadableString& aTitle)
{
mTitle = aTitle;
if(mDocShell) {
// See if we're a chrome shell.
PRInt32 type;
nsCOMPtr<nsIDocShellTreeItem> docShellAsItem(do_QueryInterface(mDocShell));
docShellAsItem->GetItemType(&type);
if(type == nsIDocShellTreeItem::typeChrome) {
nsCOMPtr<nsIBaseWindow> docShellAsWin(do_QueryInterface(mDocShell));
if(docShellAsWin) {
docShellAsWin->SetTitle(PromiseFlatString(mTitle).get());
}
}
}
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetInnerWidth(PRInt32* aInnerWidth)
{
nsCOMPtr<nsIDOMWindowInternal> parent;
GetParentInternal(getter_AddRefs(parent));
if (parent) {
PRInt32 dummy;
// Force a flush in the parent
parent->GetInnerWidth(&dummy);
}
FlushPendingNotifications();
nsCOMPtr<nsIBaseWindow> docShellWin(do_QueryInterface(mDocShell));
*aInnerWidth = 0;
if (docShellWin)
docShellWin->GetSize(aInnerWidth, nsnull);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::SetInnerWidth(PRInt32 aInnerWidth)
{
nsCOMPtr<nsIDocShellTreeItem> docShellAsItem(do_QueryInterface(mDocShell));
NS_ENSURE_TRUE(docShellAsItem, NS_ERROR_FAILURE);
nsCOMPtr<nsIDocShellTreeItem> docShellParent;
docShellAsItem->GetSameTypeParent(getter_AddRefs(docShellParent));
// It's only valid to access this from a top window. Doesn't work from
// sub-frames.
if (docShellParent)
return NS_OK; // Silent failure
nsCOMPtr<nsIDocShellTreeOwner> treeOwner;
docShellAsItem->GetTreeOwner(getter_AddRefs(treeOwner));
NS_ENSURE_TRUE(treeOwner, NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(CheckSecurityWidthAndHeight(&aInnerWidth, nsnull),
NS_ERROR_FAILURE);
nsCOMPtr<nsIBaseWindow> docShellAsWin(do_QueryInterface(mDocShell));
PRInt32 cy = 0;
docShellAsWin->GetSize(nsnull, &cy);
NS_ENSURE_SUCCESS(treeOwner->SizeShellTo(docShellAsItem, aInnerWidth, cy),
NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetInnerHeight(PRInt32* aInnerHeight)
{
nsCOMPtr<nsIDOMWindowInternal> parent;
GetParentInternal(getter_AddRefs(parent));
if (parent) {
PRInt32 dummy;
// Force a flush in the parent
parent->GetInnerHeight(&dummy);
}
FlushPendingNotifications();
nsCOMPtr<nsIBaseWindow> docShellWin(do_QueryInterface(mDocShell));
*aInnerHeight = 0;
if (docShellWin)
docShellWin->GetSize(nsnull, aInnerHeight);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::SetInnerHeight(PRInt32 aInnerHeight)
{
nsCOMPtr<nsIDocShellTreeItem> docShellAsItem(do_QueryInterface(mDocShell));
NS_ENSURE_TRUE(docShellAsItem, NS_ERROR_FAILURE);
nsCOMPtr<nsIDocShellTreeItem> docShellParent;
docShellAsItem->GetSameTypeParent(getter_AddRefs(docShellParent));
// It's only valid to access this from a top window. Doesn't work from
// sub-frames.
if (docShellParent)
return NS_OK; // Silent failure
nsCOMPtr<nsIDocShellTreeOwner> treeOwner;
docShellAsItem->GetTreeOwner(getter_AddRefs(treeOwner));
NS_ENSURE_TRUE(treeOwner, NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(CheckSecurityWidthAndHeight(nsnull, &aInnerHeight),
NS_ERROR_FAILURE);
nsCOMPtr<nsIBaseWindow> docShellAsWin(do_QueryInterface(mDocShell));
PRInt32 cx = 0;
docShellAsWin->GetSize(&cx, nsnull);
NS_ENSURE_SUCCESS(treeOwner->
SizeShellTo(docShellAsItem, cx, aInnerHeight),
NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetOuterWidth(PRInt32* aOuterWidth)
{
nsCOMPtr<nsIDOMWindowInternal> parent;
GetParentInternal(getter_AddRefs(parent));
if (parent) {
PRInt32 dummy;
// Force a flush in the parent
parent->GetOuterWidth(&dummy);
}
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
FlushPendingNotifications();
NS_ENSURE_SUCCESS(treeOwnerAsWin->GetSize(aOuterWidth, nsnull),
NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::SetOuterWidth(PRInt32 aOuterWidth)
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(CheckSecurityWidthAndHeight(&aOuterWidth, nsnull),
NS_ERROR_FAILURE);
PRInt32 cy;
NS_ENSURE_SUCCESS(treeOwnerAsWin->GetSize(nsnull, &cy), NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(treeOwnerAsWin->SetSize(aOuterWidth, cy, PR_TRUE),
NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetOuterHeight(PRInt32* aOuterHeight)
{
nsCOMPtr<nsIDOMWindowInternal> parent;
GetParentInternal(getter_AddRefs(parent));
if (parent) {
PRInt32 dummy;
// Force a flush in the parent
parent->GetOuterHeight(&dummy);
}
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
FlushPendingNotifications();
NS_ENSURE_SUCCESS(treeOwnerAsWin->GetSize(nsnull, aOuterHeight),
NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::SetOuterHeight(PRInt32 aOuterHeight)
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(CheckSecurityWidthAndHeight(nsnull, &aOuterHeight),
NS_ERROR_FAILURE);
PRInt32 cx;
NS_ENSURE_SUCCESS(treeOwnerAsWin->GetSize(&cx, nsnull), NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(treeOwnerAsWin->SetSize(cx, aOuterHeight, PR_TRUE),
NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetScreenX(PRInt32* aScreenX)
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
PRInt32 y;
NS_ENSURE_SUCCESS(treeOwnerAsWin->GetPosition(aScreenX, &y),
NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::SetScreenX(PRInt32 aScreenX)
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(CheckSecurityLeftAndTop(&aScreenX, nsnull),
NS_ERROR_FAILURE);
PRInt32 x, y;
NS_ENSURE_SUCCESS(treeOwnerAsWin->GetPosition(&x, &y),
NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(treeOwnerAsWin->SetPosition(aScreenX, y),
NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetScreenY(PRInt32* aScreenY)
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
PRInt32 x;
NS_ENSURE_SUCCESS(treeOwnerAsWin->GetPosition(&x, aScreenY),
NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::SetScreenY(PRInt32 aScreenY)
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(CheckSecurityLeftAndTop(nsnull, &aScreenY),
NS_ERROR_FAILURE);
PRInt32 x, y;
NS_ENSURE_SUCCESS(treeOwnerAsWin->GetPosition(&x, &y),
NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(treeOwnerAsWin->SetPosition(x, aScreenY),
NS_ERROR_FAILURE);
return NS_OK;
}
nsresult
GlobalWindowImpl::CheckSecurityWidthAndHeight(PRInt32* aWidth,
PRInt32* aHeight)
{
// This one is easy. Just ensure the variable is greater than 100;
if ((aWidth && *aWidth < 100) || (aHeight && *aHeight < 100)) {
// Check security state for use in determing window dimensions
nsCOMPtr<nsIScriptSecurityManager>
securityManager(do_GetService(NS_SCRIPTSECURITYMANAGER_CONTRACTID));
NS_ENSURE_TRUE(securityManager, NS_ERROR_FAILURE);
PRBool enabled;
nsresult res =
securityManager->IsCapabilityEnabled("UniversalBrowserWrite", &enabled);
if (NS_FAILED(res) || !enabled) {
//sec check failed
if (aWidth && *aWidth < 100) {
*aWidth = 100;
}
if (aHeight && *aHeight < 100) {
*aHeight = 100;
}
}
}
return NS_OK;
}
nsresult
GlobalWindowImpl::CheckSecurityLeftAndTop(PRInt32* aLeft, PRInt32* aTop)
{
// This one is harder. We have to get the screen size and window dimensions.
// Check security state for use in determing window dimensions
nsCOMPtr<nsIScriptSecurityManager>
securityManager(do_GetService(NS_SCRIPTSECURITYMANAGER_CONTRACTID));
NS_ENSURE_TRUE(securityManager, NS_ERROR_FAILURE);
PRBool enabled;
nsresult res =
securityManager->IsCapabilityEnabled("UniversalBrowserWrite", &enabled);
if (NS_FAILED(res)) {
enabled = PR_FALSE;
}
if (!enabled) {
PRInt32 screenLeft, screenTop, screenWidth, screenHeight;
PRInt32 winLeft, winTop, winWidth, winHeight;
FlushPendingNotifications();
// Get the window size
nsCOMPtr<nsIBaseWindow> treeOwner;
GetTreeOwner(getter_AddRefs(treeOwner));
if (treeOwner)
treeOwner->GetPositionAndSize(&winLeft, &winTop, &winWidth, &winHeight);
// Get the screen dimensions
// XXX This should use nsIScreenManager once it's fully fleshed out.
nsCOMPtr<nsIDOMScreen> screen;
GetScreen(getter_AddRefs(screen));
if (screen) {
screen->GetAvailLeft(&screenLeft);
screen->GetAvailWidth(&screenWidth);
screen->GetAvailHeight(&screenHeight);
#if defined(XP_MAC) || defined(XP_MACOSX)
/* The mac's coordinate system is different from the assumed Windows'
system. It offsets by the height of the menubar so that a window
placed at (0,0) will be entirely visible. Unfortunately that
correction is made elsewhere (in Widget) and the meaning of
the Avail... coordinates is overloaded. Here we allow a window
to be placed at (0,0) because it does make sense to do so.
*/
screen->GetTop(&screenTop);
#else
screen->GetAvailTop(&screenTop);
#endif
}
if (screen && treeOwner) {
if (aLeft) {
if (screenLeft+screenWidth < *aLeft+winWidth)
*aLeft = screenLeft+screenWidth - winWidth;
if (screenLeft > *aLeft)
*aLeft = screenLeft;
}
if (aTop) {
if (screenTop+screenHeight < *aTop+winHeight)
*aTop = screenTop+screenHeight - winHeight;
if (screenTop > *aTop)
*aTop = screenTop;
}
} else {
if (aLeft)
*aLeft = 0;
if (aTop)
*aTop = 0;
}
}
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetPageXOffset(PRInt32* aPageXOffset)
{
return GetScrollX(aPageXOffset);
}
NS_IMETHODIMP GlobalWindowImpl::GetPageYOffset(PRInt32* aPageYOffset)
{
return GetScrollY(aPageYOffset);
}
NS_IMETHODIMP GlobalWindowImpl::GetScrollX(PRInt32* aScrollX)
{
NS_ENSURE_ARG_POINTER(aScrollX);
nsresult result = NS_OK;
nsIScrollableView *view = nsnull; // no addref/release for views
float p2t, t2p;
*aScrollX = 0;
GetScrollInfo(&view, &p2t, &t2p);
if (view) {
nscoord xPos, yPos;
result = view->GetScrollPosition(xPos, yPos);
*aScrollX = NSTwipsToIntPixels(xPos, t2p);
}
return result;
}
NS_IMETHODIMP GlobalWindowImpl::GetScrollY(PRInt32* aScrollY)
{
NS_ENSURE_ARG_POINTER(aScrollY);
nsresult result = NS_OK;
nsIScrollableView *view = nsnull; // no addref/release for views
float p2t, t2p;
*aScrollY = 0;
GetScrollInfo(&view, &p2t, &t2p);
if (view) {
nscoord xPos, yPos;
result = view->GetScrollPosition(xPos, yPos);
*aScrollY = NSTwipsToIntPixels(yPos, t2p);
}
return result;
}
NS_IMETHODIMP GlobalWindowImpl::GetLength(PRUint32* aLength)
{
nsCOMPtr<nsIDOMWindowCollection> frames;
if (NS_SUCCEEDED(GetFrames(getter_AddRefs(frames))) && frames) {
return frames->GetLength(aLength);
}
return NS_ERROR_FAILURE;
}
NS_IMETHODIMP GlobalWindowImpl::Dump(const nsAReadableString& aStr)
1998-07-16 01:16:47 +00:00
{
#if !(defined(NS_DEBUG) || defined(MOZ_ENABLE_JS_DUMP))
{
// In optimized builds we check a pref that controls if we should
// enable output from dump() or not, in debug builds it's always
// enabled.
nsCOMPtr<nsIPref> prefs(do_GetService(kPrefServiceCID));
if (!prefs)
return NS_OK;
PRBool enable_dump = PR_FALSE;
// if pref doesn't exist, disable dump output.
nsresult rv = prefs->GetBoolPref("browser.dom.window.dump.enabled",
&enable_dump);
if (NS_FAILED(rv) || !enable_dump) {
return NS_OK;
}
}
#endif
char *cstr = ToNewUTF8String(aStr);
#ifdef XP_MAC
// have to convert \r to \n so that printing to the console works
char *c = cstr, *cEnd = cstr + aStr.Length();
while (c < cEnd) {
if (*c == '\r')
*c = '\n';
c++;
}
#endif
if (cstr) {
printf("%s", cstr);
nsCRT::free(cstr);
}
1998-07-16 01:16:47 +00:00
return NS_OK;
}
static void EnsureReflowFlushAndPaint(nsIDocShell* aDocShell)
{
if (!aDocShell)
return;
nsCOMPtr<nsIPresShell> presShell;
aDocShell->GetPresShell(getter_AddRefs(presShell));
if (!presShell)
return;
// Flush pending reflows.
presShell->FlushPendingNotifications(PR_FALSE);
// Unsuppress painting.
2001-05-01 22:54:11 +00:00
presShell->UnsuppressPainting();
}
NS_IMETHODIMP
GlobalWindowImpl::Alert(const nsAReadableString& aString)
{
NS_ENSURE_STATE(mDocShell);
nsAutoString str;
str.Assign(aString);
// XXX: Concatenation of optional args?
nsCOMPtr<nsIPrompt> prompter(do_GetInterface(mDocShell));
NS_ENSURE_TRUE(prompter, NS_ERROR_FAILURE);
// Before bringing up the window, unsuppress painting and flush
// pending reflows.
EnsureReflowFlushAndPaint(mDocShell);
return prompter->Alert(nsnull, str.get());
}
NS_IMETHODIMP
GlobalWindowImpl::Confirm(const nsAReadableString& aString, PRBool* aReturn)
{
NS_ENSURE_STATE(mDocShell);
nsAutoString str;
*aReturn = PR_FALSE;
str.Assign(aString);
// XXX: Concatenation of optional args?
nsCOMPtr<nsIPrompt> prompter(do_GetInterface(mDocShell));
NS_ENSURE_TRUE(prompter, NS_ERROR_FAILURE);
// Before bringing up the window, unsuppress painting and flush
// pending reflows.
EnsureReflowFlushAndPaint(mDocShell);
return prompter->Confirm(nsnull, str.get(), aReturn);
}
NS_IMETHODIMP
GlobalWindowImpl::Prompt(const nsAReadableString& aMessage,
const nsAReadableString& aInitial,
const nsAReadableString& aTitle,
PRUint32 aSavePassword,
nsAWritableString& aReturn)
{
NS_ENSURE_STATE(mDocShell);
aReturn.Truncate(); // XXX Null string!!!
nsresult rv = NS_OK;
nsCOMPtr<nsIAuthPrompt> prompter(do_GetInterface(mDocShell));
NS_ENSURE_TRUE(prompter, NS_ERROR_FAILURE);
PRBool b;
nsXPIDLString uniResult;
// Before bringing up the window, unsuppress painting and flush
// pending reflows.
EnsureReflowFlushAndPaint(mDocShell);
rv = prompter->Prompt(PromiseFlatString(aTitle).get(),
PromiseFlatString(aMessage).get(), nsnull,
aSavePassword,
PromiseFlatString(aInitial).get(),
getter_Copies(uniResult), &b);
NS_ENSURE_SUCCESS(rv, rv);
if (uniResult && b) {
aReturn.Assign(uniResult);
}
else {
SetDOMStringToNull(aReturn);
// XXX: Since DOMString's can't be null yet we'll haveto do this here...
if (sXPConnect) {
nsCOMPtr<nsIXPCNativeCallContext> ncc;
sXPConnect->GetCurrentNativeCallContext(getter_AddRefs(ncc));
if (ncc) {
jsval *retval = nsnull;
rv = ncc->GetRetValPtr(&retval);
NS_ENSURE_SUCCESS(rv, rv);
*retval = JSVAL_NULL;
ncc->SetReturnValueWasSet(PR_TRUE);
}
}
}
return rv;
}
NS_IMETHODIMP
GlobalWindowImpl::Prompt(nsAWritableString& aReturn)
{
NS_ENSURE_STATE(mDocShell);
NS_ENSURE_STATE(sXPConnect);
nsresult rv = NS_OK;
nsCOMPtr<nsIXPCNativeCallContext> ncc;
rv = sXPConnect->GetCurrentNativeCallContext(getter_AddRefs(ncc));
NS_ENSURE_SUCCESS(rv, rv);
if (!ncc)
return NS_ERROR_NOT_AVAILABLE;
JSContext *cx = nsnull;
rv = ncc->GetJSContext(&cx);
NS_ENSURE_SUCCESS(rv, rv);
nsAutoString message, initial, title;
PRUint32 argc;
jsval *argv = nsnull;
ncc->GetArgc(&argc);
ncc->GetArgvPtr(&argv);
PRUint32 savePassword = nsIAuthPrompt::SAVE_PASSWORD_NEVER;
if (argc > 0) {
nsJSUtils::ConvertJSValToString(message, cx, argv[0]);
if (argc > 1) {
nsJSUtils::ConvertJSValToString(initial, cx, argv[1]);
if (argc > 2) {
nsJSUtils::ConvertJSValToString(title, cx, argv[2]);
if (argc > 3) {
nsJSUtils::ConvertJSValToUint32(&savePassword, cx, argv[3]);
}
}
}
}
return Prompt(message, initial, title, savePassword, aReturn);
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::Focus()
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
if (treeOwnerAsWin)
treeOwnerAsWin->SetVisibility(PR_TRUE);
nsCOMPtr<nsIPresShell> presShell;
if (mDocShell) {
mDocShell->GetEldestPresShell(getter_AddRefs(presShell));
}
nsresult result = NS_OK;
if (presShell) {
nsCOMPtr<nsIViewManager> vm;
presShell->GetViewManager(getter_AddRefs(vm));
if (vm) {
nsCOMPtr<nsIWidget> widget;
vm->GetWidget(getter_AddRefs(widget));
if (widget)
// raise the window since this was a focus call on the window.
result = widget->SetFocus(PR_TRUE);
}
}
else {
nsCOMPtr<nsIFocusController> focusController;
GetRootFocusController(getter_AddRefs(focusController));
if (focusController)
focusController->SetFocusedWindow(this);
}
return result;
}
NS_IMETHODIMP GlobalWindowImpl::Blur()
2000-01-26 22:48:16 +00:00
{
nsresult rv = NS_ERROR_FAILURE;
nsCOMPtr<nsIDocShellTreeOwner> treeOwner;
GetTreeOwner(getter_AddRefs(treeOwner));
nsCOMPtr<nsIEmbeddingSiteWindow2> siteWindow(do_QueryInterface(treeOwner));
if (siteWindow)
rv = siteWindow->Blur();
2000-01-26 22:48:16 +00:00
if (NS_SUCCEEDED(rv))
mDocShell->SetHasFocus(PR_FALSE);
return rv;
2000-01-26 22:48:16 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::Back()
{
nsCOMPtr<nsIWebNavigation> webNav(do_QueryInterface(mDocShell));
NS_ENSURE_TRUE(webNav, NS_ERROR_FAILURE);
return webNav->GoBack();
}
NS_IMETHODIMP GlobalWindowImpl::Forward()
{
nsCOMPtr<nsIWebNavigation> webNav(do_QueryInterface(mDocShell));
NS_ENSURE_TRUE(webNav, NS_ERROR_FAILURE);
return webNav->GoForward();
}
NS_IMETHODIMP GlobalWindowImpl::Home()
{
if (!mDocShell)
return NS_OK;
nsCOMPtr<nsIPref> prefs(do_GetService(kPrefServiceCID));
NS_ENSURE_TRUE(prefs, NS_ERROR_FAILURE);
// if we get here, we know prefs is not null
nsXPIDLString url;
prefs->GetLocalizedUnicharPref(PREF_BROWSER_STARTUP_HOMEPAGE,
getter_Copies(url));
nsString homeURL;
if (!url) {
// if all else fails, use this
#ifdef DEBUG_seth
printf("all else failed. using %s as the home page\n", DEFAULT_HOME_PAGE);
#endif
homeURL.AssignWithConversion(DEFAULT_HOME_PAGE);
}
else
homeURL = url;
nsresult rv;
nsCOMPtr<nsIWebNavigation> webNav(do_QueryInterface(mDocShell));
NS_ENSURE_TRUE(webNav, NS_ERROR_FAILURE);
rv = webNav->LoadURI(homeURL.get(),
nsIWebNavigation::LOAD_FLAGS_NONE,
nsnull,
nsnull,
nsnull);
NS_ENSURE_SUCCESS(rv, NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::Stop()
{
nsCOMPtr<nsIWebNavigation> webNav(do_QueryInterface(mDocShell));
return webNav->Stop(nsIWebNavigation::STOP_ALL);
}
nsresult GlobalWindowImpl::DoPrint(nsIPrintSettings* aPrintSettings,
PRBool aDoPreview)
{
if (mDocShell) {
nsCOMPtr<nsIContentViewer> viewer;
mDocShell->GetContentViewer(getter_AddRefs(viewer));
if (viewer) {
nsCOMPtr<nsIContentViewerFile> viewerFile(do_QueryInterface(viewer));
if (viewerFile) {
if (aDoPreview) {
return viewerFile->PrintPreview(aPrintSettings);
} else {
return viewerFile->Print(PR_FALSE, aPrintSettings, nsnull);
}
}
}
}
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::Print(nsIPrintSettings* aPrintSettings)
{
return DoPrint(aPrintSettings, PR_FALSE);
}
NS_IMETHODIMP GlobalWindowImpl::PrintPreview(nsIPrintSettings* aPrintSettings)
{
return DoPrint(aPrintSettings, PR_TRUE);
}
NS_IMETHODIMP GlobalWindowImpl::MoveTo(PRInt32 aXPos, PRInt32 aYPos)
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(CheckSecurityLeftAndTop(&aXPos, &aYPos),
NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(treeOwnerAsWin->SetPosition(aXPos, aYPos),
NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::MoveBy(PRInt32 aXDif, PRInt32 aYDif)
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
PRInt32 x, y;
NS_ENSURE_SUCCESS(treeOwnerAsWin->GetPosition(&x, &y), NS_ERROR_FAILURE);
PRInt32 newX = x + aXDif;
PRInt32 newY = y + aYDif;
NS_ENSURE_SUCCESS(CheckSecurityLeftAndTop(&newX, &newY), NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(treeOwnerAsWin->SetPosition(newX, newY),
NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::ResizeTo(PRInt32 aWidth, PRInt32 aHeight)
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(CheckSecurityWidthAndHeight(&aWidth, &aHeight),
NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(treeOwnerAsWin->SetSize(aWidth, aHeight, PR_TRUE),
NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::ResizeBy(PRInt32 aWidthDif, PRInt32 aHeightDif)
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
PRInt32 cx, cy;
NS_ENSURE_SUCCESS(treeOwnerAsWin->GetSize(&cx, &cy), NS_ERROR_FAILURE);
PRInt32 newCX = cx + aWidthDif;
PRInt32 newCY = cy + aHeightDif;
NS_ENSURE_SUCCESS(CheckSecurityWidthAndHeight(&newCX, &newCY),
NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(treeOwnerAsWin->SetSize(newCX, newCY,
PR_TRUE), NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::SizeToContent()
{
nsCOMPtr<nsIDocShellTreeItem> docShellAsItem(do_QueryInterface(mDocShell));
NS_ENSURE_TRUE(docShellAsItem, NS_ERROR_FAILURE);
nsCOMPtr<nsIDocShellTreeItem> docShellParent;
docShellAsItem->GetSameTypeParent(getter_AddRefs(docShellParent));
// It's only valid to access this from a top window. Doesn't work from
// sub-frames.
if (docShellParent)
return NS_ERROR_FAILURE;
nsCOMPtr<nsIContentViewer> cv;
mDocShell->GetContentViewer(getter_AddRefs(cv));
nsCOMPtr<nsIMarkupDocumentViewer> markupViewer(do_QueryInterface(cv));
NS_ENSURE_TRUE(markupViewer, NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(markupViewer->SizeToContent(), NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetAttention()
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
NS_ENSURE_TRUE(treeOwnerAsWin, NS_ERROR_FAILURE);
nsCOMPtr<nsIWidget> widget;
treeOwnerAsWin->GetMainWidget(getter_AddRefs(widget));
NS_ENSURE_TRUE(widget, NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(widget->GetAttention(), NS_ERROR_FAILURE);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::Scroll(PRInt32 aXScroll, PRInt32 aYScroll)
{
return ScrollTo(aXScroll, aYScroll);
}
NS_IMETHODIMP GlobalWindowImpl::ScrollTo(PRInt32 aXScroll, PRInt32 aYScroll)
{
nsresult result;
nsIScrollableView *view = nsnull; // no addref/release for views
float p2t, t2p;
result = GetScrollInfo(&view, &p2t, &t2p);
if (view) {
// Here we calculate what the max pixel value is that we can
// scroll to, we do this by dividing maxint with the pixel to
// twips conversion factor, and substracting 4, the 4 comes from
// experimenting with this value, anything less makes the view
// code not scroll correctly, I have no idea why. -- jst
const PRInt32 maxpx = (PRInt32)((float)0x7fffffff / p2t) - 4;
if (aXScroll > maxpx) {
aXScroll = maxpx;
}
if (aYScroll > maxpx) {
aYScroll = maxpx;
}
result = view->ScrollTo(NSIntPixelsToTwips(aXScroll, p2t),
NSIntPixelsToTwips(aYScroll, p2t),
NS_VMREFRESH_IMMEDIATE);
}
return result;
}
NS_IMETHODIMP GlobalWindowImpl::ScrollBy(PRInt32 aXScrollDif,
PRInt32 aYScrollDif)
{
nsresult result;
nsIScrollableView *view = nsnull; // no addref/release for views
float p2t, t2p;
result = GetScrollInfo(&view, &p2t, &t2p);
if (view) {
nscoord xPos, yPos;
result = view->GetScrollPosition(xPos, yPos);
if (NS_SUCCEEDED(result)) {
result = ScrollTo(NSTwipsToIntPixels(xPos, t2p) + aXScrollDif,
NSTwipsToIntPixels(yPos, t2p) + aYScrollDif);
}
}
return result;
}
NS_IMETHODIMP GlobalWindowImpl::ScrollByLines(PRInt32 numLines)
{
nsresult result;
nsIScrollableView *view = nsnull; // no addref/release for views
float p2t, t2p;
result = GetScrollInfo(&view, &p2t, &t2p);
if (view)
{
result = view->ScrollByLines(0, numLines);
}
return result;
}
NS_IMETHODIMP GlobalWindowImpl::ScrollByPages(PRInt32 numPages)
{
nsresult result;
nsIScrollableView *view = nsnull; // no addref/release for views
float p2t, t2p;
result = GetScrollInfo(&view, &p2t, &t2p);
if (view)
{
result = view->ScrollByPages(numPages);
}
return result;
}
NS_IMETHODIMP GlobalWindowImpl::ClearTimeout(PRInt32 aTimerID)
{
return ClearTimeoutOrInterval(aTimerID);
}
NS_IMETHODIMP GlobalWindowImpl::ClearInterval(PRInt32 aTimerID)
1999-01-27 04:15:19 +00:00
{
return ClearTimeoutOrInterval(aTimerID);
1999-01-27 04:15:19 +00:00
}
NS_IMETHODIMP
GlobalWindowImpl::SetTimeout(PRBool *_retval)
{
return SetTimeoutOrInterval(PR_FALSE, _retval);
}
NS_IMETHODIMP
GlobalWindowImpl::SetInterval(PRBool *_retval)
{
return SetTimeoutOrInterval(PR_TRUE, _retval);
}
NS_IMETHODIMP
GlobalWindowImpl::SetResizable(PRBool aResizable)
{
// nop
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::CaptureEvents(PRInt32 aEventFlags)
{
nsCOMPtr<nsIEventListenerManager> manager;
if (NS_SUCCEEDED(GetListenerManager(getter_AddRefs(manager)))) {
manager->CaptureEvent(aEventFlags);
return NS_OK;
}
return NS_ERROR_FAILURE;
}
NS_IMETHODIMP GlobalWindowImpl::ReleaseEvents(PRInt32 aEventFlags)
{
nsCOMPtr<nsIEventListenerManager> manager;
if (NS_SUCCEEDED(GetListenerManager(getter_AddRefs(manager)))) {
manager->ReleaseEvent(aEventFlags);
return NS_OK;
}
return NS_ERROR_FAILURE;
}
NS_IMETHODIMP GlobalWindowImpl::RouteEvent(nsIDOMEvent* aEvt)
{
//XXX Not the best solution -joki
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::EnableExternalCapture()
{
return NS_ERROR_FAILURE;
}
NS_IMETHODIMP GlobalWindowImpl::DisableExternalCapture()
{
return NS_ERROR_FAILURE;
}
//Note: This call will lock the cursor, it will not change as it moves.
//To unlock, the cursor must be set back to CURSOR_AUTO.
NS_IMETHODIMP GlobalWindowImpl::SetCursor(const nsAReadableString& aCursor)
{
nsresult ret = NS_OK;
PRInt32 cursor;
if (aCursor.Equals(NS_LITERAL_STRING("auto")))
cursor = NS_STYLE_CURSOR_AUTO;
else if (aCursor.Equals(NS_LITERAL_STRING("default")))
cursor = NS_STYLE_CURSOR_DEFAULT;
else if (aCursor.Equals(NS_LITERAL_STRING("pointer")))
cursor = NS_STYLE_CURSOR_POINTER;
else if (aCursor.Equals(NS_LITERAL_STRING("crosshair")))
cursor = NS_STYLE_CURSOR_CROSSHAIR;
else if (aCursor.Equals(NS_LITERAL_STRING("move")))
cursor = NS_STYLE_CURSOR_MOVE;
else if (aCursor.Equals(NS_LITERAL_STRING("text")))
cursor = NS_STYLE_CURSOR_TEXT;
else if (aCursor.Equals(NS_LITERAL_STRING("wait")))
cursor = NS_STYLE_CURSOR_WAIT;
else if (aCursor.Equals(NS_LITERAL_STRING("help")))
cursor = NS_STYLE_CURSOR_HELP;
else if (aCursor.Equals(NS_LITERAL_STRING("n-resize")))
cursor = NS_STYLE_CURSOR_N_RESIZE;
else if (aCursor.Equals(NS_LITERAL_STRING("s-resize")))
cursor = NS_STYLE_CURSOR_S_RESIZE;
else if (aCursor.Equals(NS_LITERAL_STRING("w-resize")))
cursor = NS_STYLE_CURSOR_W_RESIZE;
else if (aCursor.Equals(NS_LITERAL_STRING("e-resize")))
cursor = NS_STYLE_CURSOR_E_RESIZE;
else if (aCursor.Equals(NS_LITERAL_STRING("ne-resize")))
cursor = NS_STYLE_CURSOR_NE_RESIZE;
else if (aCursor.Equals(NS_LITERAL_STRING("nw-resize")))
cursor = NS_STYLE_CURSOR_NW_RESIZE;
else if (aCursor.Equals(NS_LITERAL_STRING("se-resize")))
cursor = NS_STYLE_CURSOR_SE_RESIZE;
else if (aCursor.Equals(NS_LITERAL_STRING("sw-resize")))
cursor = NS_STYLE_CURSOR_SW_RESIZE;
else if (aCursor.Equals(NS_LITERAL_STRING("copy")))
cursor = NS_STYLE_CURSOR_COPY; // CSS3
else if (aCursor.Equals(NS_LITERAL_STRING("alias")))
cursor = NS_STYLE_CURSOR_ALIAS;
else if (aCursor.Equals(NS_LITERAL_STRING("context-menu")))
cursor = NS_STYLE_CURSOR_CONTEXT_MENU;
else if (aCursor.Equals(NS_LITERAL_STRING("cell")))
cursor = NS_STYLE_CURSOR_CELL;
else if (aCursor.Equals(NS_LITERAL_STRING("grab")))
cursor = NS_STYLE_CURSOR_GRAB;
else if (aCursor.Equals(NS_LITERAL_STRING("grabbing")))
cursor = NS_STYLE_CURSOR_GRABBING;
else if (aCursor.Equals(NS_LITERAL_STRING("spinning")))
cursor = NS_STYLE_CURSOR_SPINNING;
else if (aCursor.Equals(NS_LITERAL_STRING("count-up")))
cursor = NS_STYLE_CURSOR_COUNT_UP;
else if (aCursor.Equals(NS_LITERAL_STRING("count-down")))
cursor = NS_STYLE_CURSOR_COUNT_DOWN;
else if (aCursor.Equals(NS_LITERAL_STRING("count-up-down")))
cursor = NS_STYLE_CURSOR_COUNT_UP_DOWN;
else
return NS_OK;
nsCOMPtr<nsIPresContext> presContext;
mDocShell->GetPresContext(getter_AddRefs(presContext));
if (presContext) {
nsCOMPtr<nsIEventStateManager> esm;
if (NS_SUCCEEDED(presContext->GetEventStateManager(getter_AddRefs(esm)))) {
// Need root widget.
nsCOMPtr<nsIPresShell> presShell;
mDocShell->GetPresShell(getter_AddRefs(presShell));
NS_ENSURE_TRUE(presShell, NS_ERROR_FAILURE);
nsCOMPtr<nsIViewManager> vm;
presShell->GetViewManager(getter_AddRefs(vm));
NS_ENSURE_TRUE(vm, NS_ERROR_FAILURE);
nsIView *rootView;
vm->GetRootView(rootView);
NS_ENSURE_TRUE(rootView, NS_ERROR_FAILURE);
nsCOMPtr<nsIWidget> widget;
rootView->GetWidget(*getter_AddRefs(widget));
NS_ENSURE_TRUE(widget, NS_ERROR_FAILURE);
// Call esm and set cursor.
ret = esm->SetCursor(cursor, widget, PR_TRUE);
}
}
return ret;
}
/*
* Examine the current document state to see if we're in a way that is typically
* abused by web designers. This routine returns PR_TRUE if we're running a top
* level script, running an onload or onunload handler, or running a timeout.
* The window.open code uses this routine to determine wether or not to allow
* the new window.
*/
PRBool
GlobalWindowImpl::CheckForAbusePoint ()
{
if (!mIsDocumentLoaded || mRunningTimeout) {
#ifdef DEBUG
printf ("*** Scripts executed during (un)load or as a result of "
"setTimeout() are potential javascript abuse points.\n");
#endif
return PR_TRUE;
}
return PR_FALSE;
}
NS_IMETHODIMP
GlobalWindowImpl::Open(const nsAReadableString& aUrl,
const nsAReadableString& aName,
const nsAReadableString& aOptions,
nsIDOMWindow **_retval)
{
return OpenInternal(aUrl, aName, aOptions, PR_FALSE, nsnull, 0, nsnull,
_retval);
}
NS_IMETHODIMP
GlobalWindowImpl::Open(nsIDOMWindow **_retval)
{
NS_ENSURE_STATE(sXPConnect);
nsresult rv = NS_OK;
/* If we're in a commonly abused state (top level script, running a timeout,
* or onload/onunload), and the preference is enabled, block the window.open().
*/
if (CheckForAbusePoint()) {
nsCOMPtr<nsIPref> prefs(do_GetService(kPrefServiceCID));
if (prefs) {
PRBool blockOpenOnLoad = PR_FALSE;
prefs->GetBoolPref("dom.disable_open_during_load", &blockOpenOnLoad);
if (blockOpenOnLoad) {
#ifdef DEBUG
printf ("*** Blocking window.open.\n");
#endif
*_retval = nsnull;
return NS_OK;
}
}
}
nsCOMPtr<nsIXPCNativeCallContext> ncc;
rv = sXPConnect->GetCurrentNativeCallContext(getter_AddRefs(ncc));
NS_ENSURE_SUCCESS(rv, rv);
if (!ncc)
return NS_ERROR_NOT_AVAILABLE;
JSContext *cx = nsnull;
rv = ncc->GetJSContext(&cx);
NS_ENSURE_SUCCESS(rv, rv);
nsAutoString url, name, options;
PRUint32 argc;
jsval *argv = nsnull;
ncc->GetArgc(&argc);
ncc->GetArgvPtr(&argv);
if (argc > 0) {
nsJSUtils::ConvertJSValToString(url, cx, argv[0]);
if (argc > 1) {
nsJSUtils::ConvertJSValToString(name, cx, argv[1]);
if (argc > 2) {
nsJSUtils::ConvertJSValToString(options, cx, argv[2]);
}
}
}
return OpenInternal(url, name, options, PR_FALSE, nsnull, 0, nsnull,
_retval);
}
// like Open, but attaches to the new window any extra parameters past
// [features] as a JS property named "arguments"
NS_IMETHODIMP
GlobalWindowImpl::OpenDialog(const nsAReadableString& aUrl,
const nsAReadableString& aName,
const nsAReadableString& aOptions,
nsISupports* aExtraArgument,
nsIDOMWindow** _retval)
{
return OpenInternal(aUrl, aName, aOptions, PR_TRUE, nsnull, 0,
aExtraArgument, _retval);
}
NS_IMETHODIMP
GlobalWindowImpl::OpenDialog(nsIDOMWindow** _retval)
{
NS_ENSURE_STATE(sXPConnect);
nsresult rv = NS_OK;
nsCOMPtr<nsIXPCNativeCallContext> ncc;
rv = sXPConnect->GetCurrentNativeCallContext(getter_AddRefs(ncc));
NS_ENSURE_SUCCESS(rv, rv);
if (!ncc)
return NS_ERROR_NOT_AVAILABLE;
JSContext *cx = nsnull;
rv = ncc->GetJSContext(&cx);
NS_ENSURE_SUCCESS(rv, rv);
nsAutoString url, name, options;
PRUint32 argc;
jsval *argv = nsnull;
ncc->GetArgc(&argc);
ncc->GetArgvPtr(&argv);
if (argc > 0) {
nsJSUtils::ConvertJSValToString(url, cx, argv[0]);
if (argc > 1) {
nsJSUtils::ConvertJSValToString(name, cx, argv[1]);
if (argc > 2) {
nsJSUtils::ConvertJSValToString(options, cx, argv[2]);
}
}
}
return OpenInternal(url, name, options, PR_TRUE, argv, argc, nsnull,
_retval);
}
NS_IMETHODIMP
GlobalWindowImpl::GetFrames(nsIDOMWindow** aFrames)
{
*aFrames = this;
NS_ADDREF(*aFrames);
FlushPendingNotifications();
return NS_OK;
}
NS_IMETHODIMP
GlobalWindowImpl::Close()
{
nsCOMPtr<nsIDOMWindow> parent;
GetParent(getter_AddRefs(parent));
if (parent != NS_STATIC_CAST(nsIDOMWindow *, this)) {
// window.close() is called on a frame in a frameset, such calls
// are ignored.
return NS_OK;
}
// Note: the basic security check, rejecting windows not opened through JS,
// has been removed. This was approved long ago by ...you're going to call me
// on this, aren't you... well it was. And anyway, a better means is coming.
// In the new world of application-level interfaces being written in JS, this
// security check was causing problems.
nsCOMPtr<nsIJSContextStack> stack =
do_GetService("@mozilla.org/js/xpc/ContextStack;1");
JSContext *cx = nsnull;
if (stack) {
stack->Peek(&cx);
}
if (cx) {
nsCOMPtr<nsIScriptContext> currentCX =
NS_STATIC_CAST(nsIScriptContext *, ::JS_GetContextPrivate(cx));
if (currentCX && currentCX == mContext) {
return currentCX->SetTerminationFunction(CloseWindow,
NS_STATIC_CAST(nsIDOMWindow *,
this));
}
}
// If we get past the above we're closing the window right now.
return ReallyCloseWindow();
}
NS_IMETHODIMP
GlobalWindowImpl::ReallyCloseWindow()
{
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
// If there's no treeOwnerAsWin, this window must already be closed.
if (treeOwnerAsWin) {
treeOwnerAsWin->Destroy();
CleanUp();
}
return NS_OK;
}
NS_IMETHODIMP
GlobalWindowImpl::UpdateCommands(const nsAReadableString& anAction)
{
nsCOMPtr<nsIDOMWindowInternal> rootWindow;
GetPrivateRoot(getter_AddRefs(rootWindow));
if (!rootWindow)
return NS_OK;
2000-11-04 08:21:20 +00:00
nsCOMPtr<nsIDOMDocument> document;
rootWindow->GetDocument(getter_AddRefs(document));
if (document) {
// See if we contain a XUL document.
2000-11-04 08:21:20 +00:00
nsCOMPtr<nsIDOMXULDocument> xulDoc = do_QueryInterface(document);
if (xulDoc) {
// Retrieve the command dispatcher and call updateCommands on it.
nsCOMPtr<nsIDOMXULCommandDispatcher> xulCommandDispatcher;
xulDoc->GetCommandDispatcher(getter_AddRefs(xulCommandDispatcher));
xulCommandDispatcher->UpdateCommands(anAction);
}
}
return NS_OK;
}
NS_IMETHODIMP
GlobalWindowImpl::Escape(const nsAReadableString& aStr,
nsAWritableString& aReturn)
{
nsresult result = NS_OK;
nsCOMPtr<nsIUnicodeEncoder> encoder;
nsAutoString charset;
nsCOMPtr<nsICharsetConverterManager>
ccm(do_GetService(kCharsetConverterManagerCID));
NS_ENSURE_TRUE(ccm, NS_ERROR_FAILURE);
// Get the document character set
charset.Assign(NS_LITERAL_STRING("UTF-8")); // default to utf-8
if (mDocument) {
nsCOMPtr<nsIDocument> doc(do_QueryInterface(mDocument));
if (doc)
result = doc->GetDocumentCharacterSet(charset);
}
if (NS_FAILED(result))
return result;
// Get an encoder for the character set
result = ccm->GetUnicodeEncoder(&charset, getter_AddRefs(encoder));
if (NS_FAILED(result))
return result;
result = encoder->Reset();
if (NS_FAILED(result))
return result;
PRInt32 maxByteLen, srcLen;
srcLen = aStr.Length();
2001-04-02 19:40:52 +00:00
const nsPromiseFlatString& flatSrc = PromiseFlatString(aStr);
const PRUnichar* src = flatSrc.get();
// Get the expected length of result string
result = encoder->GetMaxLength(src, srcLen, &maxByteLen);
if (NS_FAILED(result))
return result;
// Allocate a buffer of the maximum length
char* dest = (char *) nsMemory::Alloc(maxByteLen + 1);
PRInt32 destLen2, destLen = maxByteLen;
if (!dest)
return NS_ERROR_OUT_OF_MEMORY;
// Convert from unicode to the character set
result = encoder->Convert(src, &srcLen, dest, &destLen);
if (NS_FAILED(result)) {
nsMemory::Free(dest);
return result;
}
// Allow the encoder to finish the conversion
destLen2 = maxByteLen - destLen;
encoder->Finish(dest + destLen, &destLen2);
dest[destLen + destLen2] = '\0';
// Escape the string
char *outBuf =
nsEscape(dest, nsEscapeMask(url_XAlphas | url_XPAlphas | url_Path));
CopyASCIItoUCS2(nsDependentCString(outBuf), aReturn);
nsMemory::Free(outBuf);
nsMemory::Free(dest);
return result;
}
NS_IMETHODIMP GlobalWindowImpl::Unescape(const nsAReadableString& aStr,
nsAWritableString& aReturn)
{
nsresult result = NS_OK;
nsCOMPtr<nsIUnicodeDecoder> decoder;
nsAutoString charset;
aReturn.Truncate();
nsCOMPtr<nsICharsetConverterManager>
ccm(do_GetService(kCharsetConverterManagerCID));
NS_ENSURE_TRUE(ccm, NS_ERROR_FAILURE);
// Get the document character set
charset.Assign(NS_LITERAL_STRING("UTF-8")); // default to utf-8
if (mDocument) {
nsCOMPtr<nsIDocument> doc(do_QueryInterface(mDocument));
1998-07-16 01:16:47 +00:00
if (doc)
result = doc->GetDocumentCharacterSet(charset);
}
if (NS_FAILED(result))
return result;
// Get an decoder for the character set
result = ccm->GetUnicodeDecoder(&charset, getter_AddRefs(decoder));
if (NS_FAILED(result))
return result;
result = decoder->Reset();
if (NS_FAILED(result))
return result;
// Need to copy to do the two-byte to one-byte deflation
char *inBuf = ToNewCString(aStr);
if (!inBuf)
return NS_ERROR_OUT_OF_MEMORY;
// Unescape the string
char *src = nsUnescape(inBuf);
PRInt32 maxLength, srcLen;
srcLen = nsCRT::strlen(src);
// Get the expected length of the result string
result = decoder->GetMaxLength(src, srcLen, &maxLength);
if (NS_FAILED(result) || !maxLength) {
nsMemory::Free(src);
return result;
}
// Allocate a buffer of the maximum length
PRUnichar *dest = (PRUnichar*)nsMemory::Alloc(sizeof(PRUnichar) * maxLength);
PRInt32 destLen = maxLength;
if (!dest) {
nsMemory::Free(src);
return NS_ERROR_OUT_OF_MEMORY;
}
// Convert from character set to unicode
result = decoder->Convert(src, &srcLen, dest, &destLen);
nsMemory::Free(src);
if (NS_FAILED(result)) {
nsMemory::Free(dest);
return result;
}
aReturn.Assign(dest, destLen);
nsMemory::Free(dest);
return NS_OK;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::GetSelection(nsISelection** aSelection)
{
NS_ENSURE_ARG_POINTER(aSelection);
*aSelection = nsnull;
if (!mDocShell)
return NS_OK;
nsCOMPtr<nsIPresShell> presShell;
mDocShell->GetPresShell(getter_AddRefs(presShell));
if (!presShell)
return NS_OK;
nsCOMPtr<nsIFrameSelection> selection;
presShell->GetFrameSelection(getter_AddRefs(selection));
if (!selection)
return NS_OK;
return selection->GetSelection(nsISelectionController::SELECTION_NORMAL,
aSelection);
}
// Non-scriptable version of window.find(), part of nsIDOMWindowInternal
NS_IMETHODIMP
GlobalWindowImpl::Find(const nsAReadableString& aStr,
PRBool aCaseSensitive,
PRBool aBackwards,
PRBool aWrapAround,
PRBool aWholeWord,
PRBool aSearchInFrames,
PRBool aShowDialog,
PRBool *aDidFind)
{
return FindInternal(aStr, aCaseSensitive, aBackwards, aWrapAround,
aWholeWord, aSearchInFrames, aShowDialog, aDidFind);
}
// Scriptable version of window.find() which takes a variable number of
// arguments, part of nsIDOMJSWindow.
NS_IMETHODIMP
GlobalWindowImpl::Find(PRBool *aDidFind)
{
NS_ENSURE_STATE(sXPConnect);
nsresult rv = NS_OK;
// We get the arguments passed to the function using the XPConnect native
// call context.
nsCOMPtr<nsIXPCNativeCallContext> ncc;
rv = sXPConnect->GetCurrentNativeCallContext(getter_AddRefs(ncc));
NS_ENSURE_SUCCESS(rv, rv);
NS_ASSERTION(ncc, "No Native Call Context."
"Please don't call this method from C++.");
if (!ncc) {
return NS_ERROR_NOT_AVAILABLE;
}
JSContext *cx = nsnull;
rv = ncc->GetJSContext(&cx);
NS_ENSURE_SUCCESS(rv, rv);
PRUint32 argc;
jsval *argv = nsnull;
ncc->GetArgc(&argc);
ncc->GetArgvPtr(&argv);
// Parse the arguments passed to the function
nsAutoString searchStr;
PRBool caseSensitive = PR_FALSE;
PRBool backwards = PR_FALSE;
PRBool wrapAround = PR_FALSE;
PRBool showDialog = PR_FALSE;
PRBool wholeWord = PR_FALSE;
PRBool searchInFrames = PR_FALSE;
if (argc > 0) {
// First arg is the search pattern
nsJSUtils::ConvertJSValToString(searchStr, cx, argv[0]);
}
if (argc > 1 && !JS_ValueToBoolean(cx, argv[1], &caseSensitive)) {
// Second arg is the case sensitivity
caseSensitive = PR_FALSE;
}
if (argc > 2 && !JS_ValueToBoolean(cx, argv[2], &backwards)) {
// Third arg specifies whether to search backwards
backwards = PR_FALSE;
}
if (argc > 3 && !JS_ValueToBoolean(cx, argv[3], &wrapAround)) {
// Fourth arg specifies whether we should wrap the search
wrapAround = PR_FALSE;
}
if (argc > 4 && !JS_ValueToBoolean(cx, argv[4], &wholeWord)) {
// Fifth arg specifies whether we should show the Find dialog
wholeWord = PR_FALSE;
}
if (argc > 5 && !JS_ValueToBoolean(cx, argv[5], &searchInFrames)) {
// Sixth arg specifies whether we should search only for whole words
searchInFrames = PR_FALSE;
}
if (argc > 6 && !JS_ValueToBoolean(cx, argv[6], &showDialog)) {
// Seventh arg specifies whether we should search in all frames
showDialog = PR_FALSE;
}
return FindInternal(searchStr, caseSensitive, backwards, wrapAround,
wholeWord, searchInFrames, showDialog, aDidFind);
}
nsresult
GlobalWindowImpl::FindInternal(nsAReadableString& aStr,
PRBool caseSensitive,
PRBool backwards,
PRBool wrapAround,
PRBool wholeWord,
PRBool searchInFrames,
PRBool showDialog,
PRBool *aDidFind)
{
NS_ENSURE_ARG_POINTER(aDidFind);
nsresult rv = NS_OK;
*aDidFind = PR_FALSE;
// GetInterface(NS_GET_IID(nsIWebBrowserFind))
nsCOMPtr<nsIWebBrowserFind> finder(do_GetInterface(mDocShell));
// Set the options of the search
rv = finder->SetSearchString(PromiseFlatString(aStr).get());
NS_ENSURE_SUCCESS(rv, rv);
finder->SetMatchCase(caseSensitive);
finder->SetFindBackwards(backwards);
finder->SetWrapFind(wrapAround);
finder->SetEntireWord(wholeWord);
finder->SetSearchFrames(searchInFrames);
// The Find API does not accept empty strings. Launch the Find Dialog.
if (aStr.IsEmpty() || showDialog) {
// See if the find dialog is already up using nsIWindowMediator
nsCOMPtr<nsIWindowMediator> windowMediator =
do_GetService(kWindowMediatorCID);
nsCOMPtr<nsIDOMWindowInternal> findDialog;
if (windowMediator) {
windowMediator->GetMostRecentWindow(NS_LITERAL_STRING("findInPage").get(),
getter_AddRefs(findDialog));
}
if (findDialog) {
// The Find dialog is already open, bring it to the top.
rv = findDialog->Focus();
} else { // Open a Find dialog
if (finder) {
nsCOMPtr<nsIDOMWindow> dialog;
rv = OpenDialog(NS_LITERAL_STRING("chrome://global/content/finddialog.xul"),
NS_LITERAL_STRING("_blank"),
NS_LITERAL_STRING("chrome, resizable=no, dependent=yes"),
finder, getter_AddRefs(dialog));
}
}
} else {
// Launch the search with the passed in search string
rv = finder->FindNext(aDidFind);
NS_ENSURE_SUCCESS(rv, rv);
}
return rv;
}
//*****************************************************************************
// GlobalWindowImpl::nsIDOMEventTarget
//*****************************************************************************
1998-07-16 01:16:47 +00:00
NS_IMETHODIMP
GlobalWindowImpl::AddEventListener(const nsAReadableString& aType,
nsIDOMEventListener* aListener,
PRBool aUseCapture)
{
nsCOMPtr<nsIEventListenerManager> manager;
1998-07-16 01:16:47 +00:00
if (NS_SUCCEEDED(GetListenerManager(getter_AddRefs(manager)))) {
PRInt32 flags = aUseCapture ? NS_EVENT_FLAG_CAPTURE : NS_EVENT_FLAG_BUBBLE;
manager->AddEventListenerByType(aListener, aType, flags);
return NS_OK;
}
return NS_ERROR_FAILURE;
}
NS_IMETHODIMP
GlobalWindowImpl::RemoveEventListener(const nsAReadableString& aType,
nsIDOMEventListener* aListener,
PRBool aUseCapture)
{
if (mListenerManager) {
PRInt32 flags = aUseCapture ? NS_EVENT_FLAG_CAPTURE : NS_EVENT_FLAG_BUBBLE;
1998-07-16 01:16:47 +00:00
mListenerManager->RemoveEventListenerByType(aListener, aType, flags);
return NS_OK;
}
return NS_ERROR_FAILURE;
}
1998-07-16 01:16:47 +00:00
NS_IMETHODIMP GlobalWindowImpl::DispatchEvent(nsIDOMEvent* aEvent, PRBool* _retval)
{
if (mDocument) {
nsCOMPtr<nsIDocument> idoc(do_QueryInterface(mDocument));
if (idoc) {
// Obtain a presentation context
PRInt32 count = idoc->GetNumberOfShells();
if (count == 0)
return NS_OK;
nsCOMPtr<nsIPresShell> shell;
idoc->GetShellAt(0, getter_AddRefs(shell));
// Retrieve the context
nsCOMPtr<nsIPresContext> aPresContext;
shell->GetPresContext(getter_AddRefs(aPresContext));
nsCOMPtr<nsIEventStateManager> esm;
if (NS_SUCCEEDED(aPresContext->GetEventStateManager(getter_AddRefs(esm)))) {
return esm->DispatchNewEvent(NS_STATIC_CAST(nsIScriptGlobalObject *,
this), aEvent, _retval);
}
}
}
return NS_ERROR_FAILURE;
}
//*****************************************************************************
// GlobalWindowImpl::nsIDOMEventReceiver
//*****************************************************************************
1998-07-16 01:16:47 +00:00
NS_IMETHODIMP
GlobalWindowImpl::AddEventListenerByIID(nsIDOMEventListener* aListener,
const nsIID& aIID)
{
nsCOMPtr<nsIEventListenerManager> manager;
if (NS_OK == GetListenerManager(getter_AddRefs(manager))) {
manager->AddEventListenerByIID(aListener, aIID, NS_EVENT_FLAG_BUBBLE);
return NS_OK;
}
return NS_ERROR_FAILURE;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP
GlobalWindowImpl::RemoveEventListenerByIID(nsIDOMEventListener* aListener,
const nsIID& aIID)
{
if (mListenerManager) {
mListenerManager->RemoveEventListenerByIID(aListener, aIID,
NS_EVENT_FLAG_BUBBLE);
return NS_OK;
}
return NS_ERROR_FAILURE;
}
1998-07-16 01:16:47 +00:00
NS_IMETHODIMP
GlobalWindowImpl::GetListenerManager(nsIEventListenerManager **aResult)
{
if (!mListenerManager) {
static NS_DEFINE_CID(kEventListenerManagerCID,
NS_EVENTLISTENERMANAGER_CID);
nsresult rv;
mListenerManager = do_CreateInstance(kEventListenerManagerCID, &rv);
NS_ENSURE_SUCCESS(rv, rv);
}
return CallQueryInterface(mListenerManager, aResult);
}
//XXX I need another way around the circular link problem.
NS_IMETHODIMP
GlobalWindowImpl::GetNewListenerManager(nsIEventListenerManager **aResult)
{
return NS_ERROR_FAILURE;
}
1998-07-16 01:16:47 +00:00
NS_IMETHODIMP GlobalWindowImpl::HandleEvent(nsIDOMEvent *aEvent)
{
PRBool noDefault;
return DispatchEvent(aEvent, &noDefault);
}
//*****************************************************************************
// GlobalWindowImpl::nsPIDOMWindow
//*****************************************************************************
1998-07-16 01:16:47 +00:00
NS_IMETHODIMP GlobalWindowImpl::GetPrivateParent(nsPIDOMWindow ** aParent)
{
nsCOMPtr<nsIDOMWindow> parent;
*aParent = nsnull; // Set to null so we can bail out later
GetParent(getter_AddRefs(parent));
(13163, r=alecf, scc, waterson, others; names available on request) - Fix most of bug 13163 (see TODO for rest). This entails adding a version-string argument to nsIScriptContext::EvaluateString and passing it around lots of places in content sinks. - Fix leaks and confusion about mSecurityManager and mNameSpaceManager in nsJSEnvironment.cpp. These still need to move from nsJSContext to nsGlobalWindow or thereabouts, jband and vidur are looking at that. - Added comments and expanded tabs in nsJSEnvironment.cpp, esp. to EvaluateString. Also changed various nsresult vars to be named rv. Also restored brace/style conformity to nsJSProtocolHandler.cpp. - Factored CompileFunction from AddScriptEventListener to pave the way for brutal sharing of compiled JS event handlers via JS_CloneFunctionObject. - Lots of nsCOMPtr uses added. I'm using one for mNameSpaceManager. Hold mSecurityManager as a service explicitly, on the other hand (awaiting scc's fix to allow comptrs for services), and release in nsJSContext's dtor (fixing a leak). These two managers should be moved to the window object -- TODO item below. - Hold JSRuntimeService along with JSRuntime for live of nsJSEnvironment, fix for shaver. - Fix window.setTimeout etc. so the filename and line number of the timeout expr is propagated. This meant factoring nsJSUtils.cpp code. - Fix all content sinks to use the same, and up-to-date JavaScript version parsing (whether for script type or for old language attribute); also fix SplitMimeType clones to strip whitespace. - With waterson, fix bug in brutal-sharing version of XUL content sink: script src= should not evaluate the inline content of its tag.
1999-10-31 00:43:30 +00:00
if (NS_STATIC_CAST(nsIDOMWindow *, this) == parent.get()) {
nsCOMPtr<nsIContent> chromeElement(do_QueryInterface(mChromeEventHandler));
if (!chromeElement)
return NS_OK; // This is ok, just means a null parent.
nsCOMPtr<nsIDocument> doc;
chromeElement->GetDocument(*getter_AddRefs(doc));
if (!doc)
return NS_OK; // This is ok, just means a null parent.
1998-07-16 01:16:47 +00:00
nsCOMPtr<nsIScriptGlobalObject> globalObject;
doc->GetScriptGlobalObject(getter_AddRefs(globalObject));
if (!globalObject)
return NS_OK; // This is ok, just means a null parent.
parent = do_QueryInterface(globalObject);
}
if (parent)
CallQueryInterface(parent.get(), aParent);
return NS_OK;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::GetPrivateRoot(nsIDOMWindowInternal ** aParent)
{
*aParent = nsnull; // Set to null so we can bail out later
nsCOMPtr<nsIDOMWindow> parent;
GetTop(getter_AddRefs(parent));
nsCOMPtr<nsIScriptGlobalObject> parentTop = do_QueryInterface(parent);
nsCOMPtr<nsIDocShell> docShell;
NS_ASSERTION(parentTop, "cannot get parentTop");
if(parentTop == nsnull)
return NS_ERROR_FAILURE;
parentTop->GetDocShell(getter_AddRefs(docShell));
2000-11-04 08:21:20 +00:00
// Get the chrome event handler from the doc shell, since we only
// want to deal with XUL chrome handlers and not the new kind of
// window root handler.
nsCOMPtr<nsIChromeEventHandler> chromeEventHandler;
docShell->GetChromeEventHandler(getter_AddRefs(chromeEventHandler));
nsCOMPtr<nsIContent> chromeElement(do_QueryInterface(mChromeEventHandler));
if (chromeElement) {
nsCOMPtr<nsIDocument> doc;
chromeElement->GetDocument(*getter_AddRefs(doc));
if (doc) {
nsCOMPtr<nsIScriptGlobalObject> globalObject;
doc->GetScriptGlobalObject(getter_AddRefs(globalObject));
parent = do_QueryInterface(globalObject);
nsCOMPtr<nsIDOMWindow> tempParent;
parent->GetTop(getter_AddRefs(tempParent));
CallQueryInterface(tempParent, aParent);
return NS_OK;
}
}
if (parent) {
CallQueryInterface(parent, aParent);
}
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::GetLocation(nsIDOMLocation ** aLocation)
1998-07-16 01:16:47 +00:00
{
if (!mLocation && mDocShell) {
mLocation = new LocationImpl(mDocShell);
NS_IF_ADDREF(mLocation);
}
*aLocation = mLocation;
NS_IF_ADDREF(mLocation);
return NS_OK;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP GlobalWindowImpl::GetObjectProperty(const PRUnichar *aProperty,
nsISupports ** aObject)
{
NS_ENSURE_TRUE(mJSObject, NS_ERROR_NOT_AVAILABLE);
// Get JSContext from stack.
nsCOMPtr<nsIThreadJSContextStack> stack(do_GetService(sJSStackContractID));
NS_ENSURE_TRUE(stack, NS_ERROR_FAILURE);
JSContext *cx;
NS_ENSURE_SUCCESS(stack->Peek(&cx), NS_ERROR_FAILURE);
if (!cx) {
stack->GetSafeJSContext(&cx);
NS_ENSURE_TRUE(cx, NS_ERROR_FAILURE);
}
jsval propertyVal;
if (!::JS_LookupUCProperty(cx, mJSObject,
NS_REINTERPRET_CAST(const jschar *, aProperty),
nsCRT::strlen(aProperty), &propertyVal)) {
return NS_ERROR_FAILURE;
}
if (!nsJSUtils::ConvertJSValToXPCObject(aObject, NS_GET_IID(nsISupports),
cx, propertyVal)) {
return NS_ERROR_FAILURE;
}
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::Activate()
{
/*
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
if(treeOwnerAsWin)
treeOwnerAsWin->SetVisibility(PR_TRUE);
nsCOMPtr<nsIPresShell> presShell;
mDocShell->GetPresShell(getter_AddRefs(presShell));
NS_ENSURE_TRUE(presShell, NS_ERROR_FAILURE);
nsCOMPtr<nsIViewManager> vm;
presShell->GetViewManager(getter_AddRefs(vm));
NS_ENSURE_TRUE(vm, NS_ERROR_FAILURE);
nsIView* rootView;
vm->GetRootView(rootView);
NS_ENSURE_TRUE(rootView, NS_ERROR_FAILURE);
nsCOMPtr<nsIWidget> widget;
rootView->GetWidget(*getter_AddRefs(widget));
NS_ENSURE_TRUE(widget, NS_ERROR_FAILURE);
return widget->SetFocus();
*/
nsCOMPtr<nsIBaseWindow> treeOwnerAsWin;
GetTreeOwner(getter_AddRefs(treeOwnerAsWin));
if (treeOwnerAsWin)
treeOwnerAsWin->SetVisibility(PR_TRUE);
nsCOMPtr<nsIPresShell> presShell;
mDocShell->GetPresShell(getter_AddRefs(presShell));
NS_ENSURE_TRUE(presShell, NS_ERROR_FAILURE);
nsCOMPtr<nsIViewManager> vm;
presShell->GetViewManager(getter_AddRefs(vm));
NS_ENSURE_TRUE(vm, NS_ERROR_FAILURE);
nsIView *rootView;
vm->GetRootView(rootView);
NS_ENSURE_TRUE(rootView, NS_ERROR_FAILURE);
nsCOMPtr<nsIWidget> widget;
rootView->GetWidget(*getter_AddRefs(widget));
NS_ENSURE_TRUE(widget, NS_ERROR_FAILURE);
nsEventStatus status;
nsGUIEvent guiEvent;
guiEvent.eventStructType = NS_GUI_EVENT;
guiEvent.point.x = 0;
guiEvent.point.y = 0;
guiEvent.time = PR_IntervalNow();
guiEvent.nativeMsg = nsnull;
2000-09-19 03:14:31 +00:00
guiEvent.message = NS_ACTIVATE;
guiEvent.widget = widget;
vm->DispatchEvent(&guiEvent, &status);
return NS_OK;
}
NS_IMETHODIMP GlobalWindowImpl::Deactivate()
{
nsCOMPtr<nsIPresShell> presShell;
mDocShell->GetPresShell(getter_AddRefs(presShell));
NS_ENSURE_TRUE(presShell, NS_ERROR_FAILURE);
nsCOMPtr<nsIViewManager> vm;
presShell->GetViewManager(getter_AddRefs(vm));
NS_ENSURE_TRUE(vm, NS_ERROR_FAILURE);
nsIView *rootView;
vm->GetRootView(rootView);
NS_ENSURE_TRUE(rootView, NS_ERROR_FAILURE);
nsCOMPtr<nsIWidget> widget;
rootView->GetWidget(*getter_AddRefs(widget));
NS_ENSURE_TRUE(widget, NS_ERROR_FAILURE);
nsEventStatus status;
nsGUIEvent guiEvent;
guiEvent.eventStructType = NS_GUI_EVENT;
guiEvent.point.x = 0;
guiEvent.point.y = 0;
guiEvent.time = PR_IntervalNow();
guiEvent.nativeMsg = nsnull;
guiEvent.message = NS_DEACTIVATE;
guiEvent.widget = widget;
vm->DispatchEvent(&guiEvent, &status);
return NS_OK;
}
NS_IMETHODIMP
2000-11-04 08:21:20 +00:00
GlobalWindowImpl::GetChromeEventHandler(nsIChromeEventHandler** aHandler)
{
2000-11-04 08:21:20 +00:00
*aHandler = mChromeEventHandler;
NS_IF_ADDREF(*aHandler);
return NS_OK;
}
2000-11-04 08:21:20 +00:00
NS_IMETHODIMP
GlobalWindowImpl::GetRootFocusController(nsIFocusController** aController)
{
*aController = nsnull;
nsCOMPtr<nsIDOMWindowInternal> rootWindow;
2000-09-22 05:02:20 +00:00
GetPrivateRoot(getter_AddRefs(rootWindow));
if (rootWindow) {
2000-11-04 08:21:20 +00:00
// Obtain the chrome event handler.
nsCOMPtr<nsIChromeEventHandler> chromeHandler;
nsCOMPtr<nsPIDOMWindow> piWin(do_QueryInterface(rootWindow));
piWin->GetChromeEventHandler(getter_AddRefs(chromeHandler));
if (chromeHandler) {
nsCOMPtr<nsPIWindowRoot> windowRoot(do_QueryInterface(chromeHandler));
if (windowRoot) {
windowRoot->GetFocusController(aController);
2000-11-04 08:21:20 +00:00
}
}
}
2000-11-27 07:55:20 +00:00
return NS_OK;
}
NS_IMETHODIMP
GlobalWindowImpl::HasMutationListeners(PRUint32 aMutationEventType,
PRBool* aResult)
2000-11-27 07:55:20 +00:00
{
*aResult = (mMutationBits & aMutationEventType) != 0;
return NS_OK;
}
2000-11-04 08:21:20 +00:00
2000-11-27 07:55:20 +00:00
NS_IMETHODIMP
GlobalWindowImpl::SetMutationListeners(PRUint32 aType)
{
mMutationBits |= aType;
return NS_OK;
}
//*****************************************************************************
// GlobalWindowImpl::nsIDOMViewCSS
//*****************************************************************************
NS_IMETHODIMP
GlobalWindowImpl::GetComputedStyle(nsIDOMElement* aElt,
const nsAReadableString& aPseudoElt,
nsIDOMCSSStyleDeclaration** aReturn)
{
NS_ENSURE_ARG_POINTER(aReturn);
2000-05-17 06:38:43 +00:00
NS_ENSURE_ARG_POINTER(aElt);
*aReturn = nsnull;
2000-05-17 06:38:43 +00:00
NS_ENSURE_TRUE(mDocShell, NS_OK);
2000-05-17 06:38:43 +00:00
nsCOMPtr<nsIPresShell> presShell;
mDocShell->GetPresShell(getter_AddRefs(presShell));
NS_ENSURE_TRUE(presShell, NS_OK);
nsresult rv = NS_OK;
nsCOMPtr<nsIComputedDOMStyle> compStyle;
compStyle =
do_CreateInstance("@mozilla.org/DOM/Level2/CSS/computedStyleDeclaration;1", &rv);
2000-05-17 06:38:43 +00:00
NS_ENSURE_SUCCESS(rv, rv);
rv = compStyle->Init(aElt, aPseudoElt, presShell);
NS_ENSURE_SUCCESS(rv, rv);
return compStyle->QueryInterface(NS_GET_IID(nsIDOMCSSStyleDeclaration),
(void **) aReturn);
}
//*****************************************************************************
// GlobalWindowImpl::nsIDOMAbstractView
//*****************************************************************************
NS_IMETHODIMP
GlobalWindowImpl::GetDocument(nsIDOMDocumentView ** aDocumentView)
{
NS_ENSURE_ARG_POINTER(aDocumentView);
nsresult rv = NS_OK;
if (mDocument) {
rv = mDocument->QueryInterface(NS_GET_IID(nsIDOMDocumentView),
(void **) aDocumentView);
}
else {
*aDocumentView = nsnull;
}
return rv;
}
///*****************************************************************************
// GlobalWindowImpl::nsIInterfaceRequestor
//*****************************************************************************
NS_IMETHODIMP GlobalWindowImpl::GetInterface(const nsIID & aIID, void **aSink)
{
NS_ENSURE_ARG_POINTER(aSink);
*aSink = nsnull;
if (aIID.Equals(NS_GET_IID(nsIDocCharset))) {
if (mDocShell) {
nsCOMPtr<nsIDocCharset> docCharset(do_QueryInterface(mDocShell));
*aSink = docCharset;
}
}
else if (aIID.Equals(NS_GET_IID(nsIWebNavigation))) {
if (mDocShell) {
nsCOMPtr<nsIWebNavigation> webNav(do_QueryInterface(mDocShell));
*aSink = webNav;
}
}
else {
return QueryInterface(aIID, aSink);
}
NS_IF_ADDREF(((nsISupports *) * aSink));
return NS_OK;
}
//*****************************************************************************
// GlobalWindowImpl: Window Control Functions
//*****************************************************************************
void
GlobalWindowImpl::GetParentInternal(nsIDOMWindowInternal **aParent)
{
*aParent = nsnull;
nsCOMPtr<nsIDOMWindow> parent;
GetParent(getter_AddRefs(parent));
if (parent && parent != NS_STATIC_CAST(nsIDOMWindow *, this)) {
CallQueryInterface(parent, aParent);
NS_ASSERTION(*aParent, "Huh, parent not an nsIDOMWindowInternal?");
}
}
NS_IMETHODIMP
GlobalWindowImpl::OpenInternal(const nsAReadableString& aUrl,
const nsAReadableString& aName,
const nsAReadableString& aOptions,
PRBool aDialog, jsval *argv, PRUint32 argc,
nsISupports *aExtraArgument,
nsIDOMWindow **aReturn)
{
nsCOMPtr<nsIDOMWindow> domReturn;
char *features = nsnull;
char *name = nsnull;
char *url = nsnull;
nsresult rv = NS_OK;
*aReturn = nsnull;
if (!aUrl.IsEmpty()) {
// fix bug 35076
// if the URL contains non ASCII, escape from the first non ASCII char
nsAutoString unescapedURL(aUrl);
nsAutoString escapedURL;
if (unescapedURL.IsASCII())
escapedURL = unescapedURL;
else {
// const PRUnichar *pt = unescapedURL.get();
PRUint32 len = unescapedURL.Length();
PRUint32 nonAsciiPos = 0;
nsReadingIterator<PRUnichar> iter, end;
unescapedURL.BeginReading(iter);
unescapedURL.EndReading(end);
while (iter != end) {
if ((0xFF80 & *iter) != 0)
break;
++nonAsciiPos;
++iter;
}
nsAutoString right, escapedRight;
unescapedURL.Left(escapedURL, nonAsciiPos);
unescapedURL.Right(right, len - nonAsciiPos);
if (NS_SUCCEEDED(Escape(right, escapedRight)))
escapedURL.Append(escapedRight);
else
escapedURL = unescapedURL;
}
if (!escapedURL.IsEmpty())
url = ToNewCString(escapedURL);
/* Check whether the URI is allowed, but not for dialogs --
see bug 56851. The security of this function depends on
window.openDialog being inaccessible from web scripts */
if (url && !aDialog)
rv = SecurityCheckURL(url);
}
if (!aName.IsEmpty()) {
name = ToNewUTF8String(aName);
}
if (!aOptions.IsEmpty() /* IsNullDOMString(aOptions) */) {
features = ToNewUTF8String(aOptions);
}
if (NS_SUCCEEDED(rv)) {
nsCOMPtr<nsIWindowWatcher> wwatch(do_GetService(sWindowWatcherContractID, &rv));
if (wwatch) {
if (argc) {
nsCOMPtr<nsPIWindowWatcher> pwwatch(do_QueryInterface(wwatch));
NS_ENSURE_TRUE(pwwatch, NS_ERROR_UNEXPECTED);
PRUint32 extraArgc = argc >= 3 ? argc-3 : 0;
rv = pwwatch->OpenWindowJS(this, url, name, features, aDialog,
extraArgc, argv+3,
getter_AddRefs(domReturn));
} else {
rv = wwatch->OpenWindow(this, url, name, features, aExtraArgument,
getter_AddRefs(domReturn));
}
if (domReturn)
CallQueryInterface(domReturn, aReturn);
}
}
if (features)
nsMemory::Free(features);
if (name)
nsMemory::Free(name);
if (url)
nsMemory::Free(url);
return rv;
}
void GlobalWindowImpl::CloseWindow(nsISupports *aWindow)
{
nsCOMPtr<nsPIDOMWindow> win(do_QueryInterface(aWindow));
win->ReallyCloseWindow();
}
//*****************************************************************************
// GlobalWindowImpl: Timeout Functions
//*****************************************************************************
static const char * const kSetIntervalStr = "setInterval";
static const char * const kSetTimeoutStr = "setTimeout";
nsresult
GlobalWindowImpl::SetTimeoutOrInterval(PRBool aIsInterval, PRInt32 *aReturn)
{
NS_ENSURE_STATE(sXPConnect);
nsresult rv = NS_OK;
nsCOMPtr<nsIXPCNativeCallContext> ncc;
rv = sXPConnect->GetCurrentNativeCallContext(getter_AddRefs(ncc));
NS_ENSURE_SUCCESS(rv, rv);
if (!ncc)
return NS_ERROR_NOT_AVAILABLE;
JSContext *cx = nsnull;
rv = ncc->GetJSContext(&cx);
NS_ENSURE_SUCCESS(rv, rv);
PRUint32 argc;
jsval *argv = nsnull;
ncc->GetArgc(&argc);
ncc->GetArgvPtr(&argv);
JSString *expr = nsnull;
JSObject *funobj = nsnull;
nsTimeoutImpl *timeout;
jsdouble interval = 0.0;
PRInt64 now, delta;
if (argc < 1) {
2001-08-14 08:05:28 +00:00
::JS_ReportError(cx, "Function %s requires at least 1 parameter",
aIsInterval ? kSetIntervalStr : kSetTimeoutStr);
return ncc->SetExceptionWasThrown(PR_TRUE);
}
if (argc > 1 && !::JS_ValueToNumber(cx, argv[1], &interval)) {
::JS_ReportError(cx,
"Second argument to %s must be a millisecond interval",
aIsInterval ? kSetIntervalStr : kSetTimeoutStr);
return ncc->SetExceptionWasThrown(PR_TRUE);
}
switch (::JS_TypeOfValue(cx, argv[0])) {
case JSTYPE_FUNCTION:
funobj = JSVAL_TO_OBJECT(argv[0]);
break;
case JSTYPE_STRING:
case JSTYPE_OBJECT:
if (!(expr = ::JS_ValueToString(cx, argv[0])))
return NS_ERROR_FAILURE;
if (!expr)
return NS_ERROR_OUT_OF_MEMORY;
argv[0] = STRING_TO_JSVAL(expr);
break;
default:
::JS_ReportError(cx, "useless %s call (missing quotes around argument?)",
aIsInterval ? kSetIntervalStr : kSetTimeoutStr);
return ncc->SetExceptionWasThrown(PR_TRUE);
}
timeout = new nsTimeoutImpl();
if (!timeout)
return NS_ERROR_OUT_OF_MEMORY;
// Initial ref_count to indicate that this timeout struct will
// be held as the closure of a timer.
timeout->ref_count = 1;
if (aIsInterval)
timeout->interval = (PRInt32) interval;
if (expr) {
if (!::JS_AddNamedRoot(cx, &timeout->expr, "timeout.expr")) {
delete timeout;
return NS_ERROR_OUT_OF_MEMORY;
}
timeout->expr = expr;
}
else if (funobj) {
int i;
/* Leave an extra slot for a secret final argument that
indicates to the called function how "late" the timeout is. */
timeout->argv = (jsval *) PR_MALLOC((argc - 1) * sizeof(jsval));
if (!timeout->argv) {
DropTimeout(timeout);
return NS_ERROR_OUT_OF_MEMORY;
}
if (!::JS_AddNamedRoot(cx, &timeout->funobj, "timeout.funobj")) {
DropTimeout(timeout);
return NS_ERROR_FAILURE;
}
timeout->funobj = funobj;
timeout->argc = 0;
for (i = 2; (PRUint32) i < argc; i++) {
timeout->argv[i - 2] = argv[i];
if (!::JS_AddNamedRoot(cx, &timeout->argv[i - 2], "timeout.argv[i]")) {
DropTimeout(timeout);
return NS_ERROR_FAILURE;
1998-10-06 20:59:39 +00:00
}
timeout->argc++;
}
}
const char *filename;
if (nsJSUtils::GetCallingLocation(cx, &filename, &timeout->lineno)) {
timeout->filename = PL_strdup(filename);
if (!timeout->filename) {
DropTimeout(timeout);
return NS_ERROR_OUT_OF_MEMORY;
}
}
timeout->version = ::JS_VersionToString(::JS_GetVersion(cx));
// Get principal of currently executing code, save for execution of timeout
nsCOMPtr<nsIScriptSecurityManager>
securityManager(do_GetService(NS_SCRIPTSECURITYMANAGER_CONTRACTID, &rv));
if (NS_FAILED(rv)) {
DropTimeout(timeout);
return NS_ERROR_FAILURE;
}
rv =
securityManager->GetSubjectPrincipal(getter_AddRefs(timeout->principal));
if (NS_FAILED(rv)) {
DropTimeout(timeout);
return NS_ERROR_FAILURE;
}
LL_I2L(now, PR_IntervalNow());
LL_D2L(delta, PR_MillisecondsToInterval((PRUint32) interval));
LL_ADD(timeout->when, now, delta);
nsresult err;
timeout->timer = do_CreateInstance("@mozilla.org/timer;1", &err);
if (NS_OK != err) {
DropTimeout(timeout);
return err;
}
err = timeout->timer->Init(TimerCallback, timeout, (PRInt32)interval,
NS_PRIORITY_LOWEST);
if (NS_OK != err) {
DropTimeout(timeout);
return err;
}
timeout->window = this;
NS_ADDREF(timeout->window);
InsertTimeoutIntoList(mTimeoutInsertionPoint, timeout);
timeout->public_id = ++mTimeoutPublicIdCounter;
*aReturn = timeout->public_id;
return NS_OK;
}
void GlobalWindowImpl::RunTimeout(nsTimeoutImpl *aTimeout)
{
PRBool scripts_enabled = PR_TRUE;
if (!mContext) {
return;
}
mContext->GetScriptsEnabled(&scripts_enabled);
if (!scripts_enabled) {
// Scripts were enabled once in this window (unless aTimeout ==
// nsnull) but now scripts are disabled (we might be in
// print-preview, for instance), this means we shouldn't run any
// timeouts at this point.
//
// If scrips are enabled in this window again we'll fire the
// timeouts that are due at that point.
return;
}
nsTimeoutImpl *next, *prev, *timeout;
nsTimeoutImpl *last_expired_timeout, **last_insertion_point;
nsTimeoutImpl dummy_timeout;
JSContext *cx;
PRInt64 now, deadline;
nsresult rv;
PRUint32 firingDepth = mTimeoutFiringDepth + 1;
// Make sure that the window or the script context don't go away as
// a result of running timeouts
nsCOMPtr<nsIScriptGlobalObject> windowKungFuDeathGrip(this);
nsCOMPtr<nsIScriptContext> contextKungFuDeathGrip(mContext);
cx = (JSContext *)mContext->GetNativeContext();
// A native timer has gone off. See which of our timeouts need
// servicing
LL_I2L(now, PR_IntervalNow());
if (aTimeout && LL_CMP(aTimeout->when, >, now)) {
// The OS timer fired early (yikes!), and possibly out of order
// too. Set |deadline| to be the time when the OS timer *should*
// have fired so that any timers that *should* have fired before
// aTimeout *will* be fired now. This happens most of the time on
// Win2k.
deadline = aTimeout->when;
} else {
deadline = now;
}
/* The timeout list is kept in deadline order. Discover the
latest timeout whose deadline has expired. On some platforms,
native timeout events fire "early", so we need to test the
timer as well as the deadline. */
last_expired_timeout = nsnull;
for (timeout = mTimeouts; timeout; timeout = timeout->next) {
if (((timeout == aTimeout) || !LL_CMP(timeout->when, >, deadline)) &&
(timeout->firingDepth == 0)) {
/*
* Mark any timeouts that are on the list to be fired with the
* firing depth so that we can reentrantly run timeouts
*/
timeout->firingDepth = firingDepth;
last_expired_timeout = timeout;
// If the timeout has a timer it means that it's a timeout
// that's due to be fired but it's OS timer hasn't fired yet.
// We'll go ahead and handle the timeout now so we can cancel
// the OS timer for the timeout and relese the OS timer's
// reference to the timeout (and set timeout->timer to nsnull to
// indicate that the OS timer no longer owns the timeout).
if (timeout->timer) {
timeout->timer->Cancel();
timeout->timer = nsnull;
DropTimeout(timeout);
}
}
}
/* Maybe the timeout that the event was fired for has been deleted
and there are no others timeouts with deadlines that make them
eligible for execution yet. Go away. */
if (!last_expired_timeout) {
return;
}
/* Insert a dummy timeout into the list of timeouts between the portion
of the list that we are about to process now and those timeouts that
will be processed in a future call to win_run_timeout(). This dummy
timeout serves as the head of the list for any timeouts inserted as
a result of running a timeout. */
dummy_timeout.firingDepth = firingDepth;
dummy_timeout.next = last_expired_timeout->next;
last_expired_timeout->next = &dummy_timeout;
/* Don't let ClearWindowTimeouts throw away our stack-allocated
dummy timeout. */
dummy_timeout.ref_count = 2;
last_insertion_point = mTimeoutInsertionPoint;
mTimeoutInsertionPoint = &dummy_timeout.next;
1998-08-13 04:34:53 +00:00
prev = nsnull;
for (timeout = mTimeouts; timeout != &dummy_timeout; timeout = next) {
next = timeout->next;
/*
* Check to see if it should fire at this depth. If it shouldn't, we'll
* ignore it
*/
if (timeout->firingDepth == firingDepth) {
nsTimeoutImpl *last_running_timeout = mRunningTimeout;
mRunningTimeout = timeout;
PRBool old_timeouts_were_cleared = mTimeoutsWereCleared;
mTimeoutsWereCleared = PR_FALSE;
/* Hold the timeout in case expr or funobj releases its doc. */
HoldTimeout(timeout);
++mTimeoutFiringDepth;
if (timeout->expr) {
/* Evaluate the timeout expression. */
const PRUnichar *script =
NS_REINTERPRET_CAST(const PRUnichar *,
::JS_GetStringChars(timeout->expr));
nsAutoString retval;
PRBool is_undefined;
rv = mContext->EvaluateString(nsDependentString(script), mJSObject,
timeout->principal, timeout->filename,
timeout->lineno, timeout->version,
retval, &is_undefined);
} else {
PRInt64 lateness64;
PRInt32 lateness;
/* Add "secret" final argument that indicates timeout
lateness in milliseconds */
LL_SUB(lateness64, now, timeout->when);
LL_L2I(lateness, lateness64);
lateness = PR_IntervalToMilliseconds(lateness);
timeout->argv[timeout->argc] = INT_TO_JSVAL((jsint) lateness);
PRBool bool_result;
rv = mContext->CallEventHandler(mJSObject, timeout->funobj,
timeout->argc + 1, timeout->argv,
&bool_result, PR_FALSE);
}
--mTimeoutFiringDepth;
mRunningTimeout = last_running_timeout;
PRBool timeouts_were_cleared = mTimeoutsWereCleared;
mTimeoutsWereCleared = old_timeouts_were_cleared;
// We ignore any failures from calling EvaluateString() or
// CallEventHandler() on the context here since we're in a loop
// where we're likely to be running timeouts whose OS timers
// didn't fire in time and we don't want to not fire those timers
// now just because execution of one timer failed. We can't
// propagate the error to anyone who cares about it from this
// point anyway so we just drop it.
DropTimeout(timeout, contextKungFuDeathGrip);
// If the timeout is no longer in the list of timeouts in this
// window it means that all timeouts were cleared (i.e. the
// document was either destroyed or cleared). In that case we
// simply return here since all timers were destroyed in the
// document where timeout originated from.
if (timeouts_were_cleared) {
mTimeoutInsertionPoint = last_insertion_point;
return;
}
/* If we have a regular interval timer, we re-fire the
* timeout, accounting for clock drift.
*/
if (timeout->interval) {
/* Compute time to next timeout for interval timer. */
PRInt32 delay32;
PRInt64 interval, delay;
LL_I2L(interval, PR_MillisecondsToInterval(timeout->interval));
LL_ADD(timeout->when, timeout->when, interval);
LL_I2L(now, PR_IntervalNow());
LL_SUB(delay, timeout->when, now);
LL_L2I(delay32, delay);
/* If the next interval timeout is already supposed to
* have happened then run the timeout immediately.
*/
if (delay32 < 0)
delay32 = 0;
delay32 = PR_IntervalToMilliseconds(delay32);
/* Reschedule timeout. Account for possible error return in
code below that checks for zero toid. */
timeout->timer = do_CreateInstance("@mozilla.org/timer;1");
// Since we're in a loop where we're likely to be running
// multiple timeouts that were due to be handled when we
// entered this method (but the OS timers for those timeouts
// hadn't fired yet) we don't do an early return here just
// because we couldn't create a timer for an interval timeout
// since we still want to continue running the timeouts that
// were due to be handled when we entered this method. This
// method is never called by code that cares about success or
// failure any way so dropping the error on the floor here is
// no big deal.
if (timeout->timer) {
rv = timeout->timer->Init(TimerCallback, timeout, delay32,
NS_PRIORITY_LOWEST);
// Likewise, don't return early even if we fail to
// initialize the new OS timer.
if (NS_SUCCEEDED(rv)) {
// Increment ref_count to indicate that this OS timer is
// holding on to the timeout struct.
HoldTimeout(timeout);
} else {
NS_ERROR("Error initializing timer for DOM timeout!");
// We failed to initialize the new OS timer, this timer
// does us no good here so we just null out the pointer to
// the OS timer, this will release the OS timer. As we
// continue executing the code below we'll end up deleting
// the timeout since it's not an interval timeout any more
// (since timeout->timer == nsnull).
timeout->timer = nsnull;
}
} else {
NS_ERROR("Error creating timer for DOM timeout!");
// If we weren't able to create a new OS timer for 'timeout'
// we'll fall through here and the code below will delete
// the timeout (since timeout->timer == nsnull).
}
}
/* Running a timeout can cause another timeout to be deleted,
so we need to reset the pointer to the following timeout. */
next = timeout->next;
if (!prev) {
mTimeouts = next;
} else {
prev->next = next;
}
PRBool isInterval = (timeout->interval && timeout->timer);
// Drop timeout struct since it's out of the list
DropTimeout(timeout, contextKungFuDeathGrip);
if (isInterval) {
// Reschedule an interval timeout. Insert interval timeout
// onto list sorted in deadline order.
InsertTimeoutIntoList(mTimeoutInsertionPoint, timeout);
}
} else {
// We skip the timeout since it's on the list to run at another
// depth.
prev = timeout;
}
}
/* Take the dummy timeout off the head of the list */
if (!prev) {
mTimeouts = dummy_timeout.next;
} else {
prev->next = dummy_timeout.next;
}
mTimeoutInsertionPoint = last_insertion_point;
return;
}
void GlobalWindowImpl::DropTimeout(nsTimeoutImpl *aTimeout,
nsIScriptContext *aContext)
{
JSRuntime *rt = nsnull;
if (--aTimeout->ref_count > 0)
return;
if (!aContext)
aContext = mContext;
if (aContext) {
JSContext *cx;
cx = (JSContext *)aContext->GetNativeContext();
rt = ::JS_GetRuntime(cx);
} else {
/* XXX The timeout *must* be unrooted, even if !aContext. This can be
done without a JS context using the JSRuntime. This is safe enough,
but it would be better to drop all a window's timeouts before its
context is cleared. Bug 50705 describes a situation where we're not.
In that case, at the time the context is cleared, a timeout (actually
an Interval) is still active, but temporarily removed from the window's
list of timers (placed instead on the timer manager's list). This makes
the nearly handy ClearAllTimeouts routine useless, so we we settled on
using the JSRuntime rather than relying on the window having a context.
It would be good to remedy this workable but clumsy situation someday.
*/
NS_WARNING("DropTimeout proceeding without context.");
nsCOMPtr<nsIJSRuntimeService> rtsvc =
do_GetService("@mozilla.org/js/xpc/RuntimeService;1");
if (rtsvc)
rtsvc->GetRuntime(&rt);
}
NS_ASSERTION(rt, "DropTimeout with no JSRuntime. eek!");
if (!rt) // most unexpected. not much choice but to bail.
return;
if (aTimeout->expr) {
::JS_RemoveRootRT(rt, &aTimeout->expr);
} else if (aTimeout->funobj) {
::JS_RemoveRootRT(rt, &aTimeout->funobj);
if (aTimeout->argv) {
int i;
for (i = 0; i < aTimeout->argc; i++)
::JS_RemoveRootRT(rt, &aTimeout->argv[i]);
PR_FREEIF(aTimeout->argv);
}
}
if (aTimeout->timer) {
aTimeout->timer->Cancel();
aTimeout->timer = nsnull;
}
PR_FREEIF(aTimeout->filename);
NS_IF_RELEASE(aTimeout->window);
delete aTimeout;
}
void GlobalWindowImpl::HoldTimeout(nsTimeoutImpl *aTimeout)
{
aTimeout->ref_count++;
}
nsresult
GlobalWindowImpl::ClearTimeoutOrInterval(PRInt32 aTimerID)
{
PRUint32 public_id;
nsTimeoutImpl **top, *timeout;
public_id = (PRUint32) aTimerID;
if (!public_id) { /* id of zero is reserved for internal use */
/* return silently for compatibility (see bug 30700) */
return NS_OK;
}
for (top = &mTimeouts; (timeout = *top) != NULL; top = &timeout->next) {
if (timeout->public_id == public_id) {
if (mRunningTimeout == timeout) {
/* We're running from inside the timeout. Mark this
timeout for deferred deletion by the code in
win_run_timeout() */
timeout->interval = 0;
}
else {
/* Delete the timeout from the pending timeout list */
*top = timeout->next;
if (timeout->timer) {
timeout->timer->Cancel();
timeout->timer = nsnull;
DropTimeout(timeout);
}
DropTimeout(timeout);
}
break;
}
}
return NS_OK;
}
void GlobalWindowImpl::ClearAllTimeouts()
{
nsTimeoutImpl *timeout, *next;
for (timeout = mTimeouts; timeout; timeout = next) {
/* If RunTimeout() is higher up on the stack for this
window, e.g. as a result of document.write from a timeout,
then we need to reset the list insertion point for
newly-created timeouts in case the user adds a timeout,
before we pop the stack back to RunTimeout. */
if (mRunningTimeout == timeout)
mTimeoutInsertionPoint = &mTimeouts;
next = timeout->next;
if (timeout->timer) {
timeout->timer->Cancel();
timeout->timer = nsnull;
// Drop the count since the timer isn't going to hold on
// anymore.
DropTimeout(timeout);
}
// Drop the count since we're removing it from the list.
DropTimeout(timeout);
}
mTimeouts = NULL;
mTimeoutsWereCleared = PR_TRUE;
}
void GlobalWindowImpl::InsertTimeoutIntoList(nsTimeoutImpl **aList,
nsTimeoutImpl *aTimeout)
{
nsTimeoutImpl *to;
NS_ASSERTION(aList,
"GlobalWindowImpl::InsertTimeoutIntoList null timeoutList");
while ((to = *aList) != nsnull) {
if (LL_CMP(to->when, >, aTimeout->when))
break;
aList = &to->next;
}
aTimeout->firingDepth = 0;
aTimeout->next = to;
*aList = aTimeout;
// Increment the ref_count since we're in the list
HoldTimeout(aTimeout);
}
// static
void
GlobalWindowImpl::TimerCallback(nsITimer *aTimer, void *aClosure)
{
nsTimeoutImpl *timeout = (nsTimeoutImpl *)aClosure;
// Null out the pointer to the OS timer (aTimer) in the timeout to
// to indicate that the timer (aTimer) no longer owns the
// timeout. From now on 'timeout' has the reference to the timeout.
timeout->timer = nsnull;
timeout->window->RunTimeout(timeout);
// Drop timeout's reference to the timeout.
timeout->window->DropTimeout(timeout);
}
//*****************************************************************************
// GlobalWindowImpl: Helper Functions
//*****************************************************************************
1999-03-28 22:19:42 +00:00
nsresult
GlobalWindowImpl::GetTreeOwner(nsIDocShellTreeOwner **aTreeOwner)
1999-03-28 22:19:42 +00:00
{
nsCOMPtr<nsIDocShellTreeItem> docShellAsItem(do_QueryInterface(mDocShell));
// If there's no docShellAsItem, this window must have been closed,
// in that case there is no tree owner.
if (!docShellAsItem) {
*aTreeOwner = nsnull;
return NS_OK;
}
1999-03-28 22:19:42 +00:00
return docShellAsItem->GetTreeOwner(aTreeOwner);
1999-03-28 22:19:42 +00:00
}
nsresult
GlobalWindowImpl::GetTreeOwner(nsIBaseWindow **aTreeOwner)
1999-03-28 22:19:42 +00:00
{
nsCOMPtr<nsIDocShellTreeItem> docShellAsItem(do_QueryInterface(mDocShell));
nsCOMPtr<nsIDocShellTreeOwner> treeOwner;
// If there's no docShellAsItem, this window must have been closed,
// in that case there is no tree owner.
if (docShellAsItem) {
docShellAsItem->GetTreeOwner(getter_AddRefs(treeOwner));
}
if (!treeOwner) {
*aTreeOwner = nsnull;
return NS_OK;
}
return CallQueryInterface(treeOwner, aTreeOwner);
}
nsresult
GlobalWindowImpl::GetWebBrowserChrome(nsIWebBrowserChrome **aBrowserChrome)
{
nsCOMPtr<nsIDocShellTreeOwner> treeOwner;
GetTreeOwner(getter_AddRefs(treeOwner));
nsCOMPtr<nsIWebBrowserChrome> browserChrome(do_GetInterface(treeOwner));
*aBrowserChrome = browserChrome;
NS_IF_ADDREF(*aBrowserChrome);
return NS_OK;
}
nsresult
GlobalWindowImpl::GetScrollInfo(nsIScrollableView **aScrollableView,
float *aP2T, float *aT2P)
{
*aScrollableView = nsnull;
nsCOMPtr<nsIDOMWindowInternal> parent;
GetParentInternal(getter_AddRefs(parent));
if (parent) {
PRInt32 dummy;
// Force a flush in the parent
parent->GetScrollX(&dummy);
}
// Flush pending notifications so that the presentation is up to
// date.
FlushPendingNotifications();
nsCOMPtr<nsIPresContext> presContext;
mDocShell->GetPresContext(getter_AddRefs(presContext));
if (presContext) {
presContext->GetPixelsToTwips(aP2T);
presContext->GetTwipsToPixels(aT2P);
nsCOMPtr<nsIPresShell> presShell;
presContext->GetShell(getter_AddRefs(presShell));
if (presShell) {
nsCOMPtr<nsIViewManager> vm;
presShell->GetViewManager(getter_AddRefs(vm));
if (vm)
return vm->GetRootScrollableView(aScrollableView);
}
}
return NS_OK;
}
nsresult
GlobalWindowImpl::SecurityCheckURL(const char *aURL)
{
nsresult rv;
JSContext *cx = nsnull;
// get JSContext
NS_ASSERTION(mContext, "opening window missing its context");
NS_ASSERTION(mDocument, "opening window missing its document");
if (!mContext || !mDocument)
return NS_ERROR_FAILURE;
// get the JSContext from the call stack
nsCOMPtr<nsIThreadJSContextStack> stack(do_GetService(sJSStackContractID));
if (stack)
stack->Peek(&cx);
if (!cx) // not bloody likely. but if there's no JS on the call stack,
return NS_OK; // then we should pass the security check.
/* resolve the URI, which could be relative to the calling window
(note the algorithm to get the base URI should match the one
used to actually kick off the load in nsWindowWatcher.cpp). */
nsCOMPtr<nsIURI> baseURI;
nsCOMPtr<nsIURI> uriToLoad;
nsCOMPtr<nsIScriptContext> scriptcx =
NS_STATIC_CAST(nsIScriptContext *, ::JS_GetContextPrivate(cx));
if (scriptcx) {
nsCOMPtr<nsIScriptGlobalObject> gobj;
scriptcx->GetGlobalObject(getter_AddRefs(gobj));
nsCOMPtr<nsIDOMWindow> caller(do_QueryInterface(gobj));
if (caller) {
nsCOMPtr<nsIDOMDocument> callerDOMdoc;
caller->GetDocument(getter_AddRefs(callerDOMdoc));
nsCOMPtr<nsIDocument> callerDoc(do_QueryInterface(callerDOMdoc));
if (callerDoc)
callerDoc->GetDocumentURL(getter_AddRefs(baseURI));
}
}
rv = NS_NewURI(getter_AddRefs(uriToLoad), aURL, baseURI);
if (NS_FAILED(rv))
return rv;
nsCOMPtr<nsIScriptSecurityManager> secMan;
if (NS_FAILED(mContext->GetSecurityManager(getter_AddRefs(secMan))) ||
NS_FAILED(secMan->CheckLoadURIFromScript(cx, uriToLoad)))
return NS_ERROR_FAILURE;
return NS_OK;
}
void GlobalWindowImpl::FlushPendingNotifications()
{
if (mDocument) {
nsCOMPtr<nsIDocument> doc(do_QueryInterface(mDocument));
if (doc)
doc->FlushPendingNotifications();
}
}
//*****************************************************************************
// GlobalWindowImpl: Creator Function (This should go away)
//*****************************************************************************
nsresult NS_NewScriptGlobalObject(nsIScriptGlobalObject **aResult)
1998-07-16 01:16:47 +00:00
{
NS_ENSURE_ARG_POINTER(aResult);
*aResult = nsnull;
1998-07-16 01:16:47 +00:00
GlobalWindowImpl *global =
new GlobalWindowImpl();
NS_ENSURE_TRUE(global, NS_ERROR_OUT_OF_MEMORY);
return CallQueryInterface(NS_STATIC_CAST(nsIScriptGlobalObject *, global),
aResult);
1998-07-16 01:16:47 +00:00
}
//*****************************************************************************
//*** NavigatorImpl: Object Management
//*****************************************************************************
NavigatorImpl::NavigatorImpl(nsIDocShell *aDocShell) :
mMimeTypes(nsnull),
mPlugins(nsnull),
mDocShell(aDocShell)
1998-07-16 01:16:47 +00:00
{
NS_INIT_REFCNT();
1998-07-16 01:16:47 +00:00
}
NavigatorImpl::~NavigatorImpl()
{
NS_IF_RELEASE(mMimeTypes);
NS_IF_RELEASE(mPlugins);
1998-07-16 01:16:47 +00:00
}
//*****************************************************************************
// NavigatorImpl::nsISupports
//*****************************************************************************
1998-07-16 01:16:47 +00:00
// QueryInterface implementation for NavigatorImpl
NS_INTERFACE_MAP_BEGIN(NavigatorImpl)
NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIDOMNavigator)
NS_INTERFACE_MAP_ENTRY(nsIDOMNavigator)
NS_INTERFACE_MAP_ENTRY(nsIDOMJSNavigator)
NS_DOM_INTERFACE_MAP_ENTRY_CLASSINFO(Navigator)
NS_INTERFACE_MAP_END
1998-07-16 01:16:47 +00:00
NS_IMPL_ADDREF(NavigatorImpl)
NS_IMPL_RELEASE(NavigatorImpl)
1998-07-16 01:16:47 +00:00
void NavigatorImpl::SetDocShell(nsIDocShell *aDocShell)
{
mDocShell = aDocShell;
if (mPlugins)
mPlugins->SetDocShell(aDocShell);
}
//*****************************************************************************
// NavigatorImpl::nsIDOMNavigator
//*****************************************************************************
NS_IMETHODIMP
NavigatorImpl::GetUserAgent(nsAWritableString& aUserAgent)
1998-07-16 01:16:47 +00:00
{
nsresult res;
nsCOMPtr<nsIHttpProtocolHandler>
service(do_GetService(kHTTPHandlerCID, &res));
if (NS_SUCCEEDED(res) && service) {
char *ua = nsnull;
res = service->GetUserAgent(&ua);
aUserAgent = NS_ConvertASCIItoUCS2(ua);
Recycle(ua);
}
return res;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP
NavigatorImpl::GetAppCodeName(nsAWritableString& aAppCodeName)
1998-07-16 01:16:47 +00:00
{
nsresult res;
nsCOMPtr<nsIHttpProtocolHandler>
service(do_GetService(kHTTPHandlerCID, &res));
if (NS_SUCCEEDED(res) && service) {
char *appName = nsnull;
res = service->GetAppName(&appName);
aAppCodeName = NS_ConvertASCIItoUCS2(appName);
Recycle(appName);
}
return res;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP
NavigatorImpl::GetAppVersion(nsAWritableString& aAppVersion)
1998-07-16 01:16:47 +00:00
{
nsresult res;
nsCOMPtr<nsIHttpProtocolHandler>
service(do_GetService(kHTTPHandlerCID, &res));
if (NS_SUCCEEDED(res) && service) {
char *str = nsnull;
res = service->GetAppVersion(&str);
aAppVersion = NS_ConvertASCIItoUCS2(str);
Recycle(str);
aAppVersion.Append(NS_LITERAL_STRING(" ("));
res = service->GetPlatform(&str);
if (NS_FAILED(res))
return res;
aAppVersion += NS_ConvertASCIItoUCS2(str);
Recycle(str);
aAppVersion.Append(NS_LITERAL_STRING("; "));
res = service->GetLanguage(&str);
if (NS_FAILED(res))
return res;
aAppVersion += NS_ConvertASCIItoUCS2(str);
Recycle(str);
aAppVersion.Append(PRUnichar(')'));
}
return res;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP
NavigatorImpl::GetAppName(nsAWritableString& aAppName)
1998-07-16 01:16:47 +00:00
{
aAppName.Assign(NS_LITERAL_STRING("Netscape"));
return NS_OK;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP
NavigatorImpl::GetLanguage(nsAWritableString& aLanguage)
1998-07-16 01:16:47 +00:00
{
nsresult res;
nsCOMPtr<nsIHttpProtocolHandler>
service(do_GetService(kHTTPHandlerCID, &res));
if (NS_SUCCEEDED(res) && service) {
char *lang = nsnull;
res = service->GetLanguage(&lang);
aLanguage = NS_ConvertASCIItoUCS2(lang);
Recycle(lang);
}
return res;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP
NavigatorImpl::GetPlatform(nsAWritableString& aPlatform)
1998-07-16 01:16:47 +00:00
{
nsresult res;
nsCOMPtr<nsIHttpProtocolHandler>
service(do_GetService(kHTTPHandlerCID, &res));
if (NS_SUCCEEDED(res) && service) {
// sorry for the #if platform ugliness, but Communicator is
// likewise hardcoded and we're seeking backward compatibility
// here (bug 47080)
#if defined(WIN32)
aPlatform = NS_LITERAL_STRING("Win32");
#elif defined(XP_MAC)
// XXX not sure what to do about Mac OS X on non-PPC, but since Comm 4.x
// doesn't know about it this will actually be backward compatible
aPlatform = NS_LITERAL_STRING("MacPPC");
#else
// XXX Communicator uses compiled-in build-time string defines
// to indicate the platform it was compiled *for*, not what it is
// currently running *on* which is what this does.
char *plat = nsnull;
res = service->GetOscpu(&plat);
aPlatform = NS_ConvertASCIItoUCS2(plat);
Recycle(plat);
#endif
}
return res;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP
NavigatorImpl::GetOscpu(nsAWritableString& aOSCPU)
{
nsresult res;
nsCOMPtr<nsIHttpProtocolHandler>
service(do_GetService(kHTTPHandlerCID, &res));
if (NS_SUCCEEDED(res) && service) {
char *oscpu = nsnull;
res = service->GetOscpu(&oscpu);
aOSCPU = NS_ConvertASCIItoUCS2(oscpu);
Recycle(oscpu);
}
return res;
}
NS_IMETHODIMP
NavigatorImpl::GetVendor(nsAWritableString& aVendor)
{
nsresult res;
nsCOMPtr<nsIHttpProtocolHandler>
service(do_GetService(kHTTPHandlerCID, &res));
if (NS_SUCCEEDED(res) && service) {
char *vendor = nsnull;
res = service->GetVendor(&vendor);
aVendor = NS_ConvertASCIItoUCS2(vendor);
Recycle(vendor);
}
return res;
}
NS_IMETHODIMP
NavigatorImpl::GetVendorSub(nsAWritableString& aVendorSub)
{
nsresult res;
nsCOMPtr<nsIHttpProtocolHandler>
service(do_GetService(kHTTPHandlerCID, &res));
if (NS_SUCCEEDED(res) && service) {
char *vendor = nsnull;
res = service->GetVendorSub(&vendor);
aVendorSub = NS_ConvertASCIItoUCS2(vendor);
Recycle(vendor);
}
return res;
}
NS_IMETHODIMP
NavigatorImpl::GetProduct(nsAWritableString& aProduct)
{
nsresult res;
nsCOMPtr<nsIHttpProtocolHandler>
service(do_GetService(kHTTPHandlerCID, &res));
if (NS_SUCCEEDED(res) && service) {
char *product = nsnull;
res = service->GetProduct(&product);
aProduct = NS_ConvertASCIItoUCS2(product);
Recycle(product);
}
return res;
}
NS_IMETHODIMP
NavigatorImpl::GetProductSub(nsAWritableString& aProductSub)
{
nsresult res;
nsCOMPtr<nsIHttpProtocolHandler>
service(do_GetService(kHTTPHandlerCID, &res));
if (NS_SUCCEEDED(res) && service) {
char *productSub = nsnull;
res = service->GetProductSub(&productSub);
aProductSub = NS_ConvertASCIItoUCS2(productSub);
Recycle(productSub);
}
return res;
}
NS_IMETHODIMP
NavigatorImpl::GetSecurityPolicy(nsAWritableString& aSecurityPolicy)
1998-07-16 01:16:47 +00:00
{
return NS_OK;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP
NavigatorImpl::GetMimeTypes(nsIDOMMimeTypeArray **aMimeTypes)
{
if (!mMimeTypes) {
mMimeTypes = new MimeTypeArrayImpl(this);
NS_IF_ADDREF(mMimeTypes);
}
*aMimeTypes = mMimeTypes;
NS_IF_ADDREF(mMimeTypes);
return NS_OK;
}
NS_IMETHODIMP
NavigatorImpl::GetPlugins(nsIDOMPluginArray **aPlugins)
{
if (!mPlugins) {
mPlugins = new PluginArrayImpl(this, mDocShell);
if (!mPlugins)
return NS_ERROR_OUT_OF_MEMORY;
NS_ADDREF(mPlugins);
}
*aPlugins = mPlugins;
NS_ADDREF(mPlugins);
return NS_OK;
}
NS_IMETHODIMP
NavigatorImpl::GetCookieEnabled(PRBool *aCookieEnabled)
{
nsresult rv = NS_OK;
*aCookieEnabled = PR_FALSE;
nsCOMPtr<nsIPref> prefs(do_GetService(kPrefServiceCID, &rv));
if (NS_FAILED(rv) || prefs == nsnull)
return rv;
PRInt32 cookieBehaviorPref;
rv = prefs->GetIntPref("network.cookie.cookieBehavior", &cookieBehaviorPref);
if (NS_FAILED(rv))
return rv;
const PRInt32 DONT_USE = 2;
*aCookieEnabled = (cookieBehaviorPref != DONT_USE);
return rv;
}
NS_IMETHODIMP
NavigatorImpl::JavaEnabled(PRBool *aReturn)
1998-07-16 01:16:47 +00:00
{
nsresult rv = NS_OK;
*aReturn = PR_FALSE;
// determine whether user has enabled java.
nsCOMPtr<nsIPref> prefs(do_GetService(kPrefServiceCID, &rv));
if (NS_FAILED(rv))
return rv;
// if pref doesn't exist, map result to false.
if (NS_FAILED(prefs->GetBoolPref("security.enable_java", aReturn))) {
*aReturn = PR_FALSE;
return NS_OK;
}
// if Java is not enabled, result is false and return reight away
if (!*aReturn)
return NS_OK;
// Ask the nsIJVMManager if Java is enabled
nsCOMPtr<nsIJVMManager> jvmService = do_GetService(kJVMServiceCID);
if (jvmService) {
jvmService->GetJavaEnabled(aReturn);
}
else {
*aReturn = PR_FALSE;
}
return rv;
1998-07-16 01:16:47 +00:00
}
NS_IMETHODIMP
NavigatorImpl::TaintEnabled(PRBool *aReturn)
{
*aReturn = PR_FALSE;
return NS_OK;
}
NS_IMETHODIMP
NavigatorImpl::Preference()
{
nsresult rv;
nsCOMPtr<nsIXPConnect> xpc(do_GetService(nsIXPConnect::GetCID(), &rv));
NS_ENSURE_SUCCESS(rv, rv);
nsCOMPtr<nsIXPCNativeCallContext> ncc;
rv = xpc->GetCurrentNativeCallContext(getter_AddRefs(ncc));
NS_ENSURE_SUCCESS(rv, rv);
if (!ncc)
return NS_ERROR_NOT_AVAILABLE;
PRUint32 argc;
ncc->GetArgc(&argc);
if (argc == 0) {
// No arguments means there's nothing to be done here.
return NS_OK;
}
jsval *argv = nsnull;
ncc->GetArgvPtr(&argv);
NS_ENSURE_TRUE(argv, NS_ERROR_UNEXPECTED);
JSContext *cx = nsnull;
rv = ncc->GetJSContext(&cx);
NS_ENSURE_SUCCESS(rv, rv);
//--Check to see if the caller is allowed to access prefs
nsCOMPtr<nsIScriptSecurityManager> secMan =
do_GetService("@mozilla.org/scriptsecuritymanager;1", &rv);
NS_ENSURE_SUCCESS(rv, rv);
PRUint32 action;
if (argc == 1)
action = nsIXPCSecurityManager::ACCESS_GET_PROPERTY;
else
action = nsIXPCSecurityManager::ACCESS_SET_PROPERTY;
rv = secMan->CheckPropertyAccess(cx, nsnull,
"Navigator", "preferenceinternal", action);
if (NS_FAILED(rv))
{
return NS_OK;
}
nsCOMPtr<nsIPref> pref(do_GetService(kPrefServiceCID, &rv));
NS_ENSURE_SUCCESS(rv, rv);
JSString *str = ::JS_ValueToString(cx, argv[0]);
NS_ENSURE_TRUE(str, NS_ERROR_OUT_OF_MEMORY);
jsval *retval = nsnull;
rv = ncc->GetRetValPtr(&retval);
NS_ENSURE_SUCCESS(rv, rv);
char *prefStr = ::JS_GetStringBytes(str);
if (argc == 1) {
PRInt32 prefType;
pref->GetPrefType(prefStr, &prefType);
switch (prefType & nsIPref::ePrefValuetypeMask) {
case nsIPref::ePrefString:
{
nsXPIDLCString prefCharVal;
rv = pref->CopyCharPref(prefStr, getter_Copies(prefCharVal));
NS_ENSURE_SUCCESS(rv, rv);
JSString *retStr = ::JS_NewStringCopyZ(cx, prefCharVal);
NS_ENSURE_TRUE(retStr, NS_ERROR_OUT_OF_MEMORY);
*retval = STRING_TO_JSVAL(retStr);
break;
}
case nsIPref::ePrefInt:
{
PRInt32 prefIntVal;
rv = pref->GetIntPref(prefStr, &prefIntVal);
NS_ENSURE_SUCCESS(rv, rv);
*retval = INT_TO_JSVAL(prefIntVal);
break;
}
case nsIPref::ePrefBool:
{
PRBool prefBoolVal;
rv = pref->GetBoolPref(prefStr, &prefBoolVal);
NS_ENSURE_SUCCESS(rv, rv);
*retval = BOOLEAN_TO_JSVAL(prefBoolVal);
break;
}
default:
{
// Nothing we can do here...
return ncc->SetReturnValueWasSet(PR_FALSE);
}
}
ncc->SetReturnValueWasSet(PR_TRUE);
} else {
if (JSVAL_IS_STRING(argv[1])) {
JSString *valueJSStr = ::JS_ValueToString(cx, argv[1]);
NS_ENSURE_TRUE(valueJSStr, NS_ERROR_OUT_OF_MEMORY);
rv = pref->SetCharPref(prefStr, ::JS_GetStringBytes(valueJSStr));
} else if (JSVAL_IS_INT(argv[1])) {
jsint valueInt = JSVAL_TO_INT(argv[1]);
rv = pref->SetIntPref(prefStr, (PRInt32) valueInt);
} else if (JSVAL_IS_BOOLEAN(argv[1])) {
JSBool valueBool = JSVAL_TO_BOOLEAN(argv[1]);
rv = pref->SetBoolPref(prefStr, (PRBool) valueBool);
} else if (JSVAL_IS_NULL(argv[1])) {
rv = pref->DeleteBranch(prefStr);
}
}
return rv;
}
nsresult
NavigatorImpl::RefreshMIMEArray()
{
nsresult rv = NS_OK;
if (mMimeTypes)
rv = mMimeTypes->Refresh();
return rv;
}
#ifdef XP_MAC
#pragma mark -
#endif
//*****************************************************************************
//*** DOM Controller Stuff
//*****************************************************************************
#ifdef DOM_CONTROLLER
// nsDOMWindowController
const char * const sCopyString = "cmd_copy";
const char * const sCutString = "cmd_cut";
const char * const sPasteString = "cmd_paste";
const char * const sSelectAllString = "cmd_selectAll";
const char * const sSelectNoneString = "cmd_selectNone";
const char * const sCopyLinkString = "cmd_copyLink";
const char * const sCopyImageLocationString = "cmd_copyImageLocation";
const char * const sCopyImageContentsString = "cmd_copyImageContents";
const char * const sScrollTopString = "cmd_scrollTop";
const char * const sScrollBottomString = "cmd_scrollBottom";
const char * const sScrollPageUpString = "cmd_scrollPageUp";
const char * const sScrollPageDownString = "cmd_scrollPageDown";
const char * const sScrollLineUpString = "cmd_scrollLineUp";
const char * const sScrollLineDownString = "cmd_scrollLineDown";
const char * const sScrollLeftString = "cmd_scrollLeft";
const char * const sScrollRightString = "cmd_scrollRight";
// These are so the browser can use editor navigation key bindings
// helps with accessibility (boolean pref accessibility.browsewithcaret)
const char * const sSelectCharPreviousString = "cmd_selectCharPrevious";
const char * const sSelectCharNextString = "cmd_selectCharNext";
const char * const sWordPreviousString = "cmd_wordPrevious";
const char * const sWordNextString = "cmd_wordNext";
const char * const sSelectWordPreviousString = "cmd_selectWordPrevious";
const char * const sSelectWordNextString = "cmd_selectWordNext";
const char * const sBeginLineString = "cmd_beginLine";
const char * const sEndLineString = "cmd_endLine";
const char * const sSelectBeginLineString = "cmd_selectBeginLine";
const char * const sSelectEndLineString = "cmd_selectEndLine";
1999-12-15 03:59:38 +00:00
const char * const sSelectLinePreviousString = "cmd_selectLinePrevious";
const char * const sSelectLineNextString = "cmd_selectLineNext";
const char * const sSelectPagePreviousString = "cmd_selectPagePrevious";
const char * const sSelectPageNextString = "cmd_selectPageNext";
const char * const sSelectMoveTopString = "cmd_selectMoveTop";
const char * const sSelectMoveBottomString = "cmd_selectMoveBottom";
NS_IMPL_ADDREF(nsDOMWindowController)
NS_IMPL_RELEASE(nsDOMWindowController)
NS_INTERFACE_MAP_BEGIN(nsDOMWindowController)
NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIController)
NS_INTERFACE_MAP_ENTRY(nsIController)
NS_INTERFACE_MAP_END
nsDOMWindowController::nsDOMWindowController(nsIDOMWindowInternal *aWindow)
{
NS_INIT_REFCNT();
mWindow = aWindow;
// Set browse with caret flag so we don't need to every time
mBrowseWithCaret = PR_FALSE;
nsCOMPtr<nsIPref> prefs(do_GetService(kPrefServiceCID));
if (prefs)
prefs->GetBoolPref("accessibility.browsewithcaret", &mBrowseWithCaret);
}
nsresult
nsDOMWindowController::GetEditInterface(nsIContentViewerEdit **aEditInterface)
{
nsCOMPtr<nsIScriptGlobalObject> sgo(do_QueryInterface(mWindow));
NS_ENSURE_TRUE(sgo, NS_ERROR_FAILURE);
nsCOMPtr<nsIDocShell> docShell;
sgo->GetDocShell(getter_AddRefs(docShell));
NS_ENSURE_TRUE(docShell, NS_ERROR_FAILURE);
nsCOMPtr<nsIContentViewer> viewer;
docShell->GetContentViewer(getter_AddRefs(viewer));
nsCOMPtr<nsIContentViewerEdit> edit(do_QueryInterface(viewer));
NS_ENSURE_TRUE(edit, NS_ERROR_FAILURE);
*aEditInterface = edit;
NS_ADDREF(*aEditInterface);
return NS_OK;
}
nsresult
nsDOMWindowController::GetPresShell(nsIPresShell **aPresShell)
{
nsCOMPtr<nsIScriptGlobalObject> sgo(do_QueryInterface(mWindow));
NS_ENSURE_TRUE(sgo, NS_ERROR_FAILURE);
nsCOMPtr<nsIDocShell> docShell;
sgo->GetDocShell(getter_AddRefs(docShell));
NS_ENSURE_TRUE(docShell, NS_ERROR_FAILURE);
NS_ENSURE_SUCCESS(docShell->GetPresShell(aPresShell), NS_ERROR_FAILURE);
return NS_OK;
}
nsresult
nsDOMWindowController::GetSelectionController(nsISelectionController **aSelCon)
{
nsCOMPtr<nsIPresShell> presShell;
nsresult result = GetPresShell(getter_AddRefs(presShell));
if (presShell && NS_SUCCEEDED(result)) {
nsCOMPtr<nsISelectionController> selController =
do_QueryInterface(presShell);
if (selController) {
*aSelCon = selController;
NS_ADDREF(*aSelCon);
return NS_OK;
}
}
return NS_FAILED(result) ? result : NS_ERROR_FAILURE;
}
NS_IMETHODIMP
nsDOMWindowController::IsCommandEnabled(const nsAReadableString& aCommand,
PRBool *aResult)
{
NS_ENSURE_ARG_POINTER(aResult);
*aResult = PR_FALSE;
nsresult rv = NS_ERROR_FAILURE;
nsCOMPtr<nsIContentViewerEdit> editInterface;
rv = GetEditInterface(getter_AddRefs(editInterface));
NS_ENSURE_SUCCESS(rv, rv);
NS_ENSURE_TRUE(editInterface, NS_ERROR_NOT_INITIALIZED);
nsCAutoString commandName;
commandName.AssignWithConversion(aCommand);
if (commandName.Equals(sCopyString)) {
rv = editInterface->GetCopyable(aResult);
}
else if (commandName.Equals(sCutString)) {
rv = editInterface->GetCutable(aResult);
}
else if (commandName.Equals(sPasteString)) {
rv = editInterface->GetPasteable(aResult);
}
else if (commandName.Equals(sSelectAllString)) {
*aResult = PR_TRUE;
rv = NS_OK;
}
else if (commandName.Equals(sSelectNoneString)) {
*aResult = PR_TRUE;
rv = NS_OK;
}
else if (commandName.Equals(sCopyLinkString)) {
rv = editInterface->GetInLink(aResult);
}
else if (commandName.Equals(sCopyImageLocationString)) {
rv = editInterface->GetInImage(aResult);
}
else if (commandName.Equals(sCopyImageContentsString)) {
rv = editInterface->GetInImage(aResult);
}
return rv;
}
NS_IMETHODIMP
nsDOMWindowController::SupportsCommand(const nsAReadableString& aCommand,
PRBool *outSupported)
{
NS_ENSURE_ARG_POINTER(outSupported);
*outSupported = PR_FALSE;
nsCAutoString commandName;
commandName.AssignWithConversion(aCommand);
if (commandName.Equals(sCopyString) ||
commandName.Equals(sSelectAllString) ||
commandName.Equals(sSelectNoneString) ||
commandName.Equals(sCutString) ||
commandName.Equals(sPasteString) ||
commandName.Equals(sScrollTopString) ||
commandName.Equals(sScrollBottomString) ||
commandName.Equals(sCopyLinkString) ||
commandName.Equals(sCopyImageLocationString) ||
commandName.Equals(sCopyImageContentsString) ||
commandName.Equals(sScrollPageUpString) ||
commandName.Equals(sScrollPageDownString) ||
commandName.Equals(sScrollLineUpString) ||
commandName.Equals(sScrollLineDownString) ||
commandName.Equals(sScrollLeftString) ||
commandName.Equals(sScrollRightString) ||
commandName.Equals(sSelectCharPreviousString) ||
commandName.Equals(sSelectCharNextString) ||
commandName.Equals(sWordPreviousString) ||
commandName.Equals(sWordNextString) ||
commandName.Equals(sSelectWordPreviousString) ||
commandName.Equals(sSelectWordNextString) ||
commandName.Equals(sBeginLineString) ||
commandName.Equals(sEndLineString) ||
commandName.Equals(sSelectBeginLineString) ||
commandName.Equals(sSelectEndLineString) ||
commandName.Equals(sSelectLinePreviousString) ||
commandName.Equals(sSelectLineNextString) ||
commandName.Equals(sSelectPagePreviousString) ||
commandName.Equals(sSelectPageNextString) ||
commandName.Equals(sSelectMoveTopString) ||
commandName.Equals(sSelectMoveBottomString)
)
*outSupported = PR_TRUE;
return NS_OK;
}
NS_IMETHODIMP
nsDOMWindowController::DoCommand(const nsAReadableString & aCommand)
{
nsresult rv = NS_ERROR_FAILURE;
nsCAutoString commandName;
commandName.AssignWithConversion(aCommand);
if (commandName.Equals(sCopyString) ||
commandName.Equals(sSelectAllString) ||
commandName.Equals(sSelectNoneString) ||
commandName.Equals(sCutString) ||
commandName.Equals(sPasteString) ||
commandName.Equals(sCopyLinkString) ||
commandName.Equals(sCopyImageLocationString) ||
commandName.Equals(sCopyImageContentsString)) {
rv = DoCommandWithEditInterface(commandName);
}
else if (commandName.Equals(sScrollTopString) ||
commandName.Equals(sScrollBottomString) ||
commandName.Equals(sScrollPageUpString) ||
commandName.Equals(sScrollPageDownString) ||
commandName.Equals(sScrollLineUpString) ||
commandName.Equals(sScrollLineDownString) ||
commandName.Equals(sScrollLeftString) ||
commandName.Equals(sScrollRightString) ||
commandName.Equals(sSelectCharPreviousString) ||
commandName.Equals(sSelectCharNextString) ||
commandName.Equals(sWordPreviousString) ||
commandName.Equals(sWordNextString) ||
commandName.Equals(sSelectWordPreviousString) ||
commandName.Equals(sSelectWordNextString) ||
commandName.Equals(sBeginLineString) ||
commandName.Equals(sEndLineString) ||
commandName.Equals(sSelectBeginLineString) ||
commandName.Equals(sSelectEndLineString) ||
commandName.Equals(sSelectLinePreviousString) ||
commandName.Equals(sSelectLineNextString) ||
commandName.Equals(sSelectMoveTopString) ||
commandName.Equals(sSelectMoveBottomString)) {
rv = DoCommandWithSelectionController(commandName);
}
return rv;
}
nsresult
nsDOMWindowController::DoCommandWithEditInterface(const nsCString& aCommandName)
{
// get edit interface...
nsCOMPtr<nsIContentViewerEdit> editInterface;
nsresult rv = GetEditInterface(getter_AddRefs(editInterface));
// if we can't get an edit interface, that's bad
NS_ENSURE_SUCCESS(rv, rv);
NS_ENSURE_TRUE(editInterface, NS_ERROR_NOT_INITIALIZED);
rv = NS_ERROR_FAILURE;
if (aCommandName.Equals(sCopyString))
rv = editInterface->CopySelection();
else if (aCommandName.Equals(sSelectAllString))
rv = editInterface->SelectAll();
else if (aCommandName.Equals(sSelectNoneString))
rv = editInterface->ClearSelection();
else if (aCommandName.Equals(sCutString))
rv = editInterface->CutSelection();
else if (aCommandName.Equals(sPasteString))
rv = editInterface->Paste();
else if (aCommandName.Equals(sCopyLinkString))
rv = editInterface->CopyLinkLocation();
else if (aCommandName.Equals(sCopyImageLocationString))
rv = editInterface->CopyImageLocation();
else if (aCommandName.Equals(sCopyImageContentsString))
rv = editInterface->CopyImageContents();
return rv;
}
nsresult
nsDOMWindowController::DoCommandWithSelectionController(const nsCString& aCommandName) {
// get selection controller...
nsCOMPtr<nsISelectionController> selCont;
nsresult rv = GetSelectionController(getter_AddRefs(selCont));
// if we can't get a selection controller, that's bad
NS_ENSURE_SUCCESS(rv, rv);
NS_ENSURE_TRUE(selCont, NS_ERROR_NOT_INITIALIZED);
rv = NS_ERROR_FAILURE;
if (aCommandName.Equals(sScrollTopString))
rv = (mBrowseWithCaret? selCont->CompleteMove(PR_FALSE, PR_FALSE): selCont->CompleteScroll(PR_FALSE));
else if (aCommandName.Equals(sScrollBottomString))
rv = (mBrowseWithCaret? selCont->CompleteMove(PR_TRUE, PR_FALSE): selCont->CompleteScroll(PR_TRUE));
else if (aCommandName.Equals(sScrollPageUpString))
rv = (mBrowseWithCaret? selCont->PageMove(PR_FALSE, PR_FALSE): selCont->ScrollPage(PR_FALSE));
else if (aCommandName.Equals(sScrollPageDownString))
rv = (mBrowseWithCaret? selCont->PageMove(PR_TRUE, PR_FALSE): selCont->ScrollPage(PR_TRUE));
else if (aCommandName.Equals(sScrollLineUpString))
rv = (mBrowseWithCaret? selCont->LineMove(PR_FALSE, PR_FALSE): selCont->ScrollLine(PR_FALSE));
else if (aCommandName.Equals(sScrollLineDownString))
rv = (mBrowseWithCaret? selCont->LineMove(PR_TRUE, PR_FALSE): selCont->ScrollLine(PR_TRUE));
else if (aCommandName.Equals(sScrollLeftString))
rv = (mBrowseWithCaret? selCont->CharacterMove(PR_FALSE, PR_FALSE): selCont->ScrollHorizontal(PR_TRUE));
else if (aCommandName.Equals(sScrollRightString))
rv = (mBrowseWithCaret? selCont->CharacterMove(PR_TRUE, PR_FALSE): selCont->ScrollHorizontal(PR_FALSE));
// These commands are so the browser can use editor navigation key bindings -
// Helps with accessibility - aaronl@chorus.net
else if (aCommandName.Equals(sSelectCharPreviousString))
rv = selCont->CharacterMove(PR_FALSE, PR_TRUE);
else if (aCommandName.Equals(sSelectCharNextString))
rv = selCont->CharacterMove(PR_TRUE, PR_TRUE);
else if (aCommandName.Equals(sWordPreviousString))
rv = selCont->WordMove(PR_FALSE, PR_FALSE);
else if (aCommandName.Equals(sWordNextString))
rv = selCont->WordMove(PR_TRUE, PR_FALSE);
else if (aCommandName.Equals(sSelectWordPreviousString))
rv = selCont->WordMove(PR_FALSE, PR_TRUE);
else if (aCommandName.Equals(sSelectWordNextString))
rv = selCont->WordMove(PR_TRUE, PR_TRUE);
else if (aCommandName.Equals(sBeginLineString))
rv = selCont->IntraLineMove(PR_FALSE, PR_FALSE);
else if (aCommandName.Equals(sEndLineString))
rv = selCont->IntraLineMove(PR_TRUE, PR_FALSE);
else if (aCommandName.Equals(sSelectBeginLineString))
rv = selCont->IntraLineMove(PR_FALSE, PR_TRUE);
else if (aCommandName.Equals(sSelectEndLineString))
rv = selCont->IntraLineMove(PR_TRUE, PR_TRUE);
else if (aCommandName.Equals(sSelectLinePreviousString))
rv = selCont->LineMove(PR_FALSE, PR_TRUE);
else if (aCommandName.Equals(sSelectLineNextString))
rv = selCont->LineMove(PR_TRUE, PR_TRUE);
else if (aCommandName.Equals(sSelectMoveTopString))
rv = selCont->CompleteMove(PR_FALSE, PR_TRUE);
else if (aCommandName.Equals(sSelectMoveBottomString))
rv = selCont->CompleteMove(PR_TRUE, PR_TRUE);
return rv;
}
NS_IMETHODIMP
nsDOMWindowController::OnEvent(const nsAReadableString & aEventName)
{
return NS_OK;
}
#endif