diff options
Diffstat (limited to 'layout/base/nsDocumentViewer.cpp')
-rw-r--r-- | layout/base/nsDocumentViewer.cpp | 3514 |
1 files changed, 3514 insertions, 0 deletions
diff --git a/layout/base/nsDocumentViewer.cpp b/layout/base/nsDocumentViewer.cpp new file mode 100644 index 0000000000..ab61daae87 --- /dev/null +++ b/layout/base/nsDocumentViewer.cpp @@ -0,0 +1,3514 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set ts=8 sts=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/. */ + +/* container for a document and its presentation */ + +#include "gfxContext.h" +#include "mozilla/PresShell.h" +#include "mozilla/RestyleManager.h" +#include "mozilla/ServoStyleSet.h" +#include "mozilla/StaticPrefs_print.h" +#include "mozilla/Telemetry.h" +#include "nsThreadUtils.h" +#include "nscore.h" +#include "nsCOMPtr.h" +#include "nsCRT.h" +#include "nsFrameSelection.h" +#include "nsString.h" +#include "nsReadableUtils.h" +#include "nsIContent.h" +#include "nsIDocumentViewer.h" +#include "nsIDocumentViewerPrint.h" +#include "nsIScreen.h" +#include "mozilla/dom/AutoSuppressEventHandlingAndSuspend.h" +#include "mozilla/dom/BrowsingContext.h" +#include "mozilla/dom/BeforeUnloadEvent.h" +#include "mozilla/dom/PopupBlocker.h" +#include "mozilla/dom/Document.h" +#include "mozilla/dom/DocumentInlines.h" +#include "mozilla/dom/DocGroup.h" +#include "mozilla/widget/Screen.h" +#include "nsPresContext.h" +#include "nsIFrame.h" +#include "nsIWritablePropertyBag2.h" +#include "nsSubDocumentFrame.h" +#include "nsGenericHTMLElement.h" +#include "nsStubMutationObserver.h" + +#include "nsISelectionListener.h" +#include "mozilla/dom/Selection.h" +#include "nsContentUtils.h" +#ifdef ACCESSIBILITY +# include "mozilla/a11y/DocAccessible.h" +#endif +#include "mozilla/BasicEvents.h" +#include "mozilla/Encoding.h" +#include "mozilla/ErrorResult.h" +#include "mozilla/Preferences.h" +#include "mozilla/SpinEventLoopUntil.h" +#include "mozilla/WeakPtr.h" +#include "mozilla/StaticPrefs_dom.h" +#include "mozilla/StaticPrefs_javascript.h" +#include "mozilla/StaticPrefs_fission.h" +#include "mozilla/StaticPrefs_print.h" +#include "mozilla/StyleSheet.h" +#include "mozilla/StyleSheetInlines.h" +#include "mozilla/Try.h" + +#include "nsViewManager.h" +#include "nsView.h" + +#include "nsPageSequenceFrame.h" +#include "nsNetUtil.h" +#include "nsIDocumentViewerEdit.h" +#include "mozilla/css/Loader.h" +#include "nsIInterfaceRequestor.h" +#include "nsIInterfaceRequestorUtils.h" +#include "nsDocShell.h" +#include "nsIBaseWindow.h" +#include "nsILayoutHistoryState.h" +#include "nsCharsetSource.h" +#include "mozilla/ReflowInput.h" +#include "nsIImageLoadingContent.h" +#include "nsCopySupport.h" +#include "nsXULPopupManager.h" + +#include "nsIClipboardHelper.h" + +#include "nsPIDOMWindow.h" +#include "nsGlobalWindowInner.h" +#include "nsGlobalWindowOuter.h" +#include "nsDOMNavigationTiming.h" +#include "nsPIWindowRoot.h" +#include "nsJSEnvironment.h" +#include "nsFocusManager.h" + +#include "nsIScrollableFrame.h" +#include "nsStyleSheetService.h" +#include "nsILoadContext.h" +#include "mozilla/ThrottledEventQueue.h" +#include "nsIPromptCollection.h" +#include "nsIPromptService.h" +#include "imgIContainer.h" // image animation mode constants +#include "nsIXULRuntime.h" +#include "nsSandboxFlags.h" + +//-------------------------- +// Printing Include +//--------------------------- +#ifdef NS_PRINTING + +# include "nsIWebBrowserPrint.h" + +# include "nsPrintJob.h" +# include "nsDeviceContextSpecProxy.h" + +// Print Options +# include "nsIPrintSettings.h" +# include "nsIPrintSettingsService.h" +# include "nsISimpleEnumerator.h" + +#endif // NS_PRINTING + +// focus +#include "nsIDOMEventListener.h" +#include "nsISelectionController.h" + +#include "mozilla/EventDispatcher.h" +#include "nsISHEntry.h" +#include "nsISHistory.h" +#include "nsIWebNavigation.h" +#include "mozilla/dom/XMLHttpRequestMainThread.h" + +// paint forcing +#include <stdio.h> +#include "mozilla/BasePrincipal.h" +#include "mozilla/dom/Element.h" +#include "mozilla/dom/Event.h" +#include "mozilla/Telemetry.h" +#include "mozilla/dom/ScriptLoader.h" +#include "mozilla/dom/WindowGlobalChild.h" + +namespace mozilla { +namespace dom { +class PrintPreviewResultInfo; +} // namespace dom +} // namespace mozilla + +using namespace mozilla; +using namespace mozilla::dom; + +using mozilla::layout::RemotePrintJobChild; +using PrintPreviewResolver = + std::function<void(const mozilla::dom::PrintPreviewResultInfo&)>; + +//----------------------------------------------------- +// LOGGING +#include "LayoutLogging.h" +#include "mozilla/Logging.h" + +extern mozilla::LazyLogModule gPageCacheLog; + +#ifdef NS_PRINTING +mozilla::LazyLogModule gPrintingLog("printing"); + +# define PR_PL(_p1) MOZ_LOG(gPrintingLog, mozilla::LogLevel::Debug, _p1); +#endif // NS_PRINTING + +#define PRT_YESNO(_p) ((_p) ? "YES" : "NO") +//----------------------------------------------------- + +class nsDocumentViewer; + +// a small delegate class used to avoid circular references + +class nsDocViewerSelectionListener final : public nsISelectionListener { + public: + // nsISupports interface... + NS_DECL_ISUPPORTS + + // nsISelectionListerner interface + NS_DECL_NSISELECTIONLISTENER + + explicit nsDocViewerSelectionListener(nsDocumentViewer* aDocViewer) + : mDocViewer(aDocViewer), mSelectionWasCollapsed(true) {} + + void Disconnect() { mDocViewer = nullptr; } + + protected: + virtual ~nsDocViewerSelectionListener() = default; + + nsDocumentViewer* mDocViewer; + bool mSelectionWasCollapsed; +}; + +/** editor Implementation of the FocusListener interface */ +class nsDocViewerFocusListener final : public nsIDOMEventListener { + public: + explicit nsDocViewerFocusListener(nsDocumentViewer* aDocViewer) + : mDocViewer(aDocViewer) {} + + NS_DECL_ISUPPORTS + NS_DECL_NSIDOMEVENTLISTENER + + void Disconnect() { mDocViewer = nullptr; } + + protected: + virtual ~nsDocViewerFocusListener() = default; + + nsDocumentViewer* mDocViewer; +}; + +namespace viewer_detail { + +/** + * Mutation observer for use until we hand ourselves over to our SHEntry. + */ +class BFCachePreventionObserver final : public nsStubMutationObserver { + public: + explicit BFCachePreventionObserver(Document* aDocument) + : mDocument(aDocument) {} + + NS_DECL_ISUPPORTS + + NS_DECL_NSIMUTATIONOBSERVER_CHARACTERDATACHANGED + NS_DECL_NSIMUTATIONOBSERVER_ATTRIBUTECHANGED + NS_DECL_NSIMUTATIONOBSERVER_CONTENTAPPENDED + NS_DECL_NSIMUTATIONOBSERVER_CONTENTINSERTED + NS_DECL_NSIMUTATIONOBSERVER_CONTENTREMOVED + NS_DECL_NSIMUTATIONOBSERVER_NODEWILLBEDESTROYED + + // Stop observing the document. + void Disconnect(); + + private: + ~BFCachePreventionObserver() = default; + + // Helper for the work that needs to happen when mutations happen. + void MutationHappened(); + + Document* mDocument; // Weak; we get notified if it dies +}; + +NS_IMPL_ISUPPORTS(BFCachePreventionObserver, nsIMutationObserver) + +void BFCachePreventionObserver::CharacterDataChanged( + nsIContent* aContent, const CharacterDataChangeInfo&) { + if (aContent->IsInNativeAnonymousSubtree()) { + return; + } + MutationHappened(); +} + +void BFCachePreventionObserver::AttributeChanged(Element* aElement, + int32_t aNameSpaceID, + nsAtom* aAttribute, + int32_t aModType, + const nsAttrValue* aOldValue) { + if (aElement->IsInNativeAnonymousSubtree()) { + return; + } + MutationHappened(); +} + +void BFCachePreventionObserver::ContentAppended(nsIContent* aFirstNewContent) { + if (aFirstNewContent->IsInNativeAnonymousSubtree()) { + return; + } + MutationHappened(); +} + +void BFCachePreventionObserver::ContentInserted(nsIContent* aChild) { + if (aChild->IsInNativeAnonymousSubtree()) { + return; + } + MutationHappened(); +} + +void BFCachePreventionObserver::ContentRemoved(nsIContent* aChild, + nsIContent* aPreviousSibling) { + if (aChild->IsInNativeAnonymousSubtree()) { + return; + } + MutationHappened(); +} + +void BFCachePreventionObserver::NodeWillBeDestroyed(nsINode* aNode) { + mDocument = nullptr; +} + +void BFCachePreventionObserver::Disconnect() { + if (mDocument) { + mDocument->RemoveMutationObserver(this); + // It will no longer tell us when it goes away, so make sure we're + // not holding a dangling ref. + mDocument = nullptr; + } +} + +void BFCachePreventionObserver::MutationHappened() { + MOZ_ASSERT( + mDocument, + "How can we not have a document but be getting notified for mutations?"); + mDocument->DisallowBFCaching(); + Disconnect(); +} + +} // namespace viewer_detail + +using viewer_detail::BFCachePreventionObserver; + +//------------------------------------------------------------- +class nsDocumentViewer final : public nsIDocumentViewer, + public nsIDocumentViewerEdit, + public nsIDocumentViewerPrint +#ifdef NS_PRINTING + , + public nsIWebBrowserPrint +#endif + +{ + friend class nsDocViewerSelectionListener; + friend class nsPagePrintTimer; + friend class nsPrintJob; + + public: + nsDocumentViewer(); + + // nsISupports interface... + NS_DECL_ISUPPORTS + + // nsIDocumentViewer interface... + NS_DECL_NSIDOCUMENTVIEWER + + // nsIDocumentViewerEdit + NS_DECL_NSIDOCUMENTVIEWEREDIT + +#ifdef NS_PRINTING + // nsIWebBrowserPrint + NS_DECL_NSIWEBBROWSERPRINT +#endif + + // nsIDocumentViewerPrint Printing Methods + NS_DECL_NSIDOCUMENTVIEWERPRINT + + protected: + virtual ~nsDocumentViewer(); + + private: + /** + * Creates a view manager, root view, and widget for the root view, setting + * mViewManager and mWindow. + * @param aSize the initial size in appunits + * @param aContainerView the container view to hook our root view up + * to as a child, or null if this will be the root view manager + */ + nsresult MakeWindow(const nsSize& aSize, nsView* aContainerView); + + /** + * Create our device context + */ + nsresult CreateDeviceContext(nsView* aContainerView); + + /** + * If aDoCreation is true, this creates the device context, creates a + * prescontext if necessary, and calls MakeWindow. + * + * If aForceSetNewDocument is false, then SetNewDocument won't be + * called if the window's current document is already mDocument. + */ + nsresult InitInternal(nsIWidget* aParentWidget, nsISupports* aState, + mozilla::dom::WindowGlobalChild* aActor, + const nsIntRect& aBounds, bool aDoCreation, + bool aNeedMakeCX = true, + bool aForceSetNewDocument = true); + /** + * @param aDoInitialReflow set to true if you want to kick off the initial + * reflow + */ + MOZ_CAN_RUN_SCRIPT_BOUNDARY + nsresult InitPresentationStuff(bool aDoInitialReflow); + + already_AddRefed<nsINode> GetPopupNode(); + already_AddRefed<nsINode> GetPopupLinkNode(); + already_AddRefed<nsIImageLoadingContent> GetPopupImageNode(); + + void PrepareToStartLoad(void); + + nsresult SyncParentSubDocMap(); + + void RemoveFocusListener(); + void ReinitializeFocusListener(); + + mozilla::dom::Selection* GetDocumentSelection(); + + void DestroyPresShell(); + void DestroyPresContext(); + + void InvalidatePotentialSubDocDisplayItem(); + + // Whether we should attach to the top level widget. This is true if we + // are sharing/recycling a single base widget and not creating multiple + // child widgets. + bool ShouldAttachToTopLevel(); + + std::tuple<const nsIFrame*, int32_t> GetCurrentSheetFrameAndNumber() const; + + protected: + // Returns the current viewmanager. Might be null. + nsViewManager* GetViewManager(); + + void DetachFromTopLevelWidget(); + + // IMPORTANT: The ownership implicit in the following member + // variables has been explicitly checked and set using nsCOMPtr + // for owning pointers and raw COM interface pointers for weak + // (ie, non owning) references. If you add any members to this + // class, please make the ownership explicit (pinkerton, scc). + + WeakPtr<nsDocShell> mContainer; // it owns me! + RefPtr<nsDeviceContext> mDeviceContext; // We create and own this baby + + // the following six items are explicitly in this order + // so they will be destroyed in the reverse order (pinkerton, scc) + nsCOMPtr<Document> mDocument; + nsCOMPtr<nsIWidget> mWindow; // may be null + RefPtr<nsViewManager> mViewManager; + RefPtr<nsPresContext> mPresContext; + RefPtr<PresShell> mPresShell; + + RefPtr<nsDocViewerSelectionListener> mSelectionListener; + RefPtr<nsDocViewerFocusListener> mFocusListener; + + nsCOMPtr<nsIDocumentViewer> mPreviousViewer; + nsCOMPtr<nsISHEntry> mSHEntry; + // Observer that will prevent bfcaching if it gets notified. This + // is non-null precisely when mSHEntry is non-null. + RefPtr<BFCachePreventionObserver> mBFCachePreventionObserver; + + nsIWidget* mParentWidget; // purposely won't be ref counted. May be null + bool mAttachedToParent; // view is attached to the parent widget + + nsIntRect mBounds; + + int16_t mNumURLStarts; + int16_t mDestroyBlockedCount; + + unsigned mStopped : 1; + unsigned mLoaded : 1; + unsigned mDeferredWindowClose : 1; + // document management data + // these items are specific to markup documents (html and xml) + // may consider splitting these out into a subclass + unsigned mIsSticky : 1; + unsigned mInPermitUnload : 1; + unsigned mInPermitUnloadPrompt : 1; + +#ifdef NS_PRINTING + unsigned mClosingWhilePrinting : 1; + +# if NS_PRINT_PREVIEW + RefPtr<nsPrintJob> mPrintJob; +# endif // NS_PRINT_PREVIEW + +#endif // NS_PRINTING + + /* character set member data */ + int32_t mReloadEncodingSource; + const Encoding* mReloadEncoding; + + bool mIsPageMode; + bool mInitializedForPrintPreview; + bool mHidden; +}; + +class nsDocumentShownDispatcher : public Runnable { + public: + explicit nsDocumentShownDispatcher(nsCOMPtr<Document> aDocument) + : Runnable("nsDocumentShownDispatcher"), mDocument(aDocument) {} + + NS_IMETHOD Run() override; + + private: + nsCOMPtr<Document> mDocument; +}; + +//------------------------------------------------------------------ +// nsDocumentViewer +//------------------------------------------------------------------ + +//------------------------------------------------------------------ +already_AddRefed<nsIDocumentViewer> NS_NewDocumentViewer() { + RefPtr<nsDocumentViewer> viewer = new nsDocumentViewer(); + return viewer.forget(); +} + +void nsDocumentViewer::PrepareToStartLoad() { + MOZ_DIAGNOSTIC_ASSERT(!GetIsPrintPreview(), + "Print preview tab should never navigate"); + + mStopped = false; + mLoaded = false; + mAttachedToParent = false; + mDeferredWindowClose = false; + +#ifdef NS_PRINTING + mClosingWhilePrinting = false; + + // Make sure we have destroyed it and cleared the data member + if (mPrintJob) { + mPrintJob->Destroy(); + mPrintJob = nullptr; + } + +#endif // NS_PRINTING +} + +nsDocumentViewer::nsDocumentViewer() + : mParentWidget(nullptr), + mAttachedToParent(false), + mNumURLStarts(0), + mDestroyBlockedCount(0), + mStopped(false), + mLoaded(false), + mDeferredWindowClose(false), + mIsSticky(true), + mInPermitUnload(false), + mInPermitUnloadPrompt(false), +#ifdef NS_PRINTING + mClosingWhilePrinting(false), +#endif // NS_PRINTING + mReloadEncodingSource(kCharsetUninitialized), + mReloadEncoding(nullptr), + mIsPageMode(false), + mInitializedForPrintPreview(false), + mHidden(false) { + PrepareToStartLoad(); +} + +NS_IMPL_ADDREF(nsDocumentViewer) +NS_IMPL_RELEASE(nsDocumentViewer) + +NS_INTERFACE_MAP_BEGIN(nsDocumentViewer) + NS_INTERFACE_MAP_ENTRY(nsIDocumentViewer) + NS_INTERFACE_MAP_ENTRY(nsIDocumentViewerEdit) + NS_INTERFACE_MAP_ENTRY(nsIDocumentViewerPrint) + NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIDocumentViewer) +#ifdef NS_PRINTING + NS_INTERFACE_MAP_ENTRY(nsIWebBrowserPrint) +#endif +NS_INTERFACE_MAP_END + +nsDocumentViewer::~nsDocumentViewer() { + if (mDocument) { + Close(nullptr); + mDocument->Destroy(); + } + +#ifdef NS_PRINTING + if (mPrintJob) { + mPrintJob->Destroy(); + mPrintJob = nullptr; + } +#endif + + MOZ_RELEASE_ASSERT(mDestroyBlockedCount == 0); + NS_ASSERTION(!mPresShell && !mPresContext, + "User did not call nsIDocumentViewer::Destroy"); + if (mPresShell || mPresContext) { + // Make sure we don't hand out a reference to the content viewer to + // the SHEntry! + mSHEntry = nullptr; + + Destroy(); + } + + if (mSelectionListener) { + mSelectionListener->Disconnect(); + } + + RemoveFocusListener(); + + // XXX(?) Revoke pending invalidate events +} + +/* + * This method is called by the Document Loader once a document has + * been created for a particular data stream... The content viewer + * must cache this document for later use when Init(...) is called. + * + * This method is also called when an out of band document.write() happens. + * In that case, the document passed in is the same as the previous document. + */ +/* virtual */ +void nsDocumentViewer::LoadStart(Document* aDocument) { + MOZ_ASSERT(aDocument); + + if (!mDocument) { + mDocument = aDocument; + } +} + +void nsDocumentViewer::RemoveFocusListener() { + if (RefPtr<nsDocViewerFocusListener> oldListener = + std::move(mFocusListener)) { + oldListener->Disconnect(); + if (mDocument) { + mDocument->RemoveEventListener(u"focus"_ns, oldListener, false); + mDocument->RemoveEventListener(u"blur"_ns, oldListener, false); + } + } +} + +void nsDocumentViewer::ReinitializeFocusListener() { + RemoveFocusListener(); + mFocusListener = new nsDocViewerFocusListener(this); + if (mDocument) { + mDocument->AddEventListener(u"focus"_ns, mFocusListener, false, false); + mDocument->AddEventListener(u"blur"_ns, mFocusListener, false, false); + } +} + +nsresult nsDocumentViewer::SyncParentSubDocMap() { + nsCOMPtr<nsIDocShell> docShell(mContainer); + if (!docShell) { + return NS_OK; + } + + nsCOMPtr<nsPIDOMWindowOuter> pwin(docShell->GetWindow()); + if (!mDocument || !pwin) { + return NS_OK; + } + + nsCOMPtr<Element> element = pwin->GetFrameElementInternal(); + if (!element) { + return NS_OK; + } + + nsCOMPtr<nsIDocShellTreeItem> parent; + docShell->GetInProcessParent(getter_AddRefs(parent)); + + nsCOMPtr<nsPIDOMWindowOuter> parent_win = + parent ? parent->GetWindow() : nullptr; + if (!parent_win) { + return NS_OK; + } + + nsCOMPtr<Document> parent_doc = parent_win->GetDoc(); + if (!parent_doc) { + return NS_OK; + } + + if (mDocument && parent_doc->GetSubDocumentFor(element) != mDocument && + parent_doc->EventHandlingSuppressed()) { + mDocument->SuppressEventHandling(parent_doc->EventHandlingSuppressed()); + } + return parent_doc->SetSubDocumentFor(element, mDocument); +} + +NS_IMETHODIMP +nsDocumentViewer::SetContainer(nsIDocShell* aContainer) { + mContainer = static_cast<nsDocShell*>(aContainer); + + // We're loading a new document into the window where this document + // viewer lives, sync the parent document's frame element -> sub + // document map + + return SyncParentSubDocMap(); +} + +NS_IMETHODIMP +nsDocumentViewer::GetContainer(nsIDocShell** aResult) { + NS_ENSURE_ARG_POINTER(aResult); + + nsCOMPtr<nsIDocShell> container(mContainer); + container.swap(*aResult); + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::Init(nsIWidget* aParentWidget, const nsIntRect& aBounds, + WindowGlobalChild* aActor) { + return InitInternal(aParentWidget, nullptr, aActor, aBounds, true); +} + +nsresult nsDocumentViewer::InitPresentationStuff(bool aDoInitialReflow) { + // We assert this because initializing the pres shell could otherwise cause + // re-entrancy into nsDocumentViewer methods, which might cause a different + // pres shell to be created. Callers of InitPresentationStuff should ensure + // the call is appropriately bounded by an nsAutoScriptBlocker to decide + // when it is safe for these re-entrant calls to be made. + MOZ_ASSERT(!nsContentUtils::IsSafeToRunScript(), + "InitPresentationStuff must only be called when scripts are " + "blocked"); + +#ifdef NS_PRINTING + // When getting printed, either for print or print preview, the print job + // takes care of setting up the presentation of the document. + if (mPrintJob) { + return NS_OK; + } +#endif + + NS_ASSERTION(!mPresShell, "Someone should have destroyed the presshell!"); + + // Now make the shell for the document + nsCOMPtr<Document> doc = mDocument; + RefPtr<nsPresContext> presContext = mPresContext; + RefPtr<nsViewManager> viewManager = mViewManager; + mPresShell = doc->CreatePresShell(presContext, viewManager); + if (!mPresShell) { + return NS_ERROR_FAILURE; + } + + if (aDoInitialReflow) { + // Since Initialize() will create frames for *all* items + // that are currently in the document tree, we need to flush + // any pending notifications to prevent the content sink from + // duplicating layout frames for content it has added to the tree + // but hasn't notified the document about. (Bug 154018) + // + // Note that we are flushing before we add mPresShell as an observer + // to avoid bogus notifications. + mDocument->FlushPendingNotifications(FlushType::ContentAndNotify); + } + + mPresShell->BeginObservingDocument(); + + // Initialize our view manager + + { + int32_t p2a = mPresContext->AppUnitsPerDevPixel(); + MOZ_ASSERT( + p2a == + mPresContext->DeviceContext()->AppUnitsPerDevPixelAtUnitFullZoom()); + + nscoord width = p2a * mBounds.width; + nscoord height = p2a * mBounds.height; + + mViewManager->SetWindowDimensions(width, height); + mPresContext->SetVisibleArea(nsRect(0, 0, width, height)); + // We rely on the default zoom not being initialized until here. + mPresContext->RecomputeBrowsingContextDependentData(); + } + + if (mWindow && mDocument->IsTopLevelContentDocument()) { + // Set initial safe area insets + ScreenIntMargin windowSafeAreaInsets; + LayoutDeviceIntRect windowRect = mWindow->GetScreenBounds(); + nsCOMPtr<nsIScreen> screen = mWindow->GetWidgetScreen(); + if (screen) { + windowSafeAreaInsets = nsContentUtils::GetWindowSafeAreaInsets( + screen, mWindow->GetSafeAreaInsets(), windowRect); + } + + mPresContext->SetSafeAreaInsets(windowSafeAreaInsets); + } + + if (aDoInitialReflow) { + RefPtr<PresShell> presShell = mPresShell; + // Initial reflow + presShell->Initialize(); + } + + // now register ourselves as a selection listener, so that we get + // called when the selection changes in the window + if (!mSelectionListener) { + mSelectionListener = new nsDocViewerSelectionListener(this); + } + + RefPtr<mozilla::dom::Selection> selection = GetDocumentSelection(); + if (!selection) { + return NS_ERROR_FAILURE; + } + + selection->AddSelectionListener(mSelectionListener); + + ReinitializeFocusListener(); + + if (aDoInitialReflow && mDocument) { + nsCOMPtr<Document> document = mDocument; + document->ScrollToRef(); + } + + return NS_OK; +} + +static nsPresContext* CreatePresContext(Document* aDocument, + nsPresContext::nsPresContextType aType, + nsView* aContainerView) { + if (aContainerView) { + return new nsPresContext(aDocument, aType); + } + return new nsRootPresContext(aDocument, aType); +} + +//----------------------------------------------- +// This method can be used to initial the "presentation" +// The aDoCreation indicates whether it should create +// all the new objects or just initialize the existing ones +nsresult nsDocumentViewer::InitInternal( + nsIWidget* aParentWidget, nsISupports* aState, WindowGlobalChild* aActor, + const nsIntRect& aBounds, bool aDoCreation, bool aNeedMakeCX /*= true*/, + bool aForceSetNewDocument /* = true*/) { + // We don't want any scripts to run here. That can cause flushing, + // which can cause reentry into initialization of this document viewer, + // which would be disastrous. + nsAutoScriptBlocker blockScripts; + + mParentWidget = aParentWidget; // not ref counted + mBounds = aBounds; + + nsresult rv = NS_OK; + NS_ENSURE_TRUE(mDocument, NS_ERROR_NULL_POINTER); + + nsView* containerView = FindContainerView(); + + bool makeCX = false; + if (aDoCreation) { + nsresult rv = CreateDeviceContext(containerView); + NS_ENSURE_SUCCESS(rv, rv); + + // XXXbz this is a nasty hack to do with the fact that we create + // presentations both in Init() and in Show()... Ideally we would only do + // it in one place (Show()) and require that callers call init(), open(), + // show() in that order or something. + if (!mPresContext && + (aParentWidget || containerView || mDocument->IsBeingUsedAsImage() || + (mDocument->GetDisplayDocument() && + mDocument->GetDisplayDocument()->GetPresShell()))) { + // Create presentation context + if (mIsPageMode) { + // Presentation context already created in SetPageModeForTesting which + // is calling this method + } else { + mPresContext = CreatePresContext( + mDocument, nsPresContext::eContext_Galley, containerView); + } + NS_ENSURE_TRUE(mPresContext, NS_ERROR_OUT_OF_MEMORY); + + nsresult rv = mPresContext->Init(mDeviceContext); + if (NS_FAILED(rv)) { + mPresContext = nullptr; + return rv; + } + +#if defined(NS_PRINTING) && defined(NS_PRINT_PREVIEW) + makeCX = !GetIsPrintPreview() && + aNeedMakeCX; // needs to be true except when we are already in + // PP or we are enabling/disabling paginated mode. +#else + makeCX = true; +#endif + } + + if (mPresContext) { + // Create the ViewManager and Root View... + + // We must do this before we tell the script global object about + // this new document since doing that will cause us to re-enter + // into nsSubDocumentFrame code through reflows caused by + // FlushPendingNotifications() calls down the road... + + rv = MakeWindow(nsSize(mPresContext->DevPixelsToAppUnits(aBounds.width), + mPresContext->DevPixelsToAppUnits(aBounds.height)), + containerView); + NS_ENSURE_SUCCESS(rv, rv); + Hide(); + +#ifdef NS_PRINT_PREVIEW + if (mIsPageMode) { + // I'm leaving this in a broken state for the moment; we should + // be measuring/scaling with the print device context, not the + // screen device context, but this is good enough to allow + // printing reftests to work. + double pageWidth = 0, pageHeight = 0; + mPresContext->GetPrintSettings()->GetEffectivePageSize(&pageWidth, + &pageHeight); + mPresContext->SetPageSize( + nsSize(mPresContext->CSSTwipsToAppUnits(NSToIntFloor(pageWidth)), + mPresContext->CSSTwipsToAppUnits(NSToIntFloor(pageHeight)))); + mPresContext->SetIsRootPaginatedDocument(true); + mPresContext->SetPageScale(1.0f); + } +#endif + } else { + // Avoid leaking the old viewer. + if (mPreviousViewer) { + mPreviousViewer->Destroy(); + mPreviousViewer = nullptr; + } + } + } + + nsCOMPtr<nsIInterfaceRequestor> requestor(mContainer); + if (requestor) { + // Set script-context-owner in the document + + nsCOMPtr<nsPIDOMWindowOuter> window = do_GetInterface(requestor); + + if (window) { + nsCOMPtr<Document> curDoc = window->GetExtantDoc(); + if (aForceSetNewDocument || curDoc != mDocument) { + rv = window->SetNewDocument(mDocument, aState, false, aActor); + if (NS_FAILED(rv)) { + Destroy(); + return rv; + } + } + } + } + + if (aDoCreation && mPresContext) { + // The ViewManager and Root View was created above (in + // MakeWindow())... + + rv = InitPresentationStuff(!makeCX); + } + + return rv; +} + +void nsDocumentViewer::SetNavigationTiming(nsDOMNavigationTiming* timing) { + NS_ASSERTION(mDocument, "Must have a document to set navigation timing."); + if (mDocument) { + mDocument->SetNavigationTiming(timing); + } +} + +// +// LoadComplete(aStatus) +// +// aStatus - The status returned from loading the document. +// +// This method is called by the container when the document has been +// completely loaded. +// +NS_IMETHODIMP +nsDocumentViewer::LoadComplete(nsresult aStatus) { + /* We need to protect ourself against auto-destruction in case the + window is closed while processing the OnLoad event. See bug + http://bugzilla.mozilla.org/show_bug.cgi?id=78445 for more + explanation. + */ + RefPtr<nsDocumentViewer> kungFuDeathGrip(this); + + // Flush out layout so it's up-to-date by the time onload is called. + // Note that this could destroy the window, so do this before + // checking for our mDocument and its window. + if (mPresShell && !mStopped) { + // Hold strong ref because this could conceivably run script + RefPtr<PresShell> presShell = mPresShell; + presShell->FlushPendingNotifications(FlushType::Layout); + } + + nsresult rv = NS_OK; + NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); + + // First, get the window from the document... + nsCOMPtr<nsPIDOMWindowOuter> window = mDocument->GetWindow(); + + mLoaded = true; + + // Now, fire either an OnLoad or OnError event to the document... + bool restoring = false; + // XXXbz imagelib kills off the document load for a full-page image with + // NS_ERROR_PARSED_DATA_CACHED if it's in the cache. So we want to treat + // that one as a success code; otherwise whether we fire onload for the image + // will depend on whether it's cached! + if (window && + (NS_SUCCEEDED(aStatus) || aStatus == NS_ERROR_PARSED_DATA_CACHED)) { + // If this code changes, the code in nsDocLoader::DocLoaderIsEmpty + // that fires load events for document.open() cases might need to + // be updated too. + nsEventStatus status = nsEventStatus_eIgnore; + WidgetEvent event(true, eLoad); + event.mFlags.mBubbles = false; + event.mFlags.mCancelable = false; + // XXX Dispatching to |window|, but using |document| as the target. + event.mTarget = mDocument; + + // If the document presentation is being restored, we don't want to fire + // onload to the document content since that would likely confuse scripts + // on the page. + + RefPtr<nsDocShell> docShell = nsDocShell::Cast(window->GetDocShell()); + NS_ENSURE_TRUE(docShell, NS_ERROR_UNEXPECTED); + + // Unfortunately, docShell->GetRestoringDocument() might no longer be set + // correctly. In particular, it can be false by now if someone took it upon + // themselves to block onload from inside restoration and unblock it later. + // But we can detect the restoring case very simply: by whether our + // document's readyState is COMPLETE. + restoring = + (mDocument->GetReadyStateEnum() == Document::READYSTATE_COMPLETE); + if (!restoring) { + NS_ASSERTION( + mDocument->GetReadyStateEnum() == Document::READYSTATE_INTERACTIVE || + // test_stricttransportsecurity.html has old-style + // docshell-generated about:blank docs reach this code! + (mDocument->GetReadyStateEnum() == + Document::READYSTATE_UNINITIALIZED && + NS_IsAboutBlank(mDocument->GetDocumentURI())), + "Bad readystate"); +#ifdef DEBUG + bool docShellThinksWeAreRestoring; + docShell->GetRestoringDocument(&docShellThinksWeAreRestoring); + MOZ_ASSERT(!docShellThinksWeAreRestoring, + "How can docshell think we are restoring if we don't have a " + "READYSTATE_COMPLETE document?"); +#endif // DEBUG + nsCOMPtr<Document> d = mDocument; + mDocument->SetReadyStateInternal(Document::READYSTATE_COMPLETE); + + RefPtr<nsDOMNavigationTiming> timing(d->GetNavigationTiming()); + if (timing) { + timing->NotifyLoadEventStart(); + } + + // Dispatch observer notification to notify observers document load is + // complete. + nsCOMPtr<nsIObserverService> os = mozilla::services::GetObserverService(); + if (os) { + nsIPrincipal* principal = d->NodePrincipal(); + os->NotifyObservers(ToSupports(d), + principal->IsSystemPrincipal() + ? "chrome-document-loaded" + : "content-document-loaded", + nullptr); + } + + nsPIDOMWindowInner* innerWindow = window->GetCurrentInnerWindow(); + d->SetLoadEventFiring(true); + RefPtr<nsPresContext> presContext = mPresContext; + // MOZ_KnownLive due to bug 1506441 + EventDispatcher::Dispatch( + MOZ_KnownLive(nsGlobalWindowOuter::Cast(window)), presContext, &event, + nullptr, &status); + d->SetLoadEventFiring(false); + + if (timing) { + timing->NotifyLoadEventEnd(); + } + + if (innerWindow) { + innerWindow->QueuePerformanceNavigationTiming(); + } + } + } else { + // XXX: Should fire error event to the document... + + // If our load was explicitly aborted, then we want to set our + // readyState to COMPLETE, and fire a readystatechange event. + if (aStatus == NS_BINDING_ABORTED && mDocument) { + mDocument->NotifyAbortedLoad(); + } + } + + // Notify the document that it has been shown (regardless of whether + // it was just loaded). Note: mDocument may be null now if the above + // firing of onload caused the document to unload. Or, mDocument may not be + // the "current active" document, if the above firing of onload caused our + // docshell to navigate away. NOTE: In this latter scenario, it's likely that + // we fired pagehide (when navigating away) without ever having fired + // pageshow, and that's pretty broken... Fortunately, this should be rare. + // (It requires us to spin the event loop in onload handler, e.g. via sync + // XHR, in order for the navigation-away to happen before onload completes.) + // We skip firing pageshow if we're currently handling unload, or if loading + // was explicitly aborted. + if (mDocument && mDocument->IsCurrentActiveDocument() && + aStatus != NS_BINDING_ABORTED) { + // Re-get window, since it might have changed during above firing of onload + window = mDocument->GetWindow(); + if (window) { + nsIDocShell* docShell = window->GetDocShell(); + bool isInUnload; + if (docShell && NS_SUCCEEDED(docShell->GetIsInUnload(&isInUnload)) && + !isInUnload) { + mDocument->OnPageShow(restoring, nullptr); + } + } + } + + if (!mStopped) { + if (mDocument) { + nsCOMPtr<Document> document = mDocument; + document->ScrollToRef(); + } + + // Now that the document has loaded, we can tell the presshell + // to unsuppress painting. + if (mPresShell) { + RefPtr<PresShell> presShell = mPresShell; + presShell->UnsuppressPainting(); + // mPresShell could have been removed now, see bug 378682/421432 + if (mPresShell) { + mPresShell->LoadComplete(); + } + } + } + + if (mDocument && !restoring) { + mDocument->LoadEventFired(); + } + + // It's probably a good idea to GC soon since we have finished loading. + nsJSContext::PokeGC( + JS::GCReason::LOAD_END, + mDocument ? mDocument->GetWrapperPreserveColor() : nullptr); + +#ifdef NS_PRINTING + // Check to see if someone tried to print during the load + if (window) { + auto* outerWin = nsGlobalWindowOuter::Cast(window); + outerWin->StopDelayingPrintingUntilAfterLoad(); + if (outerWin->DelayedPrintUntilAfterLoad()) { + // We call into the inner because it ensures there's an active document + // and such, and it also waits until the whole thing completes, which is + // nice because it allows us to close if needed right here. + if (RefPtr inner = + nsGlobalWindowInner::Cast(window->GetCurrentInnerWindow())) { + inner->Print(IgnoreErrors()); + } + if (outerWin->DelayedCloseForPrinting()) { + outerWin->Close(); + } + } else { + MOZ_ASSERT(!outerWin->DelayedCloseForPrinting()); + } + } +#endif + + return rv; +} + +bool nsDocumentViewer::GetLoadCompleted() { return mLoaded; } + +bool nsDocumentViewer::GetIsStopped() { return mStopped; } + +NS_IMETHODIMP +nsDocumentViewer::PermitUnload(PermitUnloadAction aAction, + bool* aPermitUnload) { + // We're going to be running JS and nested event loops, which could cause our + // DocShell to be destroyed. Make sure we stay alive until the end of the + // function. + RefPtr<nsDocumentViewer> kungFuDeathGrip(this); + + if (StaticPrefs::dom_disable_beforeunload()) { + aAction = eDontPromptAndUnload; + } + + *aPermitUnload = true; + + RefPtr<BrowsingContext> bc = mContainer->GetBrowsingContext(); + if (!bc) { + return NS_OK; + } + + // Per spec, we need to increase the ignore-opens-during-unload counter while + // dispatching the "beforeunload" event on both the document we're currently + // dispatching the event to and the document that we explicitly asked to + // unload. + IgnoreOpensDuringUnload ignoreOpens(mDocument); + + bool foundBlocker = false; + bool foundOOPListener = false; + bc->PreOrderWalk([&](BrowsingContext* aBC) { + if (!aBC->IsInProcess()) { + WindowContext* wc = aBC->GetCurrentWindowContext(); + if (wc && wc->HasBeforeUnload()) { + foundOOPListener = true; + } + } else if (aBC->GetDocShell()) { + nsCOMPtr<nsIDocumentViewer> viewer(aBC->GetDocShell()->GetDocViewer()); + if (viewer && viewer->DispatchBeforeUnload() == eRequestBlockNavigation) { + foundBlocker = true; + } + } + }); + + if (!foundOOPListener) { + if (!foundBlocker) { + return NS_OK; + } + if (aAction != ePrompt) { + *aPermitUnload = aAction == eDontPromptAndUnload; + return NS_OK; + } + } + + // NB: we nullcheck mDocument because it might now be dead as a result of + // the event being dispatched. + RefPtr<WindowGlobalChild> wgc(mDocument ? mDocument->GetWindowGlobalChild() + : nullptr); + if (!wgc) { + return NS_OK; + } + + nsAutoSyncOperation sync(mDocument, SyncOperationBehavior::eSuspendInput); + AutoSuppressEventHandlingAndSuspend seh(bc->Group()); + + mInPermitUnloadPrompt = true; + + bool done = false; + wgc->SendCheckPermitUnload( + foundBlocker, aAction, + [&](bool aPermit) { + done = true; + *aPermitUnload = aPermit; + }, + [&](auto) { + // If the prompt aborted, we tell our consumer that it is not allowed + // to unload the page. One reason that prompts abort is that the user + // performed some action that caused the page to unload while our prompt + // was active. In those cases we don't want our consumer to also unload + // the page. + // + // XXX: Are there other cases where prompts can abort? Is it ok to + // prevent unloading the page in those cases? + done = true; + *aPermitUnload = false; + }); + + SpinEventLoopUntil("nsDocumentViewer::PermitUnload"_ns, + [&]() { return done; }); + + mInPermitUnloadPrompt = false; + return NS_OK; +} + +MOZ_CAN_RUN_SCRIPT_BOUNDARY PermitUnloadResult +nsDocumentViewer::DispatchBeforeUnload() { + AutoDontWarnAboutSyncXHR disableSyncXHRWarning; + + if (!mDocument || mInPermitUnload || mInPermitUnloadPrompt) { + return eAllowNavigation; + } + + // First, get the script global object from the document... + RefPtr<nsGlobalWindowOuter> window = + nsGlobalWindowOuter::Cast(mDocument->GetWindow()); + if (!window) { + // This is odd, but not fatal + NS_WARNING("window not set for document!"); + return eAllowNavigation; + } + + NS_ASSERTION(nsContentUtils::IsSafeToRunScript(), "This is unsafe"); + + // https://html.spec.whatwg.org/multipage/browsing-the-web.html#prompt-to-unload-a-document + // Create an RAII object on mDocument that will increment the + // should-ignore-opens-during-unload counter on initialization + // and decrement it again when it goes out of score (regardless + // of how we exit this function). + IgnoreOpensDuringUnload ignoreOpens(mDocument); + + // Now, fire an BeforeUnload event to the document and see if it's ok + // to unload... + nsPresContext* presContext = mDocument->GetPresContext(); + RefPtr<BeforeUnloadEvent> event = + new BeforeUnloadEvent(mDocument, presContext, nullptr); + event->InitEvent(u"beforeunload"_ns, false, true); + + // Dispatching to |window|, but using |document| as the target. + event->SetTarget(mDocument); + event->SetTrusted(true); + + // In evil cases we might be destroyed while handling the + // onbeforeunload event, don't let that happen. (see also bug#331040) + RefPtr<nsDocumentViewer> kungFuDeathGrip(this); + + { + // Never permit popups from the beforeunload handler, no matter + // how we get here. + AutoPopupStatePusher popupStatePusher(PopupBlocker::openAbused, true); + + RefPtr<BrowsingContext> bc = mContainer->GetBrowsingContext(); + NS_ASSERTION(bc, "should have a browsing context in document viewer"); + + // Never permit dialogs from the beforeunload handler + nsGlobalWindowOuter::TemporarilyDisableDialogs disableDialogs(bc); + + Document::PageUnloadingEventTimeStamp timestamp(mDocument); + + mInPermitUnload = true; + RefPtr<nsPresContext> presContext = mPresContext; + EventDispatcher::DispatchDOMEvent(window, nullptr, event, presContext, + nullptr); + mInPermitUnload = false; + } + + nsAutoString text; + event->GetReturnValue(text); + + // NB: we nullcheck mDocument because it might now be dead as a result of + // the event being dispatched. + if (window->AreDialogsEnabled() && mDocument && + !(mDocument->GetSandboxFlags() & SANDBOXED_MODALS) && + (!StaticPrefs::dom_require_user_interaction_for_beforeunload() || + mDocument->UserHasInteracted()) && + (event->WidgetEventPtr()->DefaultPrevented() || !text.IsEmpty())) { + return eRequestBlockNavigation; + } + return eAllowNavigation; +} + +NS_IMETHODIMP +nsDocumentViewer::GetBeforeUnloadFiring(bool* aInEvent) { + *aInEvent = mInPermitUnload; + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::GetInPermitUnload(bool* aInEvent) { + *aInEvent = mInPermitUnloadPrompt; + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::PageHide(bool aIsUnload) { + AutoDontWarnAboutSyncXHR disableSyncXHRWarning; + + mHidden = true; + + if (!mDocument) { + return NS_ERROR_NULL_POINTER; + } + + if (aIsUnload) { + // Poke the GC. The window might be collectable garbage now. + nsJSContext::PokeGC(JS::GCReason::PAGE_HIDE, + mDocument->GetWrapperPreserveColor(), + TimeDuration::FromMilliseconds( + StaticPrefs::javascript_options_gc_delay() * 2)); + } + + mDocument->OnPageHide(!aIsUnload, nullptr); + + // inform the window so that the focus state is reset. + NS_ENSURE_STATE(mDocument); + nsPIDOMWindowOuter* window = mDocument->GetWindow(); + if (window) window->PageHidden(); + + if (aIsUnload) { + // if Destroy() was called during OnPageHide(), mDocument is nullptr. + NS_ENSURE_STATE(mDocument); + + // First, get the window from the document... + RefPtr<nsPIDOMWindowOuter> window = mDocument->GetWindow(); + + if (!window) { + // Fail if no window is available... + NS_WARNING("window not set for document!"); + return NS_ERROR_NULL_POINTER; + } + + // https://html.spec.whatwg.org/multipage/browsing-the-web.html#unload-a-document + // Create an RAII object on mDocument that will increment the + // should-ignore-opens-during-unload counter on initialization + // and decrement it again when it goes out of scope. + IgnoreOpensDuringUnload ignoreOpens(mDocument); + + // Now, fire an Unload event to the document... + nsEventStatus status = nsEventStatus_eIgnore; + WidgetEvent event(true, eUnload); + event.mFlags.mBubbles = false; + // XXX Dispatching to |window|, but using |document| as the target. + event.mTarget = mDocument; + + // Never permit popups from the unload handler, no matter how we get + // here. + AutoPopupStatePusher popupStatePusher(PopupBlocker::openAbused, true); + + Document::PageUnloadingEventTimeStamp timestamp(mDocument); + + RefPtr<nsPresContext> presContext = mPresContext; + // MOZ_KnownLive due to bug 1506441 + EventDispatcher::Dispatch(MOZ_KnownLive(nsGlobalWindowOuter::Cast(window)), + presContext, &event, nullptr, &status); + } + + // look for open menupopups and close them after the unload event, in case + // the unload event listeners open any new popups + nsContentUtils::HidePopupsInDocument(mDocument); + + return NS_OK; +} + +static void AttachContainerRecurse(nsIDocShell* aShell) { + nsCOMPtr<nsIDocumentViewer> viewer; + aShell->GetDocViewer(getter_AddRefs(viewer)); + if (viewer) { + viewer->SetIsHidden(false); + Document* doc = viewer->GetDocument(); + if (doc) { + doc->SetContainer(static_cast<nsDocShell*>(aShell)); + } + if (PresShell* presShell = viewer->GetPresShell()) { + presShell->SetForwardingContainer(WeakPtr<nsDocShell>()); + } + } + + // Now recurse through the children + int32_t childCount; + aShell->GetInProcessChildCount(&childCount); + for (int32_t i = 0; i < childCount; ++i) { + nsCOMPtr<nsIDocShellTreeItem> childItem; + aShell->GetInProcessChildAt(i, getter_AddRefs(childItem)); + nsCOMPtr<nsIDocShell> shell = do_QueryInterface(childItem); + AttachContainerRecurse(shell); + } +} + +NS_IMETHODIMP +nsDocumentViewer::Open(nsISupports* aState, nsISHEntry* aSHEntry) { + NS_ENSURE_TRUE(mPresShell, NS_ERROR_NOT_INITIALIZED); + + if (mDocument) { + mDocument->SetContainer(mContainer); + } + + nsresult rv = InitInternal(mParentWidget, aState, nullptr, mBounds, false); + NS_ENSURE_SUCCESS(rv, rv); + + mHidden = false; + + if (mPresShell) mPresShell->SetForwardingContainer(WeakPtr<nsDocShell>()); + + // Rehook the child presentations. The child shells are still in + // session history, so get them from there. + + if (aSHEntry) { + nsCOMPtr<nsIDocShellTreeItem> item; + int32_t itemIndex = 0; + while (NS_SUCCEEDED( + aSHEntry->ChildShellAt(itemIndex++, getter_AddRefs(item))) && + item) { + nsCOMPtr<nsIDocShell> shell = do_QueryInterface(item); + AttachContainerRecurse(shell); + } + } + + SyncParentSubDocMap(); + + ReinitializeFocusListener(); + + // XXX re-enable image animations once that works correctly + + PrepareToStartLoad(); + + // When loading a page from the bfcache with puppet widgets, we do the + // widget attachment here (it is otherwise done in MakeWindow, which is + // called for non-bfcache pages in the history, but not bfcache pages). + // Attachment is necessary, since we get detached when another page + // is browsed to. That is, if we are one page A, then when we go to + // page B, we detach. So page A's view has no widget. If we then go + // back to it, and it is in the bfcache, we will use that view, which + // doesn't have a widget. The attach call here will properly attach us. + if (nsIWidget::UsePuppetWidgets() && mPresContext && + ShouldAttachToTopLevel()) { + // If the old view is already attached to our parent, detach + DetachFromTopLevelWidget(); + + nsViewManager* vm = GetViewManager(); + MOZ_ASSERT(vm, "no view manager"); + nsView* v = vm->GetRootView(); + MOZ_ASSERT(v, "no root view"); + MOZ_ASSERT(mParentWidget, "no mParentWidget to set"); + v->AttachToTopLevelWidget(mParentWidget); + + mAttachedToParent = true; + } + + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::Close(nsISHEntry* aSHEntry) { + // All callers are supposed to call close to break circular + // references. If we do this stuff in the destructor, the + // destructor might never be called (especially if we're being + // used from JS. + + mSHEntry = aSHEntry; + + // Close is also needed to disable scripts during paint suppression, + // since we transfer the existing global object to the new document + // that is loaded. In the future, the global object may become a proxy + // for an object that can be switched in and out so that we don't need + // to disable scripts during paint suppression. + + if (!mDocument) return NS_OK; + + if (mSHEntry) { + if (mBFCachePreventionObserver) { + mBFCachePreventionObserver->Disconnect(); + } + mBFCachePreventionObserver = new BFCachePreventionObserver(mDocument); + mDocument->AddMutationObserver(mBFCachePreventionObserver); + } + +#ifdef NS_PRINTING + // A Close was called while we were printing + // so don't clear the ScriptGlobalObject + // or clear the mDocument below + if (mPrintJob && !mClosingWhilePrinting) { + mClosingWhilePrinting = true; + } else +#endif + { + // out of band cleanup of docshell + mDocument->SetScriptGlobalObject(nullptr); + + if (!mSHEntry && mDocument) mDocument->RemovedFromDocShell(); + } + + RemoveFocusListener(); + return NS_OK; +} + +static void DetachContainerRecurse(nsIDocShell* aShell) { + // Unhook this docshell's presentation + aShell->SynchronizeLayoutHistoryState(); + nsCOMPtr<nsIDocumentViewer> viewer; + aShell->GetDocViewer(getter_AddRefs(viewer)); + if (viewer) { + if (Document* doc = viewer->GetDocument()) { + doc->SetContainer(nullptr); + } + if (PresShell* presShell = viewer->GetPresShell()) { + auto weakShell = static_cast<nsDocShell*>(aShell); + presShell->SetForwardingContainer(weakShell); + } + } + + // Now recurse through the children + int32_t childCount; + aShell->GetInProcessChildCount(&childCount); + for (int32_t i = 0; i < childCount; ++i) { + nsCOMPtr<nsIDocShellTreeItem> childItem; + aShell->GetInProcessChildAt(i, getter_AddRefs(childItem)); + nsCOMPtr<nsIDocShell> shell = do_QueryInterface(childItem); + DetachContainerRecurse(shell); + } +} + +NS_IMETHODIMP +nsDocumentViewer::Destroy() { + // Don't let the document get unloaded while we are printing. + // this could happen if we hit the back button during printing. + // We also keep the viewer from being cached in session history, since + // we require all documents there to be sanitized. + if (mDestroyBlockedCount != 0) { + return NS_OK; + } + +#ifdef NS_PRINTING + // Here is where we check to see if the document was still being prepared + // for printing when it was asked to be destroy from someone externally + // This usually happens if the document is unloaded while the user is in the + // Print Dialog + // + // So we flip the bool to remember that the document is going away + // and we can clean up and abort later after returning from the Print Dialog + if (mPrintJob && mPrintJob->CheckBeforeDestroy()) { + return NS_OK; + } +#endif + + // We want to make sure to disconnect mBFCachePreventionObserver before we + // Sanitize() below. + if (mBFCachePreventionObserver) { + mBFCachePreventionObserver->Disconnect(); + mBFCachePreventionObserver = nullptr; + } + + if (mSHEntry && mDocument && !mDocument->IsBFCachingAllowed()) { + // Just drop the SHEntry now and pretend like we never even tried to bfcache + // this viewer. This should only happen when someone calls + // DisallowBFCaching() after CanSavePresentation() already ran. Ensure that + // the SHEntry has no viewer and its state is synced up. We want to do this + // via a stack reference, in case those calls mess with our members. + MOZ_LOG(gPageCacheLog, LogLevel::Debug, + ("BFCache not allowed, dropping SHEntry")); + nsCOMPtr<nsISHEntry> shEntry = std::move(mSHEntry); + shEntry->SetDocumentViewer(nullptr); + shEntry->SyncPresentationState(); + } + + // If we were told to put ourselves into session history instead of destroy + // the presentation, do that now. + if (mSHEntry) { + if (mPresShell) mPresShell->Freeze(); + + // Make sure the presentation isn't torn down by Hide(). + mSHEntry->SetSticky(mIsSticky); + mIsSticky = true; + + // Remove our root view from the view hierarchy. + if (mPresShell) { + nsViewManager* vm = mPresShell->GetViewManager(); + if (vm) { + nsView* rootView = vm->GetRootView(); + + if (rootView) { + nsView* rootViewParent = rootView->GetParent(); + if (rootViewParent) { + nsView* subdocview = rootViewParent->GetParent(); + if (subdocview) { + nsIFrame* f = subdocview->GetFrame(); + if (f) { + nsSubDocumentFrame* s = do_QueryFrame(f); + if (s) { + s->ClearDisplayItems(); + } + } + } + nsViewManager* parentVM = rootViewParent->GetViewManager(); + if (parentVM) { + parentVM->RemoveChild(rootView); + } + } + } + } + } + + Hide(); + + // This is after Hide() so that the user doesn't see the inputs clear. + if (mDocument) { + mDocument->Sanitize(); + } + + // Reverse ownership. Do this *after* calling sanitize so that sanitize + // doesn't cause mutations that make the SHEntry drop the presentation + + // Grab a reference to mSHEntry before calling into things like + // SyncPresentationState that might mess with our members. + nsCOMPtr<nsISHEntry> shEntry = + std::move(mSHEntry); // we'll need this below + + MOZ_LOG(gPageCacheLog, LogLevel::Debug, + ("Storing content viewer into cache entry")); + shEntry->SetDocumentViewer(this); + + // Always sync the presentation state. That way even if someone screws up + // and shEntry has no window state at this point we'll be ok; we just won't + // cache ourselves. + shEntry->SyncPresentationState(); + // XXX Synchronize layout history state to parent once bfcache is supported + // in session-history-in-parent. + + // Shut down accessibility for the document before we start to tear it down. +#ifdef ACCESSIBILITY + if (mPresShell) { + a11y::DocAccessible* docAcc = mPresShell->GetDocAccessible(); + if (docAcc) { + docAcc->Shutdown(); + } + } +#endif + + // Break the link from the document/presentation to the docshell, so that + // link traversals cannot affect the currently-loaded document. + // When the presentation is restored, Open() and InitInternal() will reset + // these pointers to their original values. + + if (mDocument) { + mDocument->SetContainer(nullptr); + } + if (mPresShell) { + mPresShell->SetForwardingContainer(mContainer); + } + + // Do the same for our children. Note that we need to get the child + // docshells from the SHEntry now; the docshell will have cleared them. + nsCOMPtr<nsIDocShellTreeItem> item; + int32_t itemIndex = 0; + while (NS_SUCCEEDED( + shEntry->ChildShellAt(itemIndex++, getter_AddRefs(item))) && + item) { + nsCOMPtr<nsIDocShell> shell = do_QueryInterface(item); + DetachContainerRecurse(shell); + } + + return NS_OK; + } + + // The document was not put in the bfcache + + // Protect against pres shell destruction running scripts and re-entrantly + // creating a new presentation. + nsAutoScriptBlocker scriptBlocker; + + if (mPresShell) { + DestroyPresShell(); + } + if (mDocument) { + mDocument->Destroy(); + mDocument = nullptr; + } + + // All callers are supposed to call destroy to break circular + // references. If we do this stuff in the destructor, the + // destructor might never be called (especially if we're being + // used from JS. + +#ifdef NS_PRINTING + if (mPrintJob) { + RefPtr<nsPrintJob> printJob = std::move(mPrintJob); +# ifdef NS_PRINT_PREVIEW + if (printJob->CreatedForPrintPreview()) { + printJob->FinishPrintPreview(); + } +# endif + printJob->Destroy(); + MOZ_ASSERT(!mPrintJob, + "mPrintJob shouldn't be recreated while destroying it"); + } +#endif + + // Avoid leaking the old viewer. + if (mPreviousViewer) { + mPreviousViewer->Destroy(); + mPreviousViewer = nullptr; + } + + mDeviceContext = nullptr; + + if (mPresContext) { + DestroyPresContext(); + } + + mWindow = nullptr; + mViewManager = nullptr; + mContainer = WeakPtr<nsDocShell>(); + + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::Stop(void) { + NS_ASSERTION(mDocument, "Stop called too early or too late"); + if (mDocument) { + mDocument->StopDocumentLoad(); + } + + mStopped = true; + + if (!mLoaded && mPresShell) { + // Well, we might as well paint what we have so far. + RefPtr<PresShell> presShell = mPresShell; // bug 378682 + presShell->UnsuppressPainting(); + } + + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::GetDOMDocument(Document** aResult) { + NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); + nsCOMPtr<Document> document = mDocument; + document.forget(aResult); + return NS_OK; +} + +Document* nsDocumentViewer::GetDocument() { return mDocument; } + +nsresult nsDocumentViewer::SetDocument(Document* aDocument) { + // Assumptions: + // + // 1) this document viewer has been initialized with a call to Init(). + // 2) the stylesheets associated with the document have been added + // to the document. + + // XXX Right now, this method assumes that the layout of the current + // document hasn't started yet. More cleanup will probably be + // necessary to make this method work for the case when layout *has* + // occurred for the current document. + // That work can happen when and if it is needed. + + if (!aDocument) return NS_ERROR_NULL_POINTER; + + return SetDocumentInternal(aDocument, false); +} + +NS_IMETHODIMP +nsDocumentViewer::SetDocumentInternal(Document* aDocument, + bool aForceReuseInnerWindow) { + MOZ_ASSERT(aDocument); + + // Set new container + aDocument->SetContainer(mContainer); + + if (mDocument != aDocument) { + if (aForceReuseInnerWindow) { + // Transfer the navigation timing information to the new document, since + // we're keeping the same inner and hence should really have the same + // timing information. + aDocument->SetNavigationTiming(mDocument->GetNavigationTiming()); + } + + if (mDocument && + (mDocument->IsStaticDocument() || aDocument->IsStaticDocument())) { + nsContentUtils::AddScriptRunner(NewRunnableMethod( + "Document::Destroy", mDocument, &Document::Destroy)); + } + + // Clear the list of old child docshells. Child docshells for the new + // document will be constructed as frames are created. + if (!aDocument->IsStaticDocument()) { + nsCOMPtr<nsIDocShell> node(mContainer); + if (node) { + int32_t count; + node->GetInProcessChildCount(&count); + for (int32_t i = 0; i < count; ++i) { + nsCOMPtr<nsIDocShellTreeItem> child; + node->GetInProcessChildAt(0, getter_AddRefs(child)); + node->RemoveChild(child); + } + } + } + + // Replace the old document with the new one. Do this only when + // the new document really is a new document. + mDocument = aDocument; + + // Set the script global object on the new document + nsCOMPtr<nsPIDOMWindowOuter> window = + mContainer ? mContainer->GetWindow() : nullptr; + if (window) { + nsresult rv = + window->SetNewDocument(aDocument, nullptr, aForceReuseInnerWindow); + if (NS_FAILED(rv)) { + Destroy(); + return rv; + } + } + } + + nsresult rv = SyncParentSubDocMap(); + NS_ENSURE_SUCCESS(rv, rv); + + // Replace the current pres shell with a new shell for the new document + + // Protect against pres shell destruction running scripts and re-entrantly + // creating a new presentation. + nsAutoScriptBlocker scriptBlocker; + + if (mPresShell) { + DestroyPresShell(); + } + + if (mPresContext) { + DestroyPresContext(); + + mWindow = nullptr; + rv = InitInternal(mParentWidget, nullptr, nullptr, mBounds, true, true, + false); + } + + return rv; +} + +PresShell* nsDocumentViewer::GetPresShell() { return mPresShell; } + +nsPresContext* nsDocumentViewer::GetPresContext() { return mPresContext; } + +nsViewManager* nsDocumentViewer::GetViewManager() { return mViewManager; } + +NS_IMETHODIMP +nsDocumentViewer::GetBounds(nsIntRect& aResult) { + NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); + aResult = mBounds; + return NS_OK; +} + +nsIDocumentViewer* nsDocumentViewer::GetPreviousViewer() { + return mPreviousViewer; +} + +void nsDocumentViewer::SetPreviousViewer(nsIDocumentViewer* aViewer) { + // NOTE: |Show| sets |mPreviousViewer| to null without calling this + // function. + + if (aViewer) { + NS_ASSERTION(!mPreviousViewer, + "can't set previous viewer when there already is one"); + + // In a multiple chaining situation (which occurs when running a thrashing + // test like i-bench or jrgm's tests with no delay), we can build up a + // whole chain of viewers. In order to avoid this, we always set our + // previous viewer to the MOST previous viewer in the chain, and then dump + // the intermediate link from the chain. This ensures that at most only 2 + // documents are alive and undestroyed at any given time (the one that is + // showing and the one that is loading with painting suppressed). It's very + // important that if this ever gets changed the code before the + // RestorePresentation call in nsDocShell::InternalLoad be changed + // accordingly. + // + // Make sure we hold a strong ref to prevViewer here, since we'll + // tell aViewer to drop it. + nsCOMPtr<nsIDocumentViewer> prevViewer = aViewer->GetPreviousViewer(); + if (prevViewer) { + aViewer->SetPreviousViewer(nullptr); + aViewer->Destroy(); + return SetPreviousViewer(prevViewer); + } + } + + mPreviousViewer = aViewer; +} + +NS_IMETHODIMP +nsDocumentViewer::SetBoundsWithFlags(const nsIntRect& aBounds, + uint32_t aFlags) { + NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); + + bool boundsChanged = !mBounds.IsEqualEdges(aBounds); + mBounds = aBounds; + + if (mWindow && !mAttachedToParent) { + // Resize the widget, but don't trigger repaint. Layout will generate + // repaint requests during reflow. + mWindow->Resize(aBounds.x, aBounds.y, aBounds.width, aBounds.height, false); + } else if (mPresContext && mViewManager) { + // Ensure presContext's deviceContext is up to date, as we sometimes get + // here before a resolution-change notification has been fully handled + // during display configuration changes, especially when there are lots + // of windows/widgets competing to handle the notifications. + // (See bug 1154125.) + if (mPresContext->DeviceContext()->CheckDPIChange()) { + mPresContext->UIResolutionChangedSync(); + } + + int32_t p2a = mPresContext->AppUnitsPerDevPixel(); + nscoord width = NSIntPixelsToAppUnits(mBounds.width, p2a); + nscoord height = NSIntPixelsToAppUnits(mBounds.height, p2a); + nsView* rootView = mViewManager->GetRootView(); + if (boundsChanged && rootView) { + nsRect viewDims = rootView->GetDimensions(); + // If the view/frame tree and prescontext visible area already has the new + // size but we did not, then it's likely that we got reflowed in response + // to a call to GetContentSize. Thus there is a disconnect between the + // size on the document viewer/docshell/containing widget and view + // tree/frame tree/prescontext visible area). SetWindowDimensions compares + // to the root view dimenstions to determine if it needs to do anything; + // if they are the same as the new size it won't do anything, but we still + // need to invalidate because what we want to draw to the screen has + // changed. + if (viewDims.width == width && viewDims.height == height) { + nsIFrame* f = rootView->GetFrame(); + if (f) { + f->InvalidateFrame(); + } + } + } + + mViewManager->SetWindowDimensions( + width, height, !!(aFlags & nsIDocumentViewer::eDelayResize)); + } + + // If there's a previous viewer, it's the one that's actually showing, + // so be sure to resize it as well so it paints over the right area. + // This may slow down the performance of the new page load, but resize + // during load is also probably a relatively unusual condition + // relating to things being hidden while something is loaded. It so + // happens that Firefox does this a good bit with its infobar, and it + // looks ugly if we don't do this. + if (mPreviousViewer) { + nsCOMPtr<nsIDocumentViewer> previousViewer = mPreviousViewer; + previousViewer->SetBounds(aBounds); + } + + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::SetBounds(const nsIntRect& aBounds) { + return SetBoundsWithFlags(aBounds, 0); +} + +NS_IMETHODIMP +nsDocumentViewer::Move(int32_t aX, int32_t aY) { + NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); + mBounds.MoveTo(aX, aY); + if (mWindow) { + mWindow->Move(aX, aY); + } + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::Show() { + NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_AVAILABLE); + + // We don't need the previous viewer anymore since we're not + // displaying it. + if (mPreviousViewer) { + // This little dance *may* only be to keep + // PresShell::EndObservingDocument happy, but I'm not sure. + nsCOMPtr<nsIDocumentViewer> prevViewer(mPreviousViewer); + mPreviousViewer = nullptr; + prevViewer->Destroy(); + + // Make sure we don't have too many cached ContentViewers + nsCOMPtr<nsIDocShellTreeItem> treeItem(mContainer); + if (treeItem) { + // We need to find the root DocShell since only that object has an + // SHistory and we need the SHistory to evict content viewers + nsCOMPtr<nsIDocShellTreeItem> root; + treeItem->GetInProcessSameTypeRootTreeItem(getter_AddRefs(root)); + nsCOMPtr<nsIWebNavigation> webNav = do_QueryInterface(root); + RefPtr<ChildSHistory> history = webNav->GetSessionHistory(); + if (!mozilla::SessionHistoryInParent() && history) { + int32_t prevIndex, loadedIndex; + nsCOMPtr<nsIDocShell> docShell = do_QueryInterface(treeItem); + docShell->GetPreviousEntryIndex(&prevIndex); + docShell->GetLoadedEntryIndex(&loadedIndex); + MOZ_LOG(gPageCacheLog, LogLevel::Verbose, + ("About to evict content viewers: prev=%d, loaded=%d", + prevIndex, loadedIndex)); + history->LegacySHistory()->EvictOutOfRangeDocumentViewers(loadedIndex); + } + } + } + + if (mWindow) { + // When attached to a top level xul window, we do not need to call + // Show on the widget. Underlying window management code handles + // this when the window is initialized. + if (!mAttachedToParent) { + mWindow->Show(true); + } + } + + // Hold on to the document so we can use it after the script blocker below + // has been released (which might re-entrantly call into other + // nsDocumentViewer methods). + nsCOMPtr<Document> document = mDocument; + + if (mDocument && !mPresShell) { + // The InitPresentationStuff call below requires a script blocker, because + // its PresShell::Initialize call can cause scripts to run and therefore + // re-entrant calls to nsDocumentViewer methods to be made. + nsAutoScriptBlocker scriptBlocker; + + NS_ASSERTION(!mWindow, "Window already created but no presshell?"); + + nsCOMPtr<nsIBaseWindow> base_win(mContainer); + if (base_win) { + base_win->GetParentWidget(&mParentWidget); + if (mParentWidget) { + // GetParentWidget AddRefs, but mParentWidget is weak + mParentWidget->Release(); + } + } + + nsView* containerView = FindContainerView(); + + nsresult rv = CreateDeviceContext(containerView); + NS_ENSURE_SUCCESS(rv, rv); + + // Create presentation context + NS_ASSERTION(!mPresContext, + "Shouldn't have a prescontext if we have no shell!"); + mPresContext = CreatePresContext(mDocument, nsPresContext::eContext_Galley, + containerView); + NS_ENSURE_TRUE(mPresContext, NS_ERROR_OUT_OF_MEMORY); + + rv = mPresContext->Init(mDeviceContext); + if (NS_FAILED(rv)) { + mPresContext = nullptr; + return rv; + } + + rv = MakeWindow(nsSize(mPresContext->DevPixelsToAppUnits(mBounds.width), + mPresContext->DevPixelsToAppUnits(mBounds.height)), + containerView); + if (NS_FAILED(rv)) return rv; + + if (mPresContext) { + Hide(); + + rv = InitPresentationStuff(mDocument->MayStartLayout()); + } + + // If we get here the document load has already started and the + // window is shown because some JS on the page caused it to be + // shown... + + if (mPresShell) { + RefPtr<PresShell> presShell = mPresShell; // bug 378682 + presShell->UnsuppressPainting(); + } + } + + // Notify observers that a new page has been shown. This will get run + // from the event loop after we actually draw the page. + RefPtr<nsDocumentShownDispatcher> event = + new nsDocumentShownDispatcher(document); + document->Dispatch(event.forget()); + + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::Hide() { + if (!mAttachedToParent && mWindow) { + mWindow->Show(false); + } + + if (!mPresShell) return NS_OK; + + NS_ASSERTION(mPresContext, "Can't have a presshell and no prescontext!"); + + // Avoid leaking the old viewer. + if (mPreviousViewer) { + mPreviousViewer->Destroy(); + mPreviousViewer = nullptr; + } + + if (mIsSticky) { + // This window is sticky, that means that it might be shown again + // and we don't want the presshell n' all that to be thrown away + // just because the window is hidden. + + return NS_OK; + } + + nsCOMPtr<nsIDocShell> docShell(mContainer); + if (docShell) { +#ifdef DEBUG + nsCOMPtr<nsIDocumentViewer> currentViewer; + docShell->GetDocViewer(getter_AddRefs(currentViewer)); + MOZ_ASSERT(currentViewer == this); +#endif + nsCOMPtr<nsILayoutHistoryState> layoutState; + mPresShell->CaptureHistoryState(getter_AddRefs(layoutState)); + } + + // Do not run ScriptRunners queued by DestroyPresShell() in the intermediate + // state before we're done destroying PresShell, PresContext, ViewManager, + // etc. + nsAutoScriptBlocker scriptBlocker; + + DestroyPresShell(); + + DestroyPresContext(); + + mViewManager = nullptr; + mWindow = nullptr; + mDeviceContext = nullptr; + mParentWidget = nullptr; + + nsCOMPtr<nsIBaseWindow> base_win(mContainer); + + if (base_win && !mAttachedToParent) { + base_win->SetParentWidget(nullptr); + } + + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::GetSticky(bool* aSticky) { + *aSticky = mIsSticky; + + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::SetSticky(bool aSticky) { + mIsSticky = aSticky; + + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::ClearHistoryEntry() { + if (mDocument) { + nsJSContext::PokeGC(JS::GCReason::PAGE_HIDE, + mDocument->GetWrapperPreserveColor(), + TimeDuration::FromMilliseconds( + StaticPrefs::javascript_options_gc_delay() * 2)); + } + + mSHEntry = nullptr; + return NS_OK; +} + +//------------------------------------------------------- + +nsresult nsDocumentViewer::MakeWindow(const nsSize& aSize, + nsView* aContainerView) { + if (GetIsPrintPreview()) { + return NS_OK; + } + + bool shouldAttach = ShouldAttachToTopLevel(); + + if (shouldAttach) { + // If the old view is already attached to our parent, detach + DetachFromTopLevelWidget(); + } + + mViewManager = new nsViewManager(); + + nsDeviceContext* dx = mPresContext->DeviceContext(); + + nsresult rv = mViewManager->Init(dx); + if (NS_FAILED(rv)) return rv; + + // The root view is always at 0,0. + nsRect tbounds(nsPoint(0, 0), aSize); + // Create a view + nsView* view = mViewManager->CreateView(tbounds, aContainerView); + if (!view) return NS_ERROR_OUT_OF_MEMORY; + + // Create a widget if we were given a parent widget or don't have a + // container view that we can hook up to without a widget. + // Don't create widgets for ResourceDocs (external resources & svg images), + // because when they're displayed, they're painted into *another* document's + // widget. + if (!mDocument->IsResourceDoc() && (mParentWidget || !aContainerView)) { + // pass in a native widget to be the parent widget ONLY if the view + // hierarchy will stand alone. otherwise the view will find its own parent + // widget and "do the right thing" to establish a parent/child widget + // relationship + widget::InitData initData; + widget::InitData* initDataPtr; + if (!mParentWidget) { + initDataPtr = &initData; + initData.mWindowType = widget::WindowType::Invisible; + } else { + initDataPtr = nullptr; + } + + if (shouldAttach) { + // Reuse the top level parent widget. + rv = view->AttachToTopLevelWidget(mParentWidget); + mAttachedToParent = true; + } else if (!aContainerView && mParentWidget) { + rv = view->CreateWidgetForParent(mParentWidget, initDataPtr, true, false); + } else { + rv = view->CreateWidget(initDataPtr, true, false); + } + if (NS_FAILED(rv)) return rv; + } + + // Setup hierarchical relationship in view manager + mViewManager->SetRootView(view); + + mWindow = view->GetWidget(); + + // This SetFocus is necessary so the Arrow Key and Page Key events + // go to the scrolled view as soon as the Window is created instead of going + // to the browser window (this enables keyboard scrolling of the document) + // mWindow->SetFocus(); + + return rv; +} + +void nsDocumentViewer::DetachFromTopLevelWidget() { + if (mViewManager) { + nsView* oldView = mViewManager->GetRootView(); + if (oldView && oldView->IsAttachedToTopLevel()) { + oldView->DetachFromTopLevelWidget(); + } + } + mAttachedToParent = false; +} + +nsView* nsDocumentViewer::FindContainerView() { + if (!mContainer) { + return nullptr; + } + + nsCOMPtr<nsIDocShell> docShell(mContainer); + nsCOMPtr<nsPIDOMWindowOuter> pwin(docShell->GetWindow()); + if (!pwin) { + return nullptr; + } + + nsCOMPtr<Element> containerElement = pwin->GetFrameElementInternal(); + if (!containerElement) { + return nullptr; + } + + nsIFrame* subdocFrame = containerElement->GetPrimaryFrame(); + if (!subdocFrame) { + // XXX Silenced by default in bug 1175289 + LAYOUT_WARNING("Subdocument container has no frame"); + return nullptr; + } + + // Check subdocFrame just to be safe. If this somehow fails we treat that as + // display:none, the document is not displayed. + if (!subdocFrame->IsSubDocumentFrame()) { + NS_WARNING_ASSERTION(subdocFrame->Type() == LayoutFrameType::None, + "Subdocument container has non-subdocument frame"); + return nullptr; + } + + NS_ASSERTION(subdocFrame->GetView(), "Subdoc frames must have views"); + return static_cast<nsSubDocumentFrame*>(subdocFrame)->EnsureInnerView(); +} + +nsresult nsDocumentViewer::CreateDeviceContext(nsView* aContainerView) { + MOZ_ASSERT(!mPresShell && !mWindow, + "This will screw up our existing presentation"); + MOZ_ASSERT(mDocument, "Gotta have a document here"); + + Document* doc = mDocument->GetDisplayDocument(); + if (doc) { + NS_ASSERTION(!aContainerView, + "External resource document embedded somewhere?"); + // We want to use our display document's device context if possible + nsPresContext* ctx = doc->GetPresContext(); + if (ctx) { + mDeviceContext = ctx->DeviceContext(); + return NS_OK; + } + } + + // Create a device context even if we already have one, since our widget + // might have changed. + nsIWidget* widget = nullptr; + if (aContainerView) { + widget = aContainerView->GetNearestWidget(nullptr); + } + if (!widget) { + widget = mParentWidget; + } + if (widget) { + widget = widget->GetTopLevelWidget(); + } + + mDeviceContext = new nsDeviceContext(); + mDeviceContext->Init(widget); + return NS_OK; +} + +// Return the selection for the document. Note that text fields have their +// own selection, which cannot be accessed with this method. +mozilla::dom::Selection* nsDocumentViewer::GetDocumentSelection() { + if (!mPresShell) { + return nullptr; + } + + return mPresShell->GetCurrentSelection(SelectionType::eNormal); +} + +/* ============================================================================ + * nsIDocumentViewerEdit + * ============================================================================ + */ + +MOZ_CAN_RUN_SCRIPT_BOUNDARY NS_IMETHODIMP nsDocumentViewer::ClearSelection() { + // use nsCopySupport::GetSelectionForCopy() ? + RefPtr<mozilla::dom::Selection> selection = GetDocumentSelection(); + if (!selection) { + return NS_ERROR_FAILURE; + } + + ErrorResult rv; + selection->CollapseToStart(rv); + return rv.StealNSResult(); +} + +MOZ_CAN_RUN_SCRIPT_BOUNDARY NS_IMETHODIMP nsDocumentViewer::SelectAll() { + // XXX this is a temporary implementation copied from nsWebShell + // for now. I think Document and friends should have some helper + // functions to make this easier. + + // use nsCopySupport::GetSelectionForCopy() ? + RefPtr<mozilla::dom::Selection> selection = GetDocumentSelection(); + if (!selection) { + return NS_ERROR_FAILURE; + } + + if (!mDocument) { + return NS_ERROR_FAILURE; + } + + nsCOMPtr<nsINode> bodyNode; + if (mDocument->IsHTMLOrXHTML()) { + // XXXbz why not just do GetBody() for all documents, then GetRootElement() + // if GetBody() is null? + bodyNode = mDocument->GetBody(); + } else { + bodyNode = mDocument->GetRootElement(); + } + if (!bodyNode) return NS_ERROR_FAILURE; + + ErrorResult err; + selection->RemoveAllRanges(err); + if (err.Failed()) { + return err.StealNSResult(); + } + + mozilla::dom::Selection::AutoUserInitiated userSelection(selection); + selection->SelectAllChildren(*bodyNode, err); + return err.StealNSResult(); +} + +NS_IMETHODIMP nsDocumentViewer::CopySelection() { + RefPtr<PresShell> presShell = mPresShell; + nsCopySupport::FireClipboardEvent(eCopy, nsIClipboard::kGlobalClipboard, + presShell, nullptr); + return NS_OK; +} + +NS_IMETHODIMP nsDocumentViewer::CopyLinkLocation() { + NS_ENSURE_TRUE(mPresShell, NS_ERROR_NOT_INITIALIZED); + nsCOMPtr<nsINode> node = GetPopupLinkNode(); + // make noise if we're not in a link + NS_ENSURE_TRUE(node, NS_ERROR_FAILURE); + + nsCOMPtr<dom::Element> elm(do_QueryInterface(node)); + NS_ENSURE_TRUE(elm, NS_ERROR_FAILURE); + + nsAutoString locationText; + nsContentUtils::GetLinkLocation(elm, locationText); + if (locationText.IsEmpty()) return NS_ERROR_FAILURE; + + nsresult rv = NS_OK; + nsCOMPtr<nsIClipboardHelper> clipboard( + do_GetService("@mozilla.org/widget/clipboardhelper;1", &rv)); + NS_ENSURE_SUCCESS(rv, rv); + + // copy the href onto the clipboard + return clipboard->CopyString(locationText); +} + +NS_IMETHODIMP nsDocumentViewer::CopyImage(int32_t aCopyFlags) { + NS_ENSURE_TRUE(mPresShell, NS_ERROR_NOT_INITIALIZED); + nsCOMPtr<nsIImageLoadingContent> node = GetPopupImageNode(); + // make noise if we're not in an image + NS_ENSURE_TRUE(node, NS_ERROR_FAILURE); + + nsCOMPtr<nsILoadContext> loadContext(mContainer); + return nsCopySupport::ImageCopy(node, loadContext, aCopyFlags); +} + +NS_IMETHODIMP nsDocumentViewer::GetCopyable(bool* aCopyable) { + NS_ENSURE_ARG_POINTER(aCopyable); + *aCopyable = nsCopySupport::CanCopy(mDocument); + return NS_OK; +} + +NS_IMETHODIMP nsDocumentViewer::GetContents(const char* mimeType, + bool selectionOnly, + nsAString& aOutValue) { + aOutValue.Truncate(); + + NS_ENSURE_TRUE(mPresShell, NS_ERROR_NOT_INITIALIZED); + NS_ENSURE_TRUE(mDocument, NS_ERROR_NOT_INITIALIZED); + + // Now we have the selection. Make sure it's nonzero: + RefPtr<Selection> sel; + if (selectionOnly) { + sel = nsCopySupport::GetSelectionForCopy(mDocument); + NS_ENSURE_TRUE(sel, NS_ERROR_FAILURE); + + if (NS_WARN_IF(sel->IsCollapsed())) { + return NS_OK; + } + } + + // call the copy code + return nsCopySupport::GetContents(nsDependentCString(mimeType), 0, sel, + mDocument, aOutValue); +} + +NS_IMETHODIMP nsDocumentViewer::GetCanGetContents(bool* aCanGetContents) { + NS_ENSURE_ARG_POINTER(aCanGetContents); + *aCanGetContents = false; + NS_ENSURE_STATE(mDocument); + *aCanGetContents = nsCopySupport::CanCopy(mDocument); + return NS_OK; +} + +NS_IMETHODIMP nsDocumentViewer::SetCommandNode(nsINode* aNode) { + Document* document = GetDocument(); + NS_ENSURE_STATE(document); + + nsCOMPtr<nsPIDOMWindowOuter> window(document->GetWindow()); + NS_ENSURE_TRUE(window, NS_ERROR_NOT_AVAILABLE); + + nsCOMPtr<nsPIWindowRoot> root = window->GetTopWindowRoot(); + NS_ENSURE_STATE(root); + + root->SetPopupNode(aNode); + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::GetDeviceFullZoomForTest(float* aDeviceFullZoom) { + NS_ENSURE_ARG_POINTER(aDeviceFullZoom); + nsPresContext* pc = GetPresContext(); + *aDeviceFullZoom = pc ? pc->GetDeviceFullZoom() : 1.0; + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::SetAuthorStyleDisabled(bool aStyleDisabled) { + if (mPresShell) { + mPresShell->SetAuthorStyleDisabled(aStyleDisabled); + } + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::GetAuthorStyleDisabled(bool* aStyleDisabled) { + if (mPresShell) { + *aStyleDisabled = mPresShell->GetAuthorStyleDisabled(); + } else { + *aStyleDisabled = false; + } + return NS_OK; +} + +/* [noscript,notxpcom] Encoding getHintCharset (); */ +NS_IMETHODIMP_(const Encoding*) +nsDocumentViewer::GetReloadEncodingAndSource(int32_t* aSource) { + *aSource = mReloadEncodingSource; + if (kCharsetUninitialized == mReloadEncodingSource) { + return nullptr; + } + return mReloadEncoding; +} + +NS_IMETHODIMP_(void) +nsDocumentViewer::SetReloadEncodingAndSource(const Encoding* aEncoding, + int32_t aSource) { + MOZ_ASSERT( + aSource == kCharsetUninitialized || + (aSource >= + kCharsetFromFinalAutoDetectionWouldHaveBeenUTF8InitialWasASCII && + aSource <= + kCharsetFromFinalAutoDetectionWouldNotHaveBeenUTF8DependedOnTLDInitialWasASCII) || + aSource == kCharsetFromFinalUserForcedAutoDetection); + mReloadEncoding = aEncoding; + mReloadEncodingSource = aSource; +} + +NS_IMETHODIMP_(void) +nsDocumentViewer::ForgetReloadEncoding() { + mReloadEncoding = nullptr; + mReloadEncodingSource = kCharsetUninitialized; +} + +MOZ_CAN_RUN_SCRIPT_BOUNDARY NS_IMETHODIMP nsDocumentViewer::GetContentSize( + int32_t aMaxWidth, int32_t aMaxHeight, int32_t aPrefWidth, int32_t* aWidth, + int32_t* aHeight) { + RefPtr<BrowsingContext> bc = mContainer->GetBrowsingContext(); + NS_ENSURE_TRUE(bc, NS_ERROR_NOT_AVAILABLE); + + // It's only valid to access this from a top frame. Doesn't work from + // sub-frames. + NS_ENSURE_TRUE(bc->IsTop(), NS_ERROR_FAILURE); + + // Convert max-width/height and pref-width to app units. + if (aMaxWidth > 0) { + aMaxWidth = CSSPixel::ToAppUnits(aMaxWidth); + } else { + aMaxWidth = NS_UNCONSTRAINEDSIZE; + } + if (aMaxHeight > 0) { + aMaxHeight = CSSPixel::ToAppUnits(aMaxHeight); + } else { + aMaxHeight = NS_UNCONSTRAINEDSIZE; + } + if (aPrefWidth > 0) { + aPrefWidth = CSSPixel::ToAppUnits(aPrefWidth); + } else { + aPrefWidth = 0; + } + + RefPtr<PresShell> presShell = GetPresShell(); + NS_ENSURE_TRUE(presShell, NS_ERROR_FAILURE); + + // Flush out all content and style updates. We can't use a resize reflow + // because it won't change some sizes that a style change reflow will. + mDocument->FlushPendingNotifications(FlushType::Layout); + + nsIFrame* root = presShell->GetRootFrame(); + NS_ENSURE_TRUE(root, NS_ERROR_FAILURE); + + WritingMode wm = root->GetWritingMode(); + + nscoord prefISize; + { + const auto& constraints = presShell->GetWindowSizeConstraints(); + aMaxHeight = std::min(aMaxHeight, constraints.mMaxSize.height); + aMaxWidth = std::min(aMaxWidth, constraints.mMaxSize.width); + + UniquePtr<gfxContext> rcx(presShell->CreateReferenceRenderingContext()); + const nscoord minISize = wm.IsVertical() ? constraints.mMinSize.height + : constraints.mMinSize.width; + const nscoord maxISize = wm.IsVertical() ? aMaxHeight : aMaxWidth; + if (aPrefWidth) { + prefISize = std::max(root->GetMinISize(rcx.get()), aPrefWidth); + } else { + prefISize = root->GetPrefISize(rcx.get()); + } + prefISize = nsPresContext::RoundUpAppUnitsToCSSPixel( + std::max(minISize, std::min(prefISize, maxISize))); + } + + // We should never intentionally get here with this sentinel value, but it's + // possible that a document with huge sizes might inadvertently have a + // prefISize that exactly matches NS_UNCONSTRAINEDSIZE. + // Just bail if that happens. + NS_ENSURE_TRUE(prefISize != NS_UNCONSTRAINEDSIZE, NS_ERROR_FAILURE); + + nscoord height = wm.IsVertical() ? prefISize : aMaxHeight; + nscoord width = wm.IsVertical() ? aMaxWidth : prefISize; + + presShell->ResizeReflow(width, height, ResizeReflowOptions::BSizeLimit); + + RefPtr<nsPresContext> presContext = GetPresContext(); + NS_ENSURE_TRUE(presContext, NS_ERROR_FAILURE); + + // Protect against bogus returns here + nsRect shellArea = presContext->GetVisibleArea(); + NS_ENSURE_TRUE(shellArea.width != NS_UNCONSTRAINEDSIZE && + shellArea.height != NS_UNCONSTRAINEDSIZE, + NS_ERROR_FAILURE); + + // Ceil instead of rounding here, so we can actually guarantee showing all the + // content. + *aWidth = std::ceil(CSSPixel::FromAppUnits(shellArea.width)); + *aHeight = std::ceil(CSSPixel::FromAppUnits(shellArea.height)); + + return NS_OK; +} + +NS_IMPL_ISUPPORTS(nsDocViewerSelectionListener, nsISelectionListener) + +/* + * GetPopupNode, GetPopupLinkNode and GetPopupImageNode are helpers + * for the cmd_copyLink / cmd_copyImageLocation / cmd_copyImageContents family + * of commands. The focus controller stores the popup node, these retrieve + * them and munge appropriately. Note that we have to store the popup node + * rather than retrieving it from EventStateManager::GetFocusedContent because + * not all content (images included) can receive focus. + */ + +already_AddRefed<nsINode> nsDocumentViewer::GetPopupNode() { + // get the document + Document* document = GetDocument(); + NS_ENSURE_TRUE(document, nullptr); + + // get the private dom window + nsCOMPtr<nsPIDOMWindowOuter> window(document->GetWindow()); + NS_ENSURE_TRUE(window, nullptr); + if (window) { + nsCOMPtr<nsPIWindowRoot> root = window->GetTopWindowRoot(); + NS_ENSURE_TRUE(root, nullptr); + + // get the popup node + nsCOMPtr<nsINode> node = root->GetPopupNode(); + if (!node) { + nsPIDOMWindowOuter* rootWindow = root->GetWindow(); + if (rootWindow) { + nsCOMPtr<Document> rootDoc = rootWindow->GetExtantDoc(); + if (rootDoc) { + nsXULPopupManager* pm = nsXULPopupManager::GetInstance(); + if (pm) { + node = pm->GetLastTriggerPopupNode(rootDoc); + } + } + } + } + return node.forget(); + } + + return nullptr; +} + +// GetPopupLinkNode: return popup link node or fail +already_AddRefed<nsINode> nsDocumentViewer::GetPopupLinkNode() { + // find popup node + nsCOMPtr<nsINode> node = GetPopupNode(); + + // find out if we have a link in our ancestry + while (node) { + if (const auto* element = Element::FromNode(*node)) { + if (element->IsLink()) { + return node.forget(); + } + } + + // get our parent and keep trying... + node = node->GetParentNode(); + } + + // if we have no node, fail + return nullptr; +} + +// GetPopupLinkNode: return popup image node or fail +already_AddRefed<nsIImageLoadingContent> nsDocumentViewer::GetPopupImageNode() { + // find popup node + nsCOMPtr<nsINode> node = GetPopupNode(); + nsCOMPtr<nsIImageLoadingContent> img = do_QueryInterface(node); + return img.forget(); +} + +/* + * XXX dr + * ------ + * These two functions -- GetInLink and GetInImage -- are kind of annoying + * in that they only get called from the controller (in + * nsDOMWindowController::IsCommandEnabled). The actual construction of the + * context menus in communicator (nsContextMenu.js) has its own, redundant + * tests. No big deal, but good to keep in mind if we ever clean context + * menus. + */ + +NS_IMETHODIMP nsDocumentViewer::GetInLink(bool* aInLink) { +#ifdef DEBUG_dr + printf("dr :: nsDocumentViewer::GetInLink\n"); +#endif + + NS_ENSURE_ARG_POINTER(aInLink); + + // we're not in a link unless i say so + *aInLink = false; + + // get the popup link + nsCOMPtr<nsINode> node = GetPopupLinkNode(); + if (!node) { + return NS_ERROR_FAILURE; + } + + // if we made it here, we're in a link + *aInLink = true; + return NS_OK; +} + +NS_IMETHODIMP nsDocumentViewer::GetInImage(bool* aInImage) { +#ifdef DEBUG_dr + printf("dr :: nsDocumentViewer::GetInImage\n"); +#endif + + NS_ENSURE_ARG_POINTER(aInImage); + + // we're not in an image unless i say so + *aInImage = false; + + // get the popup image + nsCOMPtr<nsIImageLoadingContent> node = GetPopupImageNode(); + if (!node) { + return NS_ERROR_FAILURE; + } + + // Make sure there is a URI assigned. This allows <input type="image"> to + // be an image but rejects other <input> types. This matches what + // nsContextMenu.js does. + nsCOMPtr<nsIURI> uri; + node->GetCurrentURI(getter_AddRefs(uri)); + if (uri) { + // if we made it here, we're in an image + *aInImage = true; + } + + return NS_OK; +} + +NS_IMETHODIMP nsDocViewerSelectionListener::NotifySelectionChanged( + Document*, Selection*, int16_t aReason, int32_t aAmount) { + if (!mDocViewer) { + return NS_OK; + } + + // get the selection state + RefPtr<mozilla::dom::Selection> selection = + mDocViewer->GetDocumentSelection(); + if (!selection) { + return NS_ERROR_FAILURE; + } + + Document* theDoc = mDocViewer->GetDocument(); + if (!theDoc) return NS_ERROR_FAILURE; + + nsCOMPtr<nsPIDOMWindowOuter> domWindow = theDoc->GetWindow(); + if (!domWindow) return NS_ERROR_FAILURE; + + bool selectionCollapsed = selection->IsCollapsed(); + // We only call UpdateCommands when the selection changes from collapsed to + // non-collapsed or vice versa, however we skip the initializing collapse. We + // might need another update string for simple selection changes, but that + // would be expenseive. + if (mSelectionWasCollapsed != selectionCollapsed) { + domWindow->UpdateCommands(u"select"_ns); + mSelectionWasCollapsed = selectionCollapsed; + } + + return NS_OK; +} + +// nsDocViewerFocusListener +NS_IMPL_ISUPPORTS(nsDocViewerFocusListener, nsIDOMEventListener) + +nsresult nsDocViewerFocusListener::HandleEvent(Event* aEvent) { + NS_ENSURE_STATE(mDocViewer); + + RefPtr<PresShell> presShell = mDocViewer->GetPresShell(); + NS_ENSURE_TRUE(presShell, NS_ERROR_FAILURE); + + RefPtr<nsFrameSelection> selection = + presShell->GetLastFocusedFrameSelection(); + NS_ENSURE_TRUE(selection, NS_ERROR_FAILURE); + auto selectionStatus = selection->GetDisplaySelection(); + nsAutoString eventType; + aEvent->GetType(eventType); + if (eventType.EqualsLiteral("focus")) { + // If selection was disabled, re-enable it. + if (selectionStatus == nsISelectionController::SELECTION_DISABLED || + selectionStatus == nsISelectionController::SELECTION_HIDDEN) { + selection->SetDisplaySelection(nsISelectionController::SELECTION_ON); + selection->RepaintSelection(SelectionType::eNormal); + } + // See EditorBase::FinalizeSelection. This fixes up the case where focus + // left the editor's selection but returned to something else. + if (selection != presShell->ConstFrameSelection()) { + RefPtr<Document> doc = presShell->GetDocument(); + const bool selectionMatchesFocus = + selection->GetLimiter() && + selection->GetLimiter()->GetChromeOnlyAccessSubtreeRootParent() == + doc->GetUnretargetedFocusedContent(); + if (NS_WARN_IF(!selectionMatchesFocus)) { + presShell->FrameSelectionWillLoseFocus(*selection); + presShell->SelectionWillTakeFocus(); + } + } + } else { + MOZ_ASSERT(eventType.EqualsLiteral("blur"), "Unexpected event type"); + // If selection was on, disable it. + if (selectionStatus == nsISelectionController::SELECTION_ON || + selectionStatus == nsISelectionController::SELECTION_ATTENTION) { + selection->SetDisplaySelection( + nsISelectionController::SELECTION_DISABLED); + selection->RepaintSelection(SelectionType::eNormal); + } + } + + return NS_OK; +} + +/** --------------------------------------------------- + * From nsIWebBrowserPrint + */ + +#ifdef NS_PRINTING + +NS_IMETHODIMP +nsDocumentViewer::Print(nsIPrintSettings* aPrintSettings, + RemotePrintJobChild* aRemotePrintJob, + nsIWebProgressListener* aWebProgressListener) { + if (NS_WARN_IF(!mContainer)) { + PR_PL(("Container was destroyed yet we are still trying to use it!")); + return NS_ERROR_FAILURE; + } + + if (NS_WARN_IF(!mDocument) || NS_WARN_IF(!mDeviceContext)) { + PR_PL(("Can't Print without a document and a device context")); + return NS_ERROR_FAILURE; + } + + if (NS_WARN_IF(mPrintJob && mPrintJob->GetIsPrinting())) { + // If we are printing another URL, then exit. + // The reason we check here is because this method can be called while + // another is still in here (the printing dialog is a good example). the + // only time we can print more than one job at a time is the regression + // tests. + nsresult rv = NS_ERROR_NOT_AVAILABLE; + RefPtr<nsPrintJob>(mPrintJob)->FirePrintingErrorEvent(rv); + return rv; + } + + OnDonePrinting(); + + // Note: mContainer and mDocument are known to be non-null via null-checks + // earlier in this function. + // TODO(dholbert) Do we need to bother with this stack-owned local RefPtr? + // (Is there an edge case where it's needed to keep the nsPrintJob alive?) + RefPtr<nsPrintJob> printJob = + new nsPrintJob(*this, *mContainer, *mDocument, + float(AppUnitsPerCSSInch()) / + float(mDeviceContext->AppUnitsPerDevPixel())); + mPrintJob = printJob; + + nsresult rv = printJob->Print(*mDocument, aPrintSettings, aRemotePrintJob, + aWebProgressListener); + if (NS_WARN_IF(NS_FAILED(rv))) { + OnDonePrinting(); + } + return rv; +} + +NS_IMETHODIMP +nsDocumentViewer::PrintPreview(nsIPrintSettings* aPrintSettings, + nsIWebProgressListener* aWebProgressListener, + PrintPreviewResolver&& aCallback) { +# ifdef NS_PRINT_PREVIEW + RefPtr<Document> doc = mDocument.get(); + NS_ENSURE_STATE(doc); + + if (NS_WARN_IF(GetIsPrinting())) { + return NS_ERROR_FAILURE; + } + + nsCOMPtr<nsIDocShell> docShell(mContainer); + if (NS_WARN_IF(!docShell) || NS_WARN_IF(!mDeviceContext)) { + PR_PL(("Can't Print Preview without device context and docshell")); + return NS_ERROR_FAILURE; + } + + NS_ENSURE_STATE(!GetIsPrinting()); + // beforeprint event may have caused ContentViewer to be shutdown. + NS_ENSURE_STATE(mContainer); + NS_ENSURE_STATE(mDeviceContext); + + OnDonePrinting(); + + // Note: mContainer and doc are known to be non-null via null-checks earlier + // in this function. + // TODO(dholbert) Do we need to bother with this stack-owned local RefPtr? + // (Is there an edge case where it's needed to keep the nsPrintJob alive?) + RefPtr<nsPrintJob> printJob = + new nsPrintJob(*this, *mContainer, *doc, + float(AppUnitsPerCSSInch()) / + float(mDeviceContext->AppUnitsPerDevPixel())); + mPrintJob = printJob; + + nsresult rv = printJob->PrintPreview( + *doc, aPrintSettings, aWebProgressListener, std::move(aCallback)); + if (NS_WARN_IF(NS_FAILED(rv))) { + OnDonePrinting(); + } + return rv; +# else + return NS_ERROR_FAILURE; +# endif // NS_PRINT_PREVIEW +} + +static const nsIFrame* GetTargetPageFrame(int32_t aTargetPageNum, + nsPageSequenceFrame* aSequenceFrame) { + MOZ_ASSERT(aTargetPageNum > 0 && + aTargetPageNum <= + aSequenceFrame->PrincipalChildList().GetLength()); + return aSequenceFrame->PrincipalChildList().FrameAt(aTargetPageNum - 1); +} + +// Calculate the scroll position where the center of |aFrame| is positioned at +// the center of |aScrollable|'s scroll port for the print preview. +// So what we do for that is; +// 1) Calculate the position of the center of |aFrame| in the print preview +// coordinates. +// 2) Reduce the half height of the scroll port from the result of 1. +static nscoord ScrollPositionForFrame(const nsIFrame* aFrame, + nsIScrollableFrame* aScrollable, + float aPreviewScale) { + // Note that even if the computed scroll position is out of the range of + // the scroll port, it gets clamped in nsIScrollableFrame::ScrollTo. + return nscoord(aPreviewScale * aFrame->GetRect().Center().y - + float(aScrollable->GetScrollPortRect().height) / 2.0f); +} + +//---------------------------------------------------------------------- +NS_IMETHODIMP +nsDocumentViewer::PrintPreviewScrollToPage(int16_t aType, int32_t aPageNum) { + if (!GetIsPrintPreview() || mPrintJob->GetIsCreatingPrintPreview()) + return NS_ERROR_FAILURE; + + nsIScrollableFrame* sf = mPresShell->GetRootScrollFrameAsScrollable(); + if (!sf) return NS_OK; + + auto [seqFrame, sheetCount] = mPrintJob->GetSeqFrameAndCountSheets(); + Unused << sheetCount; + if (!seqFrame) { + return NS_ERROR_FAILURE; + } + + float previewScale = seqFrame->GetPrintPreviewScale(); + + nsPoint dest = sf->GetScrollPosition(); + + switch (aType) { + case nsIWebBrowserPrint::PRINTPREVIEW_HOME: + dest.y = 0; + break; + case nsIWebBrowserPrint::PRINTPREVIEW_END: + dest.y = sf->GetScrollRange().YMost(); + break; + case nsIWebBrowserPrint::PRINTPREVIEW_PREV_PAGE: + case nsIWebBrowserPrint::PRINTPREVIEW_NEXT_PAGE: { + auto [currentFrame, currentSheetNumber] = GetCurrentSheetFrameAndNumber(); + Unused << currentSheetNumber; + if (!currentFrame) { + return NS_OK; + } + + const nsIFrame* targetFrame = nullptr; + if (aType == nsIWebBrowserPrint::PRINTPREVIEW_PREV_PAGE) { + targetFrame = currentFrame->GetPrevInFlow(); + } else { + targetFrame = currentFrame->GetNextInFlow(); + } + if (!targetFrame) { + return NS_OK; + } + + dest.y = ScrollPositionForFrame(targetFrame, sf, previewScale); + break; + } + case nsIWebBrowserPrint::PRINTPREVIEW_GOTO_PAGENUM: { + if (aPageNum <= 0 || aPageNum > sheetCount) { + return NS_ERROR_INVALID_ARG; + } + + const nsIFrame* targetFrame = GetTargetPageFrame(aPageNum, seqFrame); + MOZ_ASSERT(targetFrame); + + dest.y = ScrollPositionForFrame(targetFrame, sf, previewScale); + break; + } + default: + return NS_ERROR_INVALID_ARG; + break; + } + + sf->ScrollTo(dest, ScrollMode::Instant); + + return NS_OK; +} + +std::tuple<const nsIFrame*, int32_t> +nsDocumentViewer::GetCurrentSheetFrameAndNumber() const { + MOZ_ASSERT(mPrintJob); + MOZ_ASSERT(GetIsPrintPreview() && !mPrintJob->GetIsCreatingPrintPreview()); + + // in PP mPrtPreview->mPrintObject->mSeqFrame is null + auto [seqFrame, sheetCount] = mPrintJob->GetSeqFrameAndCountSheets(); + Unused << sheetCount; + if (!seqFrame) { + return {nullptr, 0}; + } + + nsIScrollableFrame* sf = mPresShell->GetRootScrollFrameAsScrollable(); + if (!sf) { + // No scrollable contents, returns 1 even if there are multiple sheets. + return {seqFrame->PrincipalChildList().FirstChild(), 1}; + } + + nsPoint currentScrollPosition = sf->GetScrollPosition(); + float halfwayPoint = + currentScrollPosition.y + float(sf->GetScrollPortRect().height) / 2.0f; + float lastDistanceFromHalfwayPoint = std::numeric_limits<float>::max(); + int32_t sheetNumber = 0; + const nsIFrame* currentSheet = nullptr; + float previewScale = seqFrame->GetPrintPreviewScale(); + for (const nsIFrame* sheetFrame : seqFrame->PrincipalChildList()) { + nsRect sheetRect = sheetFrame->GetRect(); + sheetNumber++; + currentSheet = sheetFrame; + + float bottomOfSheet = sheetRect.YMost() * previewScale; + if (bottomOfSheet < halfwayPoint) { + // If the bottom of the sheet is not yet over the halfway point, iterate + // the next frame to see if the next frame is over the halfway point and + // compare the distance from the halfway point. + lastDistanceFromHalfwayPoint = halfwayPoint - bottomOfSheet; + continue; + } + + float topOfSheet = sheetRect.Y() * previewScale; + if (topOfSheet <= halfwayPoint) { + // If the top of the sheet is not yet over the halfway point or on the + // point, it's the current sheet. + break; + } + + // Now the sheet rect is completely over the halfway point, compare the + // distances from the halfway point. + if ((topOfSheet - halfwayPoint) >= lastDistanceFromHalfwayPoint) { + // If the previous sheet distance is less than or equal to the current + // sheet distance, choose the previous one as the current. + sheetNumber--; + MOZ_ASSERT(sheetNumber > 0); + currentSheet = currentSheet->GetPrevInFlow(); + MOZ_ASSERT(currentSheet); + } + break; + } + + MOZ_ASSERT(sheetNumber <= sheetCount); + return {currentSheet, sheetNumber}; +} + +// XXXdholbert As noted in nsIWebBrowserPrint.idl, this API (the IDL attr +// 'printPreviewCurrentPageNumber') is misnamed and needs s/Page/Sheet/. See +// bug 1669762. +NS_IMETHODIMP +nsDocumentViewer::GetPrintPreviewCurrentPageNumber(int32_t* aNumber) { + NS_ENSURE_ARG_POINTER(aNumber); + NS_ENSURE_TRUE(mPrintJob, NS_ERROR_FAILURE); + if (!GetIsPrintPreview() || mPrintJob->GetIsCreatingPrintPreview()) { + return NS_ERROR_FAILURE; + } + + auto [currentFrame, currentSheetNumber] = GetCurrentSheetFrameAndNumber(); + Unused << currentFrame; + if (!currentSheetNumber) { + return NS_ERROR_FAILURE; + } + + *aNumber = currentSheetNumber; + + return NS_OK; +} + +// XXX This always returns false for subdocuments +NS_IMETHODIMP +nsDocumentViewer::GetDoingPrint(bool* aDoingPrint) { + NS_ENSURE_ARG_POINTER(aDoingPrint); + + // XXX shouldn't this be GetDoingPrint() ? + *aDoingPrint = mPrintJob ? mPrintJob->CreatedForPrintPreview() : false; + return NS_OK; +} + +// XXX This always returns false for subdocuments +NS_IMETHODIMP +nsDocumentViewer::GetDoingPrintPreview(bool* aDoingPrintPreview) { + NS_ENSURE_ARG_POINTER(aDoingPrintPreview); + + *aDoingPrintPreview = mPrintJob ? mPrintJob->CreatedForPrintPreview() : false; + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::ExitPrintPreview() { + NS_ENSURE_TRUE(mPrintJob, NS_ERROR_FAILURE); + + if (GetIsPrinting()) { + // Block exiting the print preview window if we're in the middle of an + // actual print. + return NS_ERROR_FAILURE; + } + + if (!GetIsPrintPreview()) { + NS_ERROR("Wow, we should never get here!"); + return NS_OK; + } + +# ifdef NS_PRINT_PREVIEW + mPrintJob->Destroy(); + mPrintJob = nullptr; + + // Since the print preview implementation discards the window that was used + // to show the print preview, we skip certain cleanup that we would otherwise + // want to do. Specifically, we do not call `SetIsPrintPreview(false)` to + // unblock navigation, we do not call `SetOverrideDPPX` to reset the + // devicePixelRatio, and we do not call `Show` to make such changes take + // affect. +# endif // NS_PRINT_PREVIEW + + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::GetRawNumPages(int32_t* aRawNumPages) { + NS_ENSURE_ARG_POINTER(aRawNumPages); + NS_ENSURE_TRUE(mPrintJob, NS_ERROR_FAILURE); + + *aRawNumPages = mPrintJob->GetRawNumPages(); + return *aRawNumPages > 0 ? NS_OK : NS_ERROR_FAILURE; +} + +// XXXdholbert As noted in nsIWebBrowserPrint.idl, this API (the IDL attr +// 'printPreviewNumPages') is misnamed and needs s/Page/Sheet/. +// See bug 1669762. +NS_IMETHODIMP +nsDocumentViewer::GetPrintPreviewNumPages(int32_t* aPrintPreviewNumPages) { + NS_ENSURE_ARG_POINTER(aPrintPreviewNumPages); + NS_ENSURE_TRUE(mPrintJob, NS_ERROR_FAILURE); + *aPrintPreviewNumPages = mPrintJob->GetPrintPreviewNumSheets(); + return *aPrintPreviewNumPages > 0 ? NS_OK : NS_ERROR_FAILURE; +} + +//---------------------------------------------------------------------------------- +// Printing/Print Preview Helpers +//---------------------------------------------------------------------------------- + +//---------------------------------------------------------------------------------- +// Walks the document tree and tells each DocShell whether Printing/PP is +// happening +#endif // NS_PRINTING + +bool nsDocumentViewer::ShouldAttachToTopLevel() { + if (!mParentWidget) { + return false; + } + + // We always attach when using puppet widgets + if (nsIWidget::UsePuppetWidgets()) { + return true; + } + + // FIXME(emilio): Can we unify this between macOS and aother platforms? +#ifdef XP_MACOSX + return false; +#else +# ifdef DEBUG + nsIWidgetListener* parentListener = mParentWidget->GetWidgetListener(); + MOZ_ASSERT(!parentListener || !parentListener->GetView(), + "Expect a top level widget"); +# endif + return true; +#endif +} + +//------------------------------------------------------------ +// XXX this always returns false for subdocuments +bool nsDocumentViewer::GetIsPrinting() const { +#ifdef NS_PRINTING + if (mPrintJob) { + return mPrintJob->GetIsPrinting(); + } +#endif + return false; +} + +//------------------------------------------------------------ +// The PrintJob holds the current value +// this called from inside the DocViewer. +// XXX it always returns false for subdocuments +bool nsDocumentViewer::GetIsPrintPreview() const { +#ifdef NS_PRINTING + return mPrintJob && mPrintJob->CreatedForPrintPreview(); +#else + return false; +#endif +} + +//------------------------------------------------------------ +// Notification from the PrintJob of the current PP status +void nsDocumentViewer::SetIsPrintPreview(bool aIsPrintPreview) { + // Protect against pres shell destruction running scripts. + nsAutoScriptBlocker scriptBlocker; + + if (!aIsPrintPreview) { + InvalidatePotentialSubDocDisplayItem(); + if (mPresShell) { + DestroyPresShell(); + } + mWindow = nullptr; + mViewManager = nullptr; + mPresContext = nullptr; + mPresShell = nullptr; + } +} + +//---------------------------------------------------------------------------------- +// nsIDocumentViewerPrint IFace +//---------------------------------------------------------------------------------- + +//------------------------------------------------------------ +void nsDocumentViewer::IncrementDestroyBlockedCount() { + ++mDestroyBlockedCount; +} + +void nsDocumentViewer::DecrementDestroyBlockedCount() { + --mDestroyBlockedCount; +} + +//------------------------------------------------------------ +// This called ONLY when printing has completed and the DV +// is being notified that it should get rid of the nsPrintJob. +// +// BUT, if we are in Print Preview then we want to ignore the +// notification (we do not get rid of the nsPrintJob) +// +// One small caveat: +// This IS called from two places in this module for cleaning +// up when an error occurred during the start up printing +// and print preview +// +void nsDocumentViewer::OnDonePrinting() { +#if defined(NS_PRINTING) && defined(NS_PRINT_PREVIEW) + // If Destroy() has been called during calling nsPrintJob::Print() or + // nsPrintJob::PrintPreview(), mPrintJob is already nullptr here. + // So, the following clean up does nothing in such case. + // (Do we need some of this for that case?) + if (mPrintJob) { + RefPtr<nsPrintJob> printJob = std::move(mPrintJob); + if (GetIsPrintPreview()) { + printJob->DestroyPrintingData(); + } else { + printJob->Destroy(); + } + + // We are done printing, now clean up. + // + // For non-print-preview jobs, we are actually responsible for cleaning up + // our whole <browser> or window (see the OPEN_PRINT_BROWSER code), so gotta + // run window.close(), which will take care of this. + // + // For print preview jobs the front-end code is responsible for cleaning the + // UI. + if (!printJob->CreatedForPrintPreview()) { + if (mContainer) { + if (nsCOMPtr<nsPIDOMWindowOuter> win = mContainer->GetWindow()) { + win->Close(); + } + } + } else if (mClosingWhilePrinting) { + if (mDocument) { + mDocument->Destroy(); + mDocument = nullptr; + } + mClosingWhilePrinting = false; + } + } +#endif // NS_PRINTING && NS_PRINT_PREVIEW +} + +NS_IMETHODIMP nsDocumentViewer::SetPrintSettingsForSubdocument( + nsIPrintSettings* aPrintSettings, RemotePrintJobChild* aRemotePrintJob) { +#ifdef NS_PRINTING + { + nsAutoScriptBlocker scriptBlocker; + + if (mPresShell) { + DestroyPresShell(); + } + + if (mPresContext) { + DestroyPresContext(); + } + + MOZ_ASSERT(!mPresContext); + MOZ_ASSERT(!mPresShell); + + if (MOZ_UNLIKELY(!mDocument)) { + return NS_ERROR_NOT_AVAILABLE; + } + + RefPtr<nsDeviceContextSpecProxy> devspec = + new nsDeviceContextSpecProxy(aRemotePrintJob); + nsresult rv = devspec->Init(aPrintSettings, /* aIsPrintPreview = */ true); + NS_ENSURE_SUCCESS(rv, rv); + + mDeviceContext = new nsDeviceContext(); + rv = mDeviceContext->InitForPrinting(devspec); + + NS_ENSURE_SUCCESS(rv, rv); + + mPresContext = CreatePresContext( + mDocument, nsPresContext::eContext_PrintPreview, FindContainerView()); + mPresContext->SetPrintSettings(aPrintSettings); + rv = mPresContext->Init(mDeviceContext); + NS_ENSURE_SUCCESS(rv, rv); + + rv = MakeWindow(nsSize(mPresContext->DevPixelsToAppUnits(mBounds.width), + mPresContext->DevPixelsToAppUnits(mBounds.height)), + FindContainerView()); + NS_ENSURE_SUCCESS(rv, rv); + + MOZ_TRY(InitPresentationStuff(true)); + } + + RefPtr<PresShell> shell = mPresShell; + shell->FlushPendingNotifications(FlushType::Layout); +#endif + return NS_OK; +} + +NS_IMETHODIMP nsDocumentViewer::SetPageModeForTesting( + bool aPageMode, nsIPrintSettings* aPrintSettings) { + // XXX Page mode is only partially working; it's currently used for + // reftests that require a paginated context + mIsPageMode = aPageMode; + + // The DestroyPresShell call requires a script blocker, since the + // PresShell::Destroy call it does can cause scripts to run, which could + // re-entrantly call methods on the nsDocumentViewer. + nsAutoScriptBlocker scriptBlocker; + + if (mPresShell) { + DestroyPresShell(); + } + + if (mPresContext) { + DestroyPresContext(); + } + + mViewManager = nullptr; + mWindow = nullptr; + + NS_ENSURE_STATE(mDocument); + if (aPageMode) { + mPresContext = CreatePresContext( + mDocument, nsPresContext::eContext_PageLayout, FindContainerView()); + NS_ENSURE_TRUE(mPresContext, NS_ERROR_OUT_OF_MEMORY); + mPresContext->SetPaginatedScrolling(true); + mPresContext->SetPrintSettings(aPrintSettings); + nsresult rv = mPresContext->Init(mDeviceContext); + NS_ENSURE_SUCCESS(rv, rv); + } + NS_ENSURE_SUCCESS(InitInternal(mParentWidget, nullptr, nullptr, mBounds, true, + false, false), + NS_ERROR_FAILURE); + + Show(); + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::GetHistoryEntry(nsISHEntry** aHistoryEntry) { + NS_IF_ADDREF(*aHistoryEntry = mSHEntry); + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::GetIsTabModalPromptAllowed(bool* aAllowed) { + *aAllowed = !mHidden; + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::GetIsHidden(bool* aHidden) { + *aHidden = mHidden; + return NS_OK; +} + +NS_IMETHODIMP +nsDocumentViewer::SetIsHidden(bool aHidden) { + mHidden = aHidden; + return NS_OK; +} + +void nsDocumentViewer::DestroyPresShell() { + // We assert this because destroying the pres shell could otherwise cause + // re-entrancy into nsDocumentViewer methods, and all callers of + // DestroyPresShell need to do other cleanup work afterwards before it + // is safe for those re-entrant method calls to be made. + MOZ_ASSERT(!nsContentUtils::IsSafeToRunScript(), + "DestroyPresShell must only be called when scripts are blocked"); + + // Break circular reference (or something) + mPresShell->EndObservingDocument(); + + RefPtr<mozilla::dom::Selection> selection = GetDocumentSelection(); + if (selection && mSelectionListener) + selection->RemoveSelectionListener(mSelectionListener); + + mPresShell->Destroy(); + mPresShell = nullptr; +} + +void nsDocumentViewer::InvalidatePotentialSubDocDisplayItem() { + if (mViewManager) { + if (nsView* rootView = mViewManager->GetRootView()) { + if (nsView* rootViewParent = rootView->GetParent()) { + if (nsView* subdocview = rootViewParent->GetParent()) { + if (nsIFrame* f = subdocview->GetFrame()) { + if (nsSubDocumentFrame* s = do_QueryFrame(f)) { + s->MarkNeedsDisplayItemRebuild(); + } + } + } + } + } + } +} + +void nsDocumentViewer::DestroyPresContext() { + InvalidatePotentialSubDocDisplayItem(); + mPresContext = nullptr; +} + +void nsDocumentViewer::SetPrintPreviewPresentation(nsViewManager* aViewManager, + nsPresContext* aPresContext, + PresShell* aPresShell) { + // Protect against pres shell destruction running scripts and re-entrantly + // creating a new presentation. + nsAutoScriptBlocker scriptBlocker; + + if (mPresShell) { + DestroyPresShell(); + } + + mWindow = nullptr; + mViewManager = aViewManager; + mPresContext = aPresContext; + mPresShell = aPresShell; + + if (ShouldAttachToTopLevel()) { + DetachFromTopLevelWidget(); + nsView* rootView = mViewManager->GetRootView(); + rootView->AttachToTopLevelWidget(mParentWidget); + mAttachedToParent = true; + } +} + +// Fires the "document-shown" event so that interested parties are aware of it. +NS_IMETHODIMP +nsDocumentShownDispatcher::Run() { + nsCOMPtr<nsIObserverService> observerService = + mozilla::services::GetObserverService(); + if (observerService) { + observerService->NotifyObservers(ToSupports(mDocument), "document-shown", + nullptr); + } + return NS_OK; +} |