/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this file, * You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "nsBrowserElement.h" #include "mozilla/Preferences.h" #include "mozilla/Services.h" #include "mozilla/dom/BrowserElementBinding.h" #include "mozilla/dom/DOMRequest.h" #include "mozilla/dom/ScriptSettings.h" #include "mozilla/dom/ToJSValue.h" #include "nsComponentManagerUtils.h" #include "nsContentUtils.h" #include "nsFrameLoader.h" #include "nsIDOMDOMRequest.h" #include "nsIDOMElement.h" #include "nsINode.h" #include "nsIObserver.h" #include "nsIObserverService.h" #include "nsIPrincipal.h" #include "nsWeakReference.h" using namespace mozilla::dom; namespace mozilla { static const char kRemoteBrowserPending[] = "remote-browser-pending"; static const char kInprocessBrowserShown[] = "inprocess-browser-shown"; class nsBrowserElement::BrowserShownObserver : public nsIObserver , public nsSupportsWeakReference { public: explicit BrowserShownObserver(nsBrowserElement* aBrowserElement); NS_DECL_ISUPPORTS NS_DECL_NSIOBSERVER void AddObserver(); void RemoveObserver(); private: virtual ~BrowserShownObserver(); // Weak reference to the browser element. nsBrowserElement has a // reference to us. nsBrowserElement's destructor is responsible to // null out this weak reference via RemoveObserver() nsBrowserElement* mBrowserElement; }; NS_IMPL_ISUPPORTS(nsBrowserElement::BrowserShownObserver, nsIObserver, nsISupportsWeakReference) nsBrowserElement::BrowserShownObserver::BrowserShownObserver(nsBrowserElement* aBrowserElement) : mBrowserElement(aBrowserElement) { } nsBrowserElement::BrowserShownObserver::~BrowserShownObserver() { RemoveObserver(); } NS_IMETHODIMP nsBrowserElement::BrowserShownObserver::Observe(nsISupports* aSubject, const char* aTopic, const char16_t* aData) { NS_ENSURE_TRUE(mBrowserElement, NS_OK); if (!strcmp(aTopic, kRemoteBrowserPending) || !strcmp(aTopic, kInprocessBrowserShown)) { nsCOMPtr frameLoader = do_QueryInterface(aSubject); nsCOMPtr myFrameLoader = mBrowserElement->GetFrameLoader(); // The browser element API needs the frameloader to // initialize. We still use the observer to get notified when the // frameloader is created. So we check if the frameloader created // is ours, then initialize the browser element API. if (frameLoader && frameLoader == myFrameLoader) { mBrowserElement->InitBrowserElementAPI(); } } return NS_OK; } void nsBrowserElement::BrowserShownObserver::AddObserver() { nsCOMPtr obs = services::GetObserverService(); if (obs) { obs->AddObserver(this, kRemoteBrowserPending, true); obs->AddObserver(this, kInprocessBrowserShown, true); } } void nsBrowserElement::BrowserShownObserver::RemoveObserver() { nsCOMPtr obs = services::GetObserverService(); if (obs) { obs->RemoveObserver(this, kRemoteBrowserPending); obs->RemoveObserver(this, kInprocessBrowserShown); } mBrowserElement = nullptr; } bool nsBrowserElement::IsBrowserElementOrThrow(ErrorResult& aRv) { if (mBrowserElementAPI) { return true; } aRv.Throw(NS_ERROR_DOM_INVALID_NODE_TYPE_ERR); return false; } bool nsBrowserElement::IsNotWidgetOrThrow(ErrorResult& aRv) { if (!mOwnerIsWidget) { return true; } aRv.Throw(NS_ERROR_DOM_INVALID_NODE_TYPE_ERR); return false; } void nsBrowserElement::InitBrowserElementAPI() { bool isBrowserOrApp; nsCOMPtr frameLoader = GetFrameLoader(); NS_ENSURE_TRUE_VOID(frameLoader); nsresult rv = frameLoader->GetOwnerIsBrowserOrAppFrame(&isBrowserOrApp); NS_ENSURE_SUCCESS_VOID(rv); rv = frameLoader->GetOwnerIsWidget(&mOwnerIsWidget); NS_ENSURE_SUCCESS_VOID(rv); if (!isBrowserOrApp) { return; } mBrowserElementAPI = do_CreateInstance("@mozilla.org/dom/browser-element-api;1"); if (mBrowserElementAPI) { mBrowserElementAPI->SetFrameLoader(frameLoader); } } nsBrowserElement::nsBrowserElement() : mOwnerIsWidget(false) { mObserver = new BrowserShownObserver(this); mObserver->AddObserver(); } nsBrowserElement::~nsBrowserElement() { mObserver->RemoveObserver(); } void nsBrowserElement::SetVisible(bool aVisible, ErrorResult& aRv) { NS_ENSURE_TRUE_VOID(IsBrowserElementOrThrow(aRv)); nsresult rv = mBrowserElementAPI->SetVisible(aVisible); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } } already_AddRefed nsBrowserElement::GetVisible(ErrorResult& aRv) { NS_ENSURE_TRUE(IsBrowserElementOrThrow(aRv), nullptr); nsCOMPtr req; nsresult rv = mBrowserElementAPI->GetVisible(getter_AddRefs(req)); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); return nullptr; } return req.forget().downcast(); } void nsBrowserElement::SetActive(bool aVisible, ErrorResult& aRv) { NS_ENSURE_TRUE_VOID(IsBrowserElementOrThrow(aRv)); nsresult rv = mBrowserElementAPI->SetActive(aVisible); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } } bool nsBrowserElement::GetActive(ErrorResult& aRv) { NS_ENSURE_TRUE(IsBrowserElementOrThrow(aRv), false); bool isActive; nsresult rv = mBrowserElementAPI->GetActive(&isActive); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); return false; } return isActive; } void nsBrowserElement::SendMouseEvent(const nsAString& aType, uint32_t aX, uint32_t aY, uint32_t aButton, uint32_t aClickCount, uint32_t aModifiers, ErrorResult& aRv) { NS_ENSURE_TRUE_VOID(IsBrowserElementOrThrow(aRv)); NS_ENSURE_TRUE_VOID(IsNotWidgetOrThrow(aRv)); nsresult rv = mBrowserElementAPI->SendMouseEvent(aType, aX, aY, aButton, aClickCount, aModifiers); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } } void nsBrowserElement::SendTouchEvent(const nsAString& aType, const Sequence& aIdentifiers, const Sequence& aXs, const Sequence& aYs, const Sequence& aRxs, const Sequence& aRys, const Sequence& aRotationAngles, const Sequence& aForces, uint32_t aCount, uint32_t aModifiers, ErrorResult& aRv) { NS_ENSURE_TRUE_VOID(IsBrowserElementOrThrow(aRv)); NS_ENSURE_TRUE_VOID(IsNotWidgetOrThrow(aRv)); if (aIdentifiers.Length() != aCount || aXs.Length() != aCount || aYs.Length() != aCount || aRxs.Length() != aCount || aRys.Length() != aCount || aRotationAngles.Length() != aCount || aForces.Length() != aCount) { aRv.Throw(NS_ERROR_DOM_INVALID_ACCESS_ERR); return; } nsresult rv = mBrowserElementAPI->SendTouchEvent(aType, aIdentifiers.Elements(), aXs.Elements(), aYs.Elements(), aRxs.Elements(), aRys.Elements(), aRotationAngles.Elements(), aForces.Elements(), aCount, aModifiers); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } } void nsBrowserElement::GoBack(ErrorResult& aRv) { NS_ENSURE_TRUE_VOID(IsBrowserElementOrThrow(aRv)); NS_ENSURE_TRUE_VOID(IsNotWidgetOrThrow(aRv)); nsresult rv = mBrowserElementAPI->GoBack(); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } } void nsBrowserElement::GoForward(ErrorResult& aRv) { NS_ENSURE_TRUE_VOID(IsBrowserElementOrThrow(aRv)); NS_ENSURE_TRUE_VOID(IsNotWidgetOrThrow(aRv)); nsresult rv = mBrowserElementAPI->GoForward(); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } } void nsBrowserElement::Reload(bool aHardReload, ErrorResult& aRv) { NS_ENSURE_TRUE_VOID(IsBrowserElementOrThrow(aRv)); NS_ENSURE_TRUE_VOID(IsNotWidgetOrThrow(aRv)); nsresult rv = mBrowserElementAPI->Reload(aHardReload); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } } void nsBrowserElement::Stop(ErrorResult& aRv) { NS_ENSURE_TRUE_VOID(IsBrowserElementOrThrow(aRv)); NS_ENSURE_TRUE_VOID(IsNotWidgetOrThrow(aRv)); nsresult rv = mBrowserElementAPI->Stop(); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } } already_AddRefed nsBrowserElement::Download(const nsAString& aUrl, const BrowserElementDownloadOptions& aOptions, ErrorResult& aRv) { NS_ENSURE_TRUE(IsBrowserElementOrThrow(aRv), nullptr); NS_ENSURE_TRUE(IsNotWidgetOrThrow(aRv), nullptr); nsCOMPtr req; nsCOMPtr wrappedObj = do_QueryInterface(mBrowserElementAPI); MOZ_ASSERT(wrappedObj, "Failed to get wrapped JS from XPCOM component."); AutoJSAPI jsapi; jsapi.Init(wrappedObj->GetJSObject()); JSContext* cx = jsapi.cx(); JS::Rooted options(cx); if (!ToJSValue(cx, aOptions, &options)) { aRv.Throw(NS_ERROR_OUT_OF_MEMORY); return nullptr; } nsresult rv = mBrowserElementAPI->Download(aUrl, options, getter_AddRefs(req)); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); return nullptr; } return req.forget().downcast(); } already_AddRefed nsBrowserElement::PurgeHistory(ErrorResult& aRv) { NS_ENSURE_TRUE(IsBrowserElementOrThrow(aRv), nullptr); NS_ENSURE_TRUE(IsNotWidgetOrThrow(aRv), nullptr); nsCOMPtr req; nsresult rv = mBrowserElementAPI->PurgeHistory(getter_AddRefs(req)); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); return nullptr; } return req.forget().downcast(); } already_AddRefed nsBrowserElement::GetScreenshot(uint32_t aWidth, uint32_t aHeight, const nsAString& aMimeType, ErrorResult& aRv) { NS_ENSURE_TRUE(IsBrowserElementOrThrow(aRv), nullptr); NS_ENSURE_TRUE(IsNotWidgetOrThrow(aRv), nullptr); nsCOMPtr req; nsresult rv = mBrowserElementAPI->GetScreenshot(aWidth, aHeight, aMimeType, getter_AddRefs(req)); if (NS_WARN_IF(NS_FAILED(rv))) { if (rv == NS_ERROR_INVALID_ARG) { aRv.Throw(NS_ERROR_DOM_INVALID_ACCESS_ERR); } else { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } return nullptr; } return req.forget().downcast(); } void nsBrowserElement::Zoom(float aZoom, ErrorResult& aRv) { NS_ENSURE_TRUE_VOID(IsBrowserElementOrThrow(aRv)); NS_ENSURE_TRUE_VOID(IsNotWidgetOrThrow(aRv)); nsresult rv = mBrowserElementAPI->Zoom(aZoom); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } } already_AddRefed nsBrowserElement::GetCanGoBack(ErrorResult& aRv) { NS_ENSURE_TRUE(IsBrowserElementOrThrow(aRv), nullptr); NS_ENSURE_TRUE(IsNotWidgetOrThrow(aRv), nullptr); nsCOMPtr req; nsresult rv = mBrowserElementAPI->GetCanGoBack(getter_AddRefs(req)); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); return nullptr; } return req.forget().downcast(); } already_AddRefed nsBrowserElement::GetCanGoForward(ErrorResult& aRv) { NS_ENSURE_TRUE(IsBrowserElementOrThrow(aRv), nullptr); NS_ENSURE_TRUE(IsNotWidgetOrThrow(aRv), nullptr); nsCOMPtr req; nsresult rv = mBrowserElementAPI->GetCanGoForward(getter_AddRefs(req)); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); return nullptr; } return req.forget().downcast(); } already_AddRefed nsBrowserElement::GetContentDimensions(ErrorResult& aRv) { NS_ENSURE_TRUE(IsBrowserElementOrThrow(aRv), nullptr); NS_ENSURE_TRUE(IsNotWidgetOrThrow(aRv), nullptr); nsCOMPtr req; nsresult rv = mBrowserElementAPI->GetContentDimensions(getter_AddRefs(req)); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); return nullptr; } return req.forget().downcast(); } void nsBrowserElement::AddNextPaintListener(BrowserElementNextPaintEventCallback& aListener, ErrorResult& aRv) { NS_ENSURE_TRUE_VOID(IsBrowserElementOrThrow(aRv)); CallbackObjectHolder holder(&aListener); nsCOMPtr listener = holder.ToXPCOMCallback(); nsresult rv = mBrowserElementAPI->AddNextPaintListener(listener); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } } void nsBrowserElement::RemoveNextPaintListener(BrowserElementNextPaintEventCallback& aListener, ErrorResult& aRv) { NS_ENSURE_TRUE_VOID(IsBrowserElementOrThrow(aRv)); CallbackObjectHolder holder(&aListener); nsCOMPtr listener = holder.ToXPCOMCallback(); nsresult rv = mBrowserElementAPI->RemoveNextPaintListener(listener); if (NS_WARN_IF(NS_FAILED(rv))) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } } already_AddRefed nsBrowserElement::SetInputMethodActive(bool aIsActive, ErrorResult& aRv) { NS_ENSURE_TRUE(IsBrowserElementOrThrow(aRv), nullptr); nsCOMPtr frameLoader = GetFrameLoader(); if (!frameLoader) { aRv.Throw(NS_ERROR_OUT_OF_MEMORY); return nullptr; } nsCOMPtr ownerElement; nsresult rv = frameLoader->GetOwnerElement(getter_AddRefs(ownerElement)); if (NS_FAILED(rv)) { aRv.Throw(rv); return nullptr; } nsCOMPtr node = do_QueryInterface(ownerElement); nsCOMPtr principal = node->NodePrincipal(); if (!nsContentUtils::IsExactSitePermAllow(principal, "input-manage")) { aRv.Throw(NS_ERROR_DOM_INVALID_ACCESS_ERR); return nullptr; } nsCOMPtr req; rv = mBrowserElementAPI->SetInputMethodActive(aIsActive, getter_AddRefs(req)); if (NS_WARN_IF(NS_FAILED(rv))) { if (rv == NS_ERROR_INVALID_ARG) { aRv.Throw(NS_ERROR_DOM_INVALID_ACCESS_ERR); } else { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); } return nullptr; } return req.forget().downcast(); } } // namespace mozilla