/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- * vim: set ts=2 sw=2 et tw=78: * 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/. * * This Original Code has been modified by IBM Corporation. * Modifications made by IBM described herein are * Copyright (c) International Business Machines * Corporation, 2000 * * Modifications to Mozilla code or documentation * identified per MPL Section 3.3 * * Date Modified by Description of modification * 05/03/2000 IBM Corp. Observer events for reflow states */ /* a presentation of a document, part 2 */ #include "mozilla/Logging.h" #include "mozilla/ArrayUtils.h" #include "mozilla/StyleSheetInlines.h" #include "mozilla/EventDispatcher.h" #include "mozilla/EventStateManager.h" #include "mozilla/EventStates.h" #include "mozilla/IMEStateManager.h" #include "mozilla/MemoryReporting.h" #include "mozilla/dom/TabChild.h" #include "mozilla/Likely.h" #include "mozilla/MouseEvents.h" #include "mozilla/Sprintf.h" #include "mozilla/TextEvents.h" #include "mozilla/TouchEvents.h" #include "mozilla/UniquePtr.h" #include "mozilla/Unused.h" #include "mozilla/StyleBackendType.h" #include <algorithm> #ifdef XP_WIN #include "winuser.h" #endif #include "gfxPrefs.h" #include "gfxUserFontSet.h" #include "nsPresShell.h" #include "nsPresContext.h" #include "nsIContent.h" #include "nsIContentIterator.h" #include "mozilla/dom/BeforeAfterKeyboardEvent.h" #include "mozilla/dom/Element.h" #include "mozilla/dom/Event.h" // for Event::GetEventPopupControlState() #include "mozilla/dom/ShadowRoot.h" #include "mozilla/dom/PointerEvent.h" #include "nsIDocument.h" #include "nsAnimationManager.h" #include "nsNameSpaceManager.h" // for Pref-related rule management (bugs 22963,20760,31816) #include "nsFrame.h" #include "FrameLayerBuilder.h" #include "nsViewManager.h" #include "nsView.h" #include "nsCRTGlue.h" #include "prprf.h" #include "prinrval.h" #include "nsTArray.h" #include "nsCOMArray.h" #include "nsContainerFrame.h" #include "nsISelection.h" #include "mozilla/dom/Selection.h" #include "nsGkAtoms.h" #include "nsIDOMRange.h" #include "nsIDOMDocument.h" #include "nsIDOMNode.h" #include "nsIDOMNodeList.h" #include "nsIDOMElement.h" #include "nsRange.h" #include "nsCOMPtr.h" #include "nsAutoPtr.h" #include "nsReadableUtils.h" #include "nsIPageSequenceFrame.h" #include "nsIPermissionManager.h" #include "nsIMozBrowserFrame.h" #include "nsCaret.h" #include "AccessibleCaretEventHub.h" #include "nsIDOMHTMLDocument.h" #include "nsFrameManager.h" #include "nsXPCOM.h" #include "nsILayoutHistoryState.h" #include "nsILineIterator.h" // for ScrollContentIntoView #include "PLDHashTable.h" #include "mozilla/dom/BeforeAfterKeyboardEventBinding.h" #include "mozilla/dom/Touch.h" #include "mozilla/dom/TouchEvent.h" #include "mozilla/dom/PointerEventBinding.h" #include "nsIObserverService.h" #include "nsDocShell.h" // for reflow observation #include "nsIBaseWindow.h" #include "nsError.h" #include "nsLayoutUtils.h" #include "nsViewportInfo.h" #include "nsCSSRendering.h" // for |#ifdef DEBUG| code #include "prenv.h" #include "nsDisplayList.h" #include "nsRegion.h" #include "nsRenderingContext.h" #include "nsAutoLayoutPhase.h" #ifdef MOZ_REFLOW_PERF #include "nsFontMetrics.h" #endif #include "PositionedEventTargeting.h" #include "nsIReflowCallback.h" #include "nsPIDOMWindow.h" #include "nsFocusManager.h" #include "nsIObjectFrame.h" #include "nsIObjectLoadingContent.h" #include "nsNetUtil.h" #include "nsThreadUtils.h" #include "nsStyleSheetService.h" #include "gfxContext.h" #include "gfxUtils.h" #include "nsSMILAnimationController.h" #include "SVGContentUtils.h" #include "nsSVGEffects.h" #include "SVGFragmentIdentifier.h" #include "nsArenaMemoryStats.h" #include "nsFrameSelection.h" #include "mozilla/dom/Performance.h" #include "nsRefreshDriver.h" #include "nsDOMNavigationTiming.h" // Drag & Drop, Clipboard #include "nsIDocShellTreeItem.h" #include "nsIURI.h" #include "nsIScrollableFrame.h" #include "nsITimer.h" #ifdef ACCESSIBILITY #include "nsAccessibilityService.h" #include "mozilla/a11y/DocAccessible.h" #ifdef DEBUG #include "mozilla/a11y/Logging.h" #endif #endif // For style data reconstruction #include "nsStyleChangeList.h" #include "nsCSSFrameConstructor.h" #ifdef MOZ_XUL #include "nsMenuFrame.h" #include "nsTreeBodyFrame.h" #include "nsIBoxObject.h" #include "nsITreeBoxObject.h" #include "nsMenuPopupFrame.h" #include "nsITreeColumns.h" #include "nsIDOMXULMultSelectCntrlEl.h" #include "nsIDOMXULSelectCntrlItemEl.h" #include "nsIDOMXULMenuListElement.h" #endif #include "mozilla/layers/CompositorBridgeChild.h" #include "GeckoProfiler.h" #include "gfxPlatform.h" #include "Layers.h" #include "LayerTreeInvalidation.h" #include "mozilla/css/ImageLoader.h" #include "mozilla/dom/DocumentTimeline.h" #include "mozilla/Preferences.h" #include "mozilla/Telemetry.h" #include "nsCanvasFrame.h" #include "nsIImageLoadingContent.h" #include "nsImageFrame.h" #include "nsIScreen.h" #include "nsIScreenManager.h" #include "nsPlaceholderFrame.h" #include "nsTransitionManager.h" #include "ChildIterator.h" #include "mozilla/RestyleManagerHandle.h" #include "mozilla/RestyleManagerHandleInlines.h" #include "nsIDOMHTMLElement.h" #include "nsIDragSession.h" #include "nsIFrameInlines.h" #include "mozilla/gfx/2D.h" #include "nsSubDocumentFrame.h" #include "nsQueryObject.h" #include "nsLayoutStylesheetCache.h" #include "mozilla/layers/InputAPZContext.h" #include "mozilla/layers/ScrollInputMethods.h" #include "nsStyleSet.h" #include "mozilla/StyleSetHandle.h" #include "mozilla/StyleSetHandleInlines.h" #include "mozilla/StyleSheet.h" #include "mozilla/StyleSheetInlines.h" #include "mozilla/dom/ImageTracker.h" #ifdef ANDROID #include "nsIDocShellTreeOwner.h" #endif #ifdef MOZ_B2G #include "nsIHardwareKeyHandler.h" #endif #ifdef MOZ_TASK_TRACER #include "GeckoTaskTracer.h" using namespace mozilla::tasktracer; #endif #define ANCHOR_SCROLL_FLAGS \ (nsIPresShell::SCROLL_OVERFLOW_HIDDEN | nsIPresShell::SCROLL_NO_PARENT_FRAMES) // define the scalfactor of drag and drop images // relative to the max screen height/width #define RELATIVE_SCALEFACTOR 0.0925f using namespace mozilla; using namespace mozilla::css; using namespace mozilla::dom; using namespace mozilla::gfx; using namespace mozilla::layers; using namespace mozilla::gfx; using namespace mozilla::layout; using PaintFrameFlags = nsLayoutUtils::PaintFrameFlags; CapturingContentInfo nsIPresShell::gCaptureInfo = { false /* mAllowed */, false /* mPointerLock */, false /* mRetargetToElement */, false /* mPreventDrag */ }; nsIContent* nsIPresShell::gKeyDownTarget; // Keeps a map between pointerId and element that currently capturing pointer // with such pointerId. If pointerId is absent in this map then nobody is // capturing it. Additionally keep information about pending capturing content. static nsClassHashtable<nsUint32HashKey, nsIPresShell::PointerCaptureInfo>* sPointerCaptureList; // Keeps information about pointers such as pointerId, activeState, pointerType, // primaryState static nsClassHashtable<nsUint32HashKey, nsIPresShell::PointerInfo>* sActivePointersIds; // RangePaintInfo is used to paint ranges to offscreen buffers struct RangePaintInfo { RefPtr<nsRange> mRange; nsDisplayListBuilder mBuilder; nsDisplayList mList; // offset of builder's reference frame to the root frame nsPoint mRootOffset; RangePaintInfo(nsRange* aRange, nsIFrame* aFrame) : mRange(aRange), mBuilder(aFrame, nsDisplayListBuilderMode::PAINTING, false) { MOZ_COUNT_CTOR(RangePaintInfo); } ~RangePaintInfo() { mList.DeleteAll(); MOZ_COUNT_DTOR(RangePaintInfo); } }; #undef NOISY // ---------------------------------------------------------------------- #ifdef DEBUG // Set the environment variable GECKO_VERIFY_REFLOW_FLAGS to one or // more of the following flags (comma separated) for handy debug // output. static uint32_t gVerifyReflowFlags; struct VerifyReflowFlags { const char* name; uint32_t bit; }; static const VerifyReflowFlags gFlags[] = { { "verify", VERIFY_REFLOW_ON }, { "reflow", VERIFY_REFLOW_NOISY }, { "all", VERIFY_REFLOW_ALL }, { "list-commands", VERIFY_REFLOW_DUMP_COMMANDS }, { "noisy-commands", VERIFY_REFLOW_NOISY_RC }, { "really-noisy-commands", VERIFY_REFLOW_REALLY_NOISY_RC }, { "resize", VERIFY_REFLOW_DURING_RESIZE_REFLOW }, }; #define NUM_VERIFY_REFLOW_FLAGS (sizeof(gFlags) / sizeof(gFlags[0])) static void ShowVerifyReflowFlags() { printf("Here are the available GECKO_VERIFY_REFLOW_FLAGS:\n"); const VerifyReflowFlags* flag = gFlags; const VerifyReflowFlags* limit = gFlags + NUM_VERIFY_REFLOW_FLAGS; while (flag < limit) { printf(" %s\n", flag->name); ++flag; } printf("Note: GECKO_VERIFY_REFLOW_FLAGS is a comma separated list of flag\n"); printf("names (no whitespace)\n"); } #endif //======================================================================== //======================================================================== //======================================================================== #ifdef MOZ_REFLOW_PERF class ReflowCountMgr; static const char kGrandTotalsStr[] = "Grand Totals"; // Counting Class class ReflowCounter { public: explicit ReflowCounter(ReflowCountMgr * aMgr = nullptr); ~ReflowCounter(); void ClearTotals(); void DisplayTotals(const char * aStr); void DisplayDiffTotals(const char * aStr); void DisplayHTMLTotals(const char * aStr); void Add() { mTotal++; } void Add(uint32_t aTotal) { mTotal += aTotal; } void CalcDiffInTotals(); void SetTotalsCache(); void SetMgr(ReflowCountMgr * aMgr) { mMgr = aMgr; } uint32_t GetTotal() { return mTotal; } protected: void DisplayTotals(uint32_t aTotal, const char * aTitle); void DisplayHTMLTotals(uint32_t aTotal, const char * aTitle); uint32_t mTotal; uint32_t mCacheTotal; ReflowCountMgr * mMgr; // weak reference (don't delete) }; // Counting Class class IndiReflowCounter { public: explicit IndiReflowCounter(ReflowCountMgr * aMgr = nullptr) : mFrame(nullptr), mCount(0), mMgr(aMgr), mCounter(aMgr), mHasBeenOutput(false) {} virtual ~IndiReflowCounter() {} nsAutoString mName; nsIFrame * mFrame; // weak reference (don't delete) int32_t mCount; ReflowCountMgr * mMgr; // weak reference (don't delete) ReflowCounter mCounter; bool mHasBeenOutput; }; //-------------------- // Manager Class //-------------------- class ReflowCountMgr { public: ReflowCountMgr(); virtual ~ReflowCountMgr(); void ClearTotals(); void ClearGrandTotals(); void DisplayTotals(const char * aStr); void DisplayHTMLTotals(const char * aStr); void DisplayDiffsInTotals(); void Add(const char * aName, nsIFrame * aFrame); ReflowCounter * LookUp(const char * aName); void PaintCount(const char *aName, nsRenderingContext* aRenderingContext, nsPresContext *aPresContext, nsIFrame *aFrame, const nsPoint &aOffset, uint32_t aColor); FILE * GetOutFile() { return mFD; } PLHashTable * GetIndiFrameHT() { return mIndiFrameCounts; } void SetPresContext(nsPresContext * aPresContext) { mPresContext = aPresContext; } // weak reference void SetPresShell(nsIPresShell* aPresShell) { mPresShell= aPresShell; } // weak reference void SetDumpFrameCounts(bool aVal) { mDumpFrameCounts = aVal; } void SetDumpFrameByFrameCounts(bool aVal) { mDumpFrameByFrameCounts = aVal; } void SetPaintFrameCounts(bool aVal) { mPaintFrameByFrameCounts = aVal; } bool IsPaintingFrameCounts() { return mPaintFrameByFrameCounts; } protected: void DisplayTotals(uint32_t aTotal, uint32_t * aDupArray, char * aTitle); void DisplayHTMLTotals(uint32_t aTotal, uint32_t * aDupArray, char * aTitle); static int RemoveItems(PLHashEntry *he, int i, void *arg); static int RemoveIndiItems(PLHashEntry *he, int i, void *arg); void CleanUp(); // stdout Output Methods static int DoSingleTotal(PLHashEntry *he, int i, void *arg); static int DoSingleIndi(PLHashEntry *he, int i, void *arg); void DoGrandTotals(); void DoIndiTotalsTree(); // HTML Output Methods static int DoSingleHTMLTotal(PLHashEntry *he, int i, void *arg); void DoGrandHTMLTotals(); // Zero Out the Totals static int DoClearTotals(PLHashEntry *he, int i, void *arg); // Displays the Diff Totals static int DoDisplayDiffTotals(PLHashEntry *he, int i, void *arg); PLHashTable * mCounts; PLHashTable * mIndiFrameCounts; FILE * mFD; bool mDumpFrameCounts; bool mDumpFrameByFrameCounts; bool mPaintFrameByFrameCounts; bool mCycledOnce; // Root Frame for Individual Tracking nsPresContext * mPresContext; nsIPresShell* mPresShell; // ReflowCountMgr gReflowCountMgr; }; #endif //======================================================================== // comment out to hide caret #define SHOW_CARET // The upper bound on the amount of time to spend reflowing, in // microseconds. When this bound is exceeded and reflow commands are // still queued up, a reflow event is posted. The idea is for reflow // to not hog the processor beyond the time specifed in // gMaxRCProcessingTime. This data member is initialized from the // layout.reflow.timeslice pref. #define NS_MAX_REFLOW_TIME 1000000 static int32_t gMaxRCProcessingTime = -1; struct nsCallbackEventRequest { nsIReflowCallback* callback; nsCallbackEventRequest* next; }; // ---------------------------------------------------------------------------- #define ASSERT_REFLOW_SCHEDULED_STATE() \ NS_ASSERTION(mReflowScheduled == \ GetPresContext()->RefreshDriver()-> \ IsLayoutFlushObserver(this), "Unexpected state") class nsAutoCauseReflowNotifier { public: explicit nsAutoCauseReflowNotifier(PresShell* aShell) : mShell(aShell) { mShell->WillCauseReflow(); } ~nsAutoCauseReflowNotifier() { // This check should not be needed. Currently the only place that seem // to need it is the code that deals with bug 337586. if (!mShell->mHaveShutDown) { mShell->DidCauseReflow(); } else { nsContentUtils::RemoveScriptBlocker(); } } PresShell* mShell; }; class MOZ_STACK_CLASS nsPresShellEventCB : public EventDispatchingCallback { public: explicit nsPresShellEventCB(PresShell* aPresShell) : mPresShell(aPresShell) {} virtual void HandleEvent(EventChainPostVisitor& aVisitor) override { if (aVisitor.mPresContext && aVisitor.mEvent->mClass != eBasicEventClass) { if (aVisitor.mEvent->mMessage == eMouseDown || aVisitor.mEvent->mMessage == eMouseUp) { // Mouse-up and mouse-down events call nsFrame::HandlePress/Release // which call GetContentOffsetsFromPoint which requires up-to-date layout. // Bring layout up-to-date now so that GetCurrentEventFrame() below // will return a real frame and we don't have to worry about // destroying it by flushing later. mPresShell->FlushPendingNotifications(Flush_Layout); } else if (aVisitor.mEvent->mMessage == eWheel && aVisitor.mEventStatus != nsEventStatus_eConsumeNoDefault) { nsIFrame* frame = mPresShell->GetCurrentEventFrame(); if (frame) { // chrome (including addons) should be able to know if content // handles both D3E "wheel" event and legacy mouse scroll events. // We should dispatch legacy mouse events before dispatching the // "wheel" event into system group. RefPtr<EventStateManager> esm = aVisitor.mPresContext->EventStateManager(); esm->DispatchLegacyMouseScrollEvents(frame, aVisitor.mEvent->AsWheelEvent(), &aVisitor.mEventStatus); } } nsIFrame* frame = mPresShell->GetCurrentEventFrame(); if (!frame && (aVisitor.mEvent->mMessage == eMouseUp || aVisitor.mEvent->mMessage == eTouchEnd)) { // Redirect BUTTON_UP and TOUCH_END events to the root frame to ensure // that capturing is released. frame = mPresShell->GetRootFrame(); } if (frame) { frame->HandleEvent(aVisitor.mPresContext, aVisitor.mEvent->AsGUIEvent(), &aVisitor.mEventStatus); } } } RefPtr<PresShell> mPresShell; }; class nsBeforeFirstPaintDispatcher : public Runnable { public: explicit nsBeforeFirstPaintDispatcher(nsIDocument* aDocument) : mDocument(aDocument) {} // Fires the "before-first-paint" event so that interested parties (right now, the // mobile browser) are aware of it. NS_IMETHOD Run() override { nsCOMPtr<nsIObserverService> observerService = mozilla::services::GetObserverService(); if (observerService) { observerService->NotifyObservers(mDocument, "before-first-paint", nullptr); } return NS_OK; } private: nsCOMPtr<nsIDocument> mDocument; }; bool PresShell::sDisableNonTestMouseEvents = false; mozilla::LazyLogModule PresShell::gLog("PresShell"); #ifdef DEBUG static void VerifyStyleTree(nsPresContext* aPresContext, nsFrameManager* aFrameManager) { if (nsFrame::GetVerifyStyleTreeEnable()) { if (aPresContext->RestyleManager()->IsServo()) { NS_ERROR("stylo: cannot verify style tree with a ServoRestyleManager"); return; } nsIFrame* rootFrame = aFrameManager->GetRootFrame(); aPresContext->RestyleManager()->AsGecko()->DebugVerifyStyleTree(rootFrame); } } #define VERIFY_STYLE_TREE ::VerifyStyleTree(mPresContext, mFrameConstructor) #else #define VERIFY_STYLE_TREE #endif static bool gVerifyReflowEnabled; bool nsIPresShell::GetVerifyReflowEnable() { #ifdef DEBUG static bool firstTime = true; if (firstTime) { firstTime = false; char* flags = PR_GetEnv("GECKO_VERIFY_REFLOW_FLAGS"); if (flags) { bool error = false; for (;;) { char* comma = PL_strchr(flags, ','); if (comma) *comma = '\0'; bool found = false; const VerifyReflowFlags* flag = gFlags; const VerifyReflowFlags* limit = gFlags + NUM_VERIFY_REFLOW_FLAGS; while (flag < limit) { if (PL_strcasecmp(flag->name, flags) == 0) { gVerifyReflowFlags |= flag->bit; found = true; break; } ++flag; } if (! found) error = true; if (! comma) break; *comma = ','; flags = comma + 1; } if (error) ShowVerifyReflowFlags(); } if (VERIFY_REFLOW_ON & gVerifyReflowFlags) { gVerifyReflowEnabled = true; printf("Note: verifyreflow is enabled"); if (VERIFY_REFLOW_NOISY & gVerifyReflowFlags) { printf(" (noisy)"); } if (VERIFY_REFLOW_ALL & gVerifyReflowFlags) { printf(" (all)"); } if (VERIFY_REFLOW_DUMP_COMMANDS & gVerifyReflowFlags) { printf(" (show reflow commands)"); } if (VERIFY_REFLOW_NOISY_RC & gVerifyReflowFlags) { printf(" (noisy reflow commands)"); if (VERIFY_REFLOW_REALLY_NOISY_RC & gVerifyReflowFlags) { printf(" (REALLY noisy reflow commands)"); } } printf("\n"); } } #endif return gVerifyReflowEnabled; } void PresShell::AddInvalidateHiddenPresShellObserver(nsRefreshDriver *aDriver) { if (!mHiddenInvalidationObserverRefreshDriver && !mIsDestroying && !mHaveShutDown) { aDriver->AddPresShellToInvalidateIfHidden(this); mHiddenInvalidationObserverRefreshDriver = aDriver; } } void nsIPresShell::InvalidatePresShellIfHidden() { if (!IsVisible() && mPresContext) { mPresContext->NotifyInvalidation(0); } mHiddenInvalidationObserverRefreshDriver = nullptr; } void nsIPresShell::CancelInvalidatePresShellIfHidden() { if (mHiddenInvalidationObserverRefreshDriver) { mHiddenInvalidationObserverRefreshDriver->RemovePresShellToInvalidateIfHidden(this); mHiddenInvalidationObserverRefreshDriver = nullptr; } } void nsIPresShell::SetVerifyReflowEnable(bool aEnabled) { gVerifyReflowEnabled = aEnabled; } /* virtual */ void nsIPresShell::AddWeakFrameExternal(nsWeakFrame* aWeakFrame) { AddWeakFrameInternal(aWeakFrame); } void nsIPresShell::AddWeakFrameInternal(nsWeakFrame* aWeakFrame) { if (aWeakFrame->GetFrame()) { aWeakFrame->GetFrame()->AddStateBits(NS_FRAME_EXTERNAL_REFERENCE); } aWeakFrame->SetPreviousWeakFrame(mWeakFrames); mWeakFrames = aWeakFrame; } /* virtual */ void nsIPresShell::RemoveWeakFrameExternal(nsWeakFrame* aWeakFrame) { RemoveWeakFrameInternal(aWeakFrame); } void nsIPresShell::RemoveWeakFrameInternal(nsWeakFrame* aWeakFrame) { if (mWeakFrames == aWeakFrame) { mWeakFrames = aWeakFrame->GetPreviousWeakFrame(); return; } nsWeakFrame* nextWeak = mWeakFrames; while (nextWeak && nextWeak->GetPreviousWeakFrame() != aWeakFrame) { nextWeak = nextWeak->GetPreviousWeakFrame(); } if (nextWeak) { nextWeak->SetPreviousWeakFrame(aWeakFrame->GetPreviousWeakFrame()); } } already_AddRefed<nsFrameSelection> nsIPresShell::FrameSelection() { RefPtr<nsFrameSelection> ret = mSelection; return ret.forget(); } //---------------------------------------------------------------------- static bool sSynthMouseMove = true; static uint32_t sNextPresShellId; static bool sPointerEventEnabled = true; static bool sPointerEventImplicitCapture = false; static bool sAccessibleCaretEnabled = false; static bool sAccessibleCaretOnTouch = false; static bool sBeforeAfterKeyboardEventEnabled = false; /* static */ bool PresShell::AccessibleCaretEnabled(nsIDocShell* aDocShell) { static bool initialized = false; if (!initialized) { Preferences::AddBoolVarCache(&sAccessibleCaretEnabled, "layout.accessiblecaret.enabled"); Preferences::AddBoolVarCache(&sAccessibleCaretOnTouch, "layout.accessiblecaret.enabled_on_touch"); initialized = true; } // If the pref forces it on, then enable it. if (sAccessibleCaretEnabled) { return true; } // If the touch pref is on, and touch events are enabled (this depends // on the specific device running), then enable it. if (sAccessibleCaretOnTouch && dom::TouchEvent::PrefEnabled(aDocShell)) { return true; } // Otherwise, disabled. return false; } /* static */ bool PresShell::BeforeAfterKeyboardEventEnabled() { static bool sInitialized = false; if (!sInitialized) { Preferences::AddBoolVarCache(&sBeforeAfterKeyboardEventEnabled, "dom.beforeAfterKeyboardEvent.enabled"); sInitialized = true; } return sBeforeAfterKeyboardEventEnabled; } /* static */ bool PresShell::IsTargetIframe(nsINode* aTarget) { return aTarget && aTarget->IsHTMLElement(nsGkAtoms::iframe); } PresShell::PresShell() : mMouseLocation(NS_UNCONSTRAINEDSIZE, NS_UNCONSTRAINEDSIZE) { #ifdef MOZ_REFLOW_PERF mReflowCountMgr = new ReflowCountMgr(); mReflowCountMgr->SetPresContext(mPresContext); mReflowCountMgr->SetPresShell(this); #endif mLoadBegin = TimeStamp::Now(); mSelectionFlags = nsISelectionDisplay::DISPLAY_TEXT | nsISelectionDisplay::DISPLAY_IMAGES; mIsThemeSupportDisabled = false; mIsActive = true; // FIXME/bug 735029: find a better solution to this problem mIsFirstPaint = true; mPresShellId = sNextPresShellId++; mFrozen = false; mRenderFlags = 0; mScrollPositionClampingScrollPortSizeSet = false; static bool addedSynthMouseMove = false; if (!addedSynthMouseMove) { Preferences::AddBoolVarCache(&sSynthMouseMove, "layout.reflow.synthMouseMove", true); addedSynthMouseMove = true; } static bool addedPointerEventEnabled = false; if (!addedPointerEventEnabled) { Preferences::AddBoolVarCache(&sPointerEventEnabled, "dom.w3c_pointer_events.enabled", true); addedPointerEventEnabled = true; } static bool addedPointerEventImplicitCapture = false; if (!addedPointerEventImplicitCapture) { Preferences::AddBoolVarCache(&sPointerEventImplicitCapture, "dom.w3c_pointer_events.implicit_capture", true); addedPointerEventImplicitCapture = true; } mPaintingIsFrozen = false; mHasCSSBackgroundColor = true; mIsLastChromeOnlyEscapeKeyConsumed = false; mHasReceivedPaintMessage = false; } NS_IMPL_ISUPPORTS(PresShell, nsIPresShell, nsIDocumentObserver, nsISelectionController, nsISelectionDisplay, nsIObserver, nsISupportsWeakReference, nsIMutationObserver) PresShell::~PresShell() { if (!mHaveShutDown) { NS_NOTREACHED("Someone did not call nsIPresShell::destroy"); Destroy(); } NS_ASSERTION(mCurrentEventContentStack.Count() == 0, "Huh, event content left on the stack in pres shell dtor!"); NS_ASSERTION(mFirstCallbackEventRequest == nullptr && mLastCallbackEventRequest == nullptr, "post-reflow queues not empty. This means we're leaking"); // Verify that if painting was frozen, but we're being removed from the tree, // that we now re-enable painting on our refresh driver, since it may need to // be re-used by another presentation. if (mPaintingIsFrozen) { mPresContext->RefreshDriver()->Thaw(); } MOZ_ASSERT(mAllocatedPointers.IsEmpty(), "Some pres arena objects were not freed"); mStyleSet->Delete(); delete mFrameConstructor; mCurrentEventContent = nullptr; } /** * Initialize the presentation shell. Create view manager and style * manager. * Note this can't be merged into our constructor because caret initialization * calls AddRef() on us. */ void PresShell::Init(nsIDocument* aDocument, nsPresContext* aPresContext, nsViewManager* aViewManager, StyleSetHandle aStyleSet) { NS_PRECONDITION(aDocument, "null ptr"); NS_PRECONDITION(aPresContext, "null ptr"); NS_PRECONDITION(aViewManager, "null ptr"); NS_PRECONDITION(!mDocument, "already initialized"); if (!aDocument || !aPresContext || !aViewManager || mDocument) { return; } mDocument = aDocument; mViewManager = aViewManager; // Create our frame constructor. mFrameConstructor = new nsCSSFrameConstructor(mDocument, this); mFrameManager = mFrameConstructor; // The document viewer owns both view manager and pres shell. mViewManager->SetPresShell(this); // Bind the context to the presentation shell. mPresContext = aPresContext; StyleBackendType backend = aStyleSet->IsServo() ? StyleBackendType::Servo : StyleBackendType::Gecko; aPresContext->AttachShell(this, backend); // Now we can initialize the style set. Make sure to set the member before // calling Init, since various subroutines need to find the style set off // the PresContext during initialization. mStyleSet = aStyleSet; mStyleSet->Init(aPresContext); // Notify our prescontext that it now has a compatibility mode. Note that // this MUST happen after we set up our style set but before we create any // frames. mPresContext->CompatibilityModeChanged(); // Add the preference style sheet. UpdatePreferenceStyles(); if (AccessibleCaretEnabled(mDocument->GetDocShell())) { // Need to happen before nsFrameSelection has been set up. mAccessibleCaretEventHub = new AccessibleCaretEventHub(this); } mSelection = new nsFrameSelection(); RefPtr<nsFrameSelection> frameSelection = mSelection; frameSelection->Init(this, nullptr); // Important: this has to happen after the selection has been set up #ifdef SHOW_CARET // make the caret mCaret = new nsCaret(); mCaret->Init(this); mOriginalCaret = mCaret; //SetCaretEnabled(true); // make it show in browser windows #endif //set up selection to be displayed in document // Don't enable selection for print media nsPresContext::nsPresContextType type = aPresContext->Type(); if (type != nsPresContext::eContext_PrintPreview && type != nsPresContext::eContext_Print) SetDisplaySelection(nsISelectionController::SELECTION_DISABLED); if (gMaxRCProcessingTime == -1) { gMaxRCProcessingTime = Preferences::GetInt("layout.reflow.timeslice", NS_MAX_REFLOW_TIME); } { nsCOMPtr<nsIObserverService> os = mozilla::services::GetObserverService(); if (os) { os->AddObserver(this, "agent-sheet-added", false); os->AddObserver(this, "user-sheet-added", false); os->AddObserver(this, "author-sheet-added", false); os->AddObserver(this, "agent-sheet-removed", false); os->AddObserver(this, "user-sheet-removed", false); os->AddObserver(this, "author-sheet-removed", false); #ifdef MOZ_XUL os->AddObserver(this, "chrome-flush-skin-caches", false); #endif os->AddObserver(this, "memory-pressure", false); } } #ifdef MOZ_REFLOW_PERF if (mReflowCountMgr) { bool paintFrameCounts = Preferences::GetBool("layout.reflow.showframecounts"); bool dumpFrameCounts = Preferences::GetBool("layout.reflow.dumpframecounts"); bool dumpFrameByFrameCounts = Preferences::GetBool("layout.reflow.dumpframebyframecounts"); mReflowCountMgr->SetDumpFrameCounts(dumpFrameCounts); mReflowCountMgr->SetDumpFrameByFrameCounts(dumpFrameByFrameCounts); mReflowCountMgr->SetPaintFrameCounts(paintFrameCounts); } #endif if (mDocument->HasAnimationController()) { nsSMILAnimationController* animCtrl = mDocument->GetAnimationController(); animCtrl->NotifyRefreshDriverCreated(GetPresContext()->RefreshDriver()); } for (DocumentTimeline* timeline : mDocument->Timelines()) { timeline->NotifyRefreshDriverCreated(GetPresContext()->RefreshDriver()); } // Get our activeness from the docShell. QueryIsActive(); // Setup our font inflation preferences. SetupFontInflation(); mTouchManager.Init(this, mDocument); if (mPresContext->IsRootContentDocument()) { mZoomConstraintsClient = new ZoomConstraintsClient(); mZoomConstraintsClient->Init(this, mDocument); if (gfxPrefs::MetaViewportEnabled() || gfxPrefs::APZAllowZooming()) { mMobileViewportManager = new MobileViewportManager(this, mDocument); } } } enum TextPerfLogType { eLog_reflow, eLog_loaddone, eLog_totals }; static void LogTextPerfStats(gfxTextPerfMetrics* aTextPerf, PresShell* aPresShell, const gfxTextPerfMetrics::TextCounts& aCounts, float aTime, TextPerfLogType aLogType, const char* aURL) { LogModule* tpLog = gfxPlatform::GetLog(eGfxLog_textperf); // ignore XUL contexts unless at debug level mozilla::LogLevel logLevel = LogLevel::Warning; if (aCounts.numContentTextRuns == 0) { logLevel = LogLevel::Debug; } if (!MOZ_LOG_TEST(tpLog, logLevel)) { return; } char prefix[256]; switch (aLogType) { case eLog_reflow: SprintfLiteral(prefix, "(textperf-reflow) %p time-ms: %7.0f", aPresShell, aTime); break; case eLog_loaddone: SprintfLiteral(prefix, "(textperf-loaddone) %p time-ms: %7.0f", aPresShell, aTime); break; default: MOZ_ASSERT(aLogType == eLog_totals, "unknown textperf log type"); SprintfLiteral(prefix, "(textperf-totals) %p", aPresShell); } double hitRatio = 0.0; uint32_t lookups = aCounts.wordCacheHit + aCounts.wordCacheMiss; if (lookups) { hitRatio = double(aCounts.wordCacheHit) / double(lookups); } if (aLogType == eLog_loaddone) { MOZ_LOG(tpLog, logLevel, ("%s reflow: %d chars: %d " "[%s] " "content-textruns: %d chrome-textruns: %d " "max-textrun-len: %d " "word-cache-lookups: %d word-cache-hit-ratio: %4.3f " "word-cache-space: %d word-cache-long: %d " "pref-fallbacks: %d system-fallbacks: %d " "textruns-const: %d textruns-destr: %d " "generic-lookups: %d " "cumulative-textruns-destr: %d\n", prefix, aTextPerf->reflowCount, aCounts.numChars, (aURL ? aURL : ""), aCounts.numContentTextRuns, aCounts.numChromeTextRuns, aCounts.maxTextRunLen, lookups, hitRatio, aCounts.wordCacheSpaceRules, aCounts.wordCacheLong, aCounts.fallbackPrefs, aCounts.fallbackSystem, aCounts.textrunConst, aCounts.textrunDestr, aCounts.genericLookups, aTextPerf->cumulative.textrunDestr)); } else { MOZ_LOG(tpLog, logLevel, ("%s reflow: %d chars: %d " "content-textruns: %d chrome-textruns: %d " "max-textrun-len: %d " "word-cache-lookups: %d word-cache-hit-ratio: %4.3f " "word-cache-space: %d word-cache-long: %d " "pref-fallbacks: %d system-fallbacks: %d " "textruns-const: %d textruns-destr: %d " "generic-lookups: %d " "cumulative-textruns-destr: %d\n", prefix, aTextPerf->reflowCount, aCounts.numChars, aCounts.numContentTextRuns, aCounts.numChromeTextRuns, aCounts.maxTextRunLen, lookups, hitRatio, aCounts.wordCacheSpaceRules, aCounts.wordCacheLong, aCounts.fallbackPrefs, aCounts.fallbackSystem, aCounts.textrunConst, aCounts.textrunDestr, aCounts.genericLookups, aTextPerf->cumulative.textrunDestr)); } } void PresShell::Destroy() { // Do not add code before this line please! if (mHaveShutDown) { return; } NS_ASSERTION(!nsContentUtils::IsSafeToRunScript(), "destroy called on presshell while scripts not blocked"); // dump out cumulative text perf metrics gfxTextPerfMetrics* tp; if (mPresContext && (tp = mPresContext->GetTextPerfMetrics())) { tp->Accumulate(); if (tp->cumulative.numChars > 0) { LogTextPerfStats(tp, this, tp->cumulative, 0.0, eLog_totals, nullptr); } } if (mPresContext) { const bool mayFlushUserFontSet = false; gfxUserFontSet* fs = mPresContext->GetUserFontSet(mayFlushUserFontSet); if (fs) { uint32_t fontCount; uint64_t fontSize; fs->GetLoadStatistics(fontCount, fontSize); Telemetry::Accumulate(Telemetry::WEBFONT_PER_PAGE, fontCount); Telemetry::Accumulate(Telemetry::WEBFONT_SIZE_PER_PAGE, uint32_t(fontSize/1024)); } else { Telemetry::Accumulate(Telemetry::WEBFONT_PER_PAGE, 0); Telemetry::Accumulate(Telemetry::WEBFONT_SIZE_PER_PAGE, 0); } } #ifdef MOZ_REFLOW_PERF DumpReflows(); if (mReflowCountMgr) { delete mReflowCountMgr; mReflowCountMgr = nullptr; } #endif if (mZoomConstraintsClient) { mZoomConstraintsClient->Destroy(); mZoomConstraintsClient = nullptr; } if (mMobileViewportManager) { mMobileViewportManager->Destroy(); mMobileViewportManager = nullptr; } #ifdef ACCESSIBILITY if (mDocAccessible) { #ifdef DEBUG if (a11y::logging::IsEnabled(a11y::logging::eDocDestroy)) a11y::logging::DocDestroy("presshell destroyed", mDocument); #endif mDocAccessible->Shutdown(); mDocAccessible = nullptr; } #endif // ACCESSIBILITY MaybeReleaseCapturingContent(); if (gKeyDownTarget && gKeyDownTarget->OwnerDoc() == mDocument) { NS_RELEASE(gKeyDownTarget); } if (mContentToScrollTo) { mContentToScrollTo->DeleteProperty(nsGkAtoms::scrolling); mContentToScrollTo = nullptr; } if (mPresContext) { // We need to notify the destroying the nsPresContext to ESM for // suppressing to use from ESM. mPresContext->EventStateManager()->NotifyDestroyPresContext(mPresContext); } { nsCOMPtr<nsIObserverService> os = mozilla::services::GetObserverService(); if (os) { os->RemoveObserver(this, "agent-sheet-added"); os->RemoveObserver(this, "user-sheet-added"); os->RemoveObserver(this, "author-sheet-added"); os->RemoveObserver(this, "agent-sheet-removed"); os->RemoveObserver(this, "user-sheet-removed"); os->RemoveObserver(this, "author-sheet-removed"); #ifdef MOZ_XUL os->RemoveObserver(this, "chrome-flush-skin-caches"); #endif os->RemoveObserver(this, "memory-pressure"); } } // If our paint suppression timer is still active, kill it. if (mPaintSuppressionTimer) { mPaintSuppressionTimer->Cancel(); mPaintSuppressionTimer = nullptr; } // Same for our reflow continuation timer if (mReflowContinueTimer) { mReflowContinueTimer->Cancel(); mReflowContinueTimer = nullptr; } if (mDelayedPaintTimer) { mDelayedPaintTimer->Cancel(); mDelayedPaintTimer = nullptr; } mSynthMouseMoveEvent.Revoke(); mUpdateApproximateFrameVisibilityEvent.Revoke(); ClearApproximatelyVisibleFramesList(Some(OnNonvisible::DISCARD_IMAGES)); if (mCaret) { mCaret->Terminate(); mCaret = nullptr; } if (mSelection) { RefPtr<nsFrameSelection> frameSelection = mSelection; frameSelection->DisconnectFromPresShell(); } if (mAccessibleCaretEventHub) { mAccessibleCaretEventHub->Terminate(); mAccessibleCaretEventHub = nullptr; } // release our pref style sheet, if we have one still RemovePreferenceStyles(); mIsDestroying = true; // We can't release all the event content in // mCurrentEventContentStack here since there might be code on the // stack that will release the event content too. Double release // bad! // The frames will be torn down, so remove them from the current // event frame stack (since they'd be dangling references if we'd // leave them in) and null out the mCurrentEventFrame pointer as // well. mCurrentEventFrame = nullptr; int32_t i, count = mCurrentEventFrameStack.Length(); for (i = 0; i < count; i++) { mCurrentEventFrameStack[i] = nullptr; } mFramesToDirty.Clear(); if (mViewManager) { // Clear the view manager's weak pointer back to |this| in case it // was leaked. mViewManager->SetPresShell(nullptr); mViewManager = nullptr; } // mFrameArena will be destroyed soon. Clear out any ArenaRefPtrs // pointing to objects in the arena now. This is done: // // (a) before mFrameArena's destructor runs so that our // mAllocatedPointers becomes empty and doesn't trip the assertion // in ~PresShell, // (b) before the mPresContext->DetachShell() below, so // that when we clear the ArenaRefPtrs they'll still be able to // get back to this PresShell to deregister themselves (e.g. note // how nsStyleContext::Arena returns the PresShell got from its // rule node's nsPresContext, which would return null if we'd already // called mPresContext->DetachShell()), and // (c) before the mStyleSet->BeginShutdown() call just below, so that // the nsStyleContexts don't complain they're being destroyed later // than the rule tree is. mFrameArena.ClearArenaRefPtrs(); mStyleSet->BeginShutdown(); nsRefreshDriver* rd = GetPresContext()->RefreshDriver(); // This shell must be removed from the document before the frame // hierarchy is torn down to avoid finding deleted frames through // this presshell while the frames are being torn down if (mDocument) { NS_ASSERTION(mDocument->GetShell() == this, "Wrong shell?"); mDocument->DeleteShell(); if (mDocument->HasAnimationController()) { mDocument->GetAnimationController()->NotifyRefreshDriverDestroying(rd); } for (DocumentTimeline* timeline : mDocument->Timelines()) { timeline->NotifyRefreshDriverDestroying(rd); } } if (mPresContext) { mPresContext->AnimationManager()->ClearEventQueue(); mPresContext->TransitionManager()->ClearEventQueue(); } // Revoke any pending events. We need to do this and cancel pending reflows // before we destroy the frame manager, since apparently frame destruction // sometimes spins the event queue when plug-ins are involved(!). rd->RemoveLayoutFlushObserver(this); if (mHiddenInvalidationObserverRefreshDriver) { mHiddenInvalidationObserverRefreshDriver->RemovePresShellToInvalidateIfHidden(this); } if (rd->GetPresContext() == GetPresContext()) { rd->RevokeViewManagerFlush(); } mResizeEvent.Revoke(); if (mAsyncResizeTimerIsActive) { mAsyncResizeEventTimer->Cancel(); mAsyncResizeTimerIsActive = false; } CancelAllPendingReflows(); CancelPostedReflowCallbacks(); // Destroy the frame manager. This will destroy the frame hierarchy mFrameConstructor->WillDestroyFrameTree(); // Destroy all frame properties (whose destruction was suppressed // while destroying the frame tree, but which might contain more // frames within the properties. if (mPresContext) { // Clear out the prescontext's property table -- since our frame tree is // now dead, we shouldn't be looking up any more properties in that table. // We want to do this before we call DetachShell() on the prescontext, so // property destructors can usefully call GetPresShell() on the // prescontext. mPresContext->PropertyTable()->DeleteAll(); } NS_WARNING_ASSERTION(!mWeakFrames, "Weak frames alive after destroying FrameManager"); while (mWeakFrames) { mWeakFrames->Clear(this); } // Let the style set do its cleanup. mStyleSet->Shutdown(); if (mPresContext) { // We hold a reference to the pres context, and it holds a weak link back // to us. To avoid the pres context having a dangling reference, set its // pres shell to nullptr mPresContext->DetachShell(); // Clear the link handler (weak reference) as well mPresContext->SetLinkHandler(nullptr); } mHaveShutDown = true; mTouchManager.Destroy(); } void PresShell::MakeZombie() { mIsZombie = true; CancelAllPendingReflows(); } nsRefreshDriver* nsIPresShell::GetRefreshDriver() const { return mPresContext ? mPresContext->RefreshDriver() : nullptr; } void nsIPresShell::SetAuthorStyleDisabled(bool aStyleDisabled) { if (aStyleDisabled != mStyleSet->GetAuthorStyleDisabled()) { mStyleSet->SetAuthorStyleDisabled(aStyleDisabled); RestyleForCSSRuleChanges(); nsCOMPtr<nsIObserverService> observerService = mozilla::services::GetObserverService(); if (observerService) { observerService->NotifyObservers(mDocument, "author-style-disabled-changed", nullptr); } } } bool nsIPresShell::GetAuthorStyleDisabled() const { return mStyleSet->GetAuthorStyleDisabled(); } void PresShell::UpdatePreferenceStyles() { if (!mDocument) { return; } // If the document doesn't have a window there's no need to notify // its presshell about changes to preferences since the document is // in a state where it doesn't matter any more (see // nsDocumentViewer::Close()). if (!mDocument->GetWindow()) { return; } // Documents in chrome shells do not have any preference style rules applied. if (nsContentUtils::IsInChromeDocshell(mDocument)) { return; } // We need to pass in mPresContext so that if the nsLayoutStylesheetCache // needs to recreate the pref style sheet, it has somewhere to get the // pref styling information from. All pres contexts for // IsChromeOriginImage() == false will have the same pref styling information, // and similarly for IsChromeOriginImage() == true, so it doesn't really // matter which pres context we pass in when it does need to be recreated. // (See nsPresContext::GetDocumentColorPreferences for how whether we // are a chrome origin image affects some pref styling information.) auto cache = nsLayoutStylesheetCache::For(mStyleSet->BackendType()); RefPtr<StyleSheet> newPrefSheet = mPresContext->IsChromeOriginImage() ? cache->ChromePreferenceSheet(mPresContext) : cache->ContentPreferenceSheet(mPresContext); if (mPrefStyleSheet == newPrefSheet) { return; } mStyleSet->BeginUpdate(); RemovePreferenceStyles(); mStyleSet->AppendStyleSheet(SheetType::User, newPrefSheet); mPrefStyleSheet = newPrefSheet; mStyleSet->EndUpdate(); } void PresShell::RemovePreferenceStyles() { if (mPrefStyleSheet) { mStyleSet->RemoveStyleSheet(SheetType::User, mPrefStyleSheet); mPrefStyleSheet = nullptr; } } void PresShell::AddUserSheet(nsISupports* aSheet) { if (mStyleSet->IsServo()) { NS_ERROR("stylo: nsStyleSheetService doesn't handle ServoStyleSheets yet"); return; } // Make sure this does what nsDocumentViewer::CreateStyleSet does wrt // ordering. We want this new sheet to come after all the existing stylesheet // service sheets, but before other user sheets; see nsIStyleSheetService.idl // for the ordering. Just remove and readd all the nsStyleSheetService // sheets. nsCOMPtr<nsIStyleSheetService> dummy = do_GetService(NS_STYLESHEETSERVICE_CONTRACTID); mStyleSet->BeginUpdate(); nsStyleSheetService* sheetService = nsStyleSheetService::gInstance; nsTArray<RefPtr<StyleSheet>>& userSheets = *sheetService->UserStyleSheets(); // Iterate forwards when removing so the searches for RemoveStyleSheet are as // short as possible. for (StyleSheet* sheet : userSheets) { mStyleSet->RemoveStyleSheet(SheetType::User, sheet); } // Now iterate backwards, so that the order of userSheets will be the same as // the order of sheets from it in the style set. for (StyleSheet* sheet : Reversed(userSheets)) { mStyleSet->PrependStyleSheet(SheetType::User, sheet); } mStyleSet->EndUpdate(); RestyleForCSSRuleChanges(); } void PresShell::AddAgentSheet(nsISupports* aSheet) { // Make sure this does what nsDocumentViewer::CreateStyleSet does // wrt ordering. // XXXheycam This needs to work with ServoStyleSheets too. RefPtr<CSSStyleSheet> sheet = do_QueryObject(aSheet); if (!sheet) { NS_ERROR("stylo: AddAgentSheet needs to support ServoStyleSheets"); return; } mStyleSet->AppendStyleSheet(SheetType::Agent, sheet); RestyleForCSSRuleChanges(); } void PresShell::AddAuthorSheet(nsISupports* aSheet) { // XXXheycam This needs to work with ServoStyleSheets too. RefPtr<CSSStyleSheet> sheet = do_QueryObject(aSheet); if (!sheet) { NS_ERROR("stylo: AddAuthorSheet needs to support ServoStyleSheets"); return; } // Document specific "additional" Author sheets should be stronger than the // ones added with the StyleSheetService. StyleSheet* firstAuthorSheet = mDocument->GetFirstAdditionalAuthorSheet(); if (firstAuthorSheet) { mStyleSet->InsertStyleSheetBefore(SheetType::Doc, sheet, firstAuthorSheet); } else { mStyleSet->AppendStyleSheet(SheetType::Doc, sheet); } RestyleForCSSRuleChanges(); } void PresShell::RemoveSheet(SheetType aType, nsISupports* aSheet) { RefPtr<CSSStyleSheet> sheet = do_QueryObject(aSheet); if (!sheet) { NS_ERROR("stylo: RemoveSheet needs to support ServoStyleSheets"); return; } mStyleSet->RemoveStyleSheet(aType, sheet); RestyleForCSSRuleChanges(); } NS_IMETHODIMP PresShell::SetDisplaySelection(int16_t aToggle) { RefPtr<nsFrameSelection> frameSelection = mSelection; frameSelection->SetDisplaySelection(aToggle); return NS_OK; } NS_IMETHODIMP PresShell::GetDisplaySelection(int16_t *aToggle) { RefPtr<nsFrameSelection> frameSelection = mSelection; *aToggle = frameSelection->GetDisplaySelection(); return NS_OK; } NS_IMETHODIMP PresShell::GetSelection(RawSelectionType aRawSelectionType, nsISelection **aSelection) { if (!aSelection || !mSelection) return NS_ERROR_NULL_POINTER; RefPtr<nsFrameSelection> frameSelection = mSelection; nsCOMPtr<nsISelection> selection = frameSelection->GetSelection(ToSelectionType(aRawSelectionType)); if (!selection) { return NS_ERROR_INVALID_ARG; } selection.forget(aSelection); return NS_OK; } Selection* PresShell::GetCurrentSelection(SelectionType aSelectionType) { if (!mSelection) return nullptr; RefPtr<nsFrameSelection> frameSelection = mSelection; return frameSelection->GetSelection(aSelectionType); } already_AddRefed<nsISelectionController> PresShell::GetSelectionControllerForFocusedContent(nsIContent** aFocusedContent) { if (aFocusedContent) { *aFocusedContent = nullptr; } if (mDocument) { nsCOMPtr<nsPIDOMWindowOuter> focusedWindow; nsCOMPtr<nsIContent> focusedContent = nsFocusManager::GetFocusedDescendant(mDocument->GetWindow(), false, getter_AddRefs(focusedWindow)); if (focusedContent) { nsIFrame* frame = focusedContent->GetPrimaryFrame(); if (frame) { nsCOMPtr<nsISelectionController> selectionController; frame->GetSelectionController(mPresContext, getter_AddRefs(selectionController)); if (selectionController) { if (aFocusedContent) { focusedContent.forget(aFocusedContent); } return selectionController.forget(); } } } } nsCOMPtr<nsISelectionController> self(this); return self.forget(); } NS_IMETHODIMP PresShell::ScrollSelectionIntoView(RawSelectionType aRawSelectionType, SelectionRegion aRegion, int16_t aFlags) { if (!mSelection) return NS_ERROR_NULL_POINTER; RefPtr<nsFrameSelection> frameSelection = mSelection; return frameSelection->ScrollSelectionIntoView( ToSelectionType(aRawSelectionType), aRegion, aFlags); } NS_IMETHODIMP PresShell::RepaintSelection(RawSelectionType aRawSelectionType) { if (!mSelection) return NS_ERROR_NULL_POINTER; RefPtr<nsFrameSelection> frameSelection = mSelection; return frameSelection->RepaintSelection(ToSelectionType(aRawSelectionType)); } // Make shell be a document observer void PresShell::BeginObservingDocument() { if (mDocument && !mIsDestroying) { mDocument->AddObserver(this); if (mIsDocumentGone) { NS_WARNING("Adding a presshell that was disconnected from the document " "as a document observer? Sounds wrong..."); mIsDocumentGone = false; } } } // Make shell stop being a document observer void PresShell::EndObservingDocument() { // XXXbz do we need to tell the frame constructor that the document // is gone, perhaps? Except for printing it's NOT gone, sometimes. mIsDocumentGone = true; if (mDocument) { mDocument->RemoveObserver(this); } } #ifdef DEBUG_kipp char* nsPresShell_ReflowStackPointerTop; #endif class XBLConstructorRunner : public Runnable { public: explicit XBLConstructorRunner(nsIDocument* aDocument) : mDocument(aDocument) { } NS_IMETHOD Run() override { mDocument->BindingManager()->ProcessAttachedQueue(); return NS_OK; } private: nsCOMPtr<nsIDocument> mDocument; }; nsresult PresShell::Initialize(nscoord aWidth, nscoord aHeight) { if (mIsDestroying) { return NS_OK; } if (!mDocument) { // Nothing to do return NS_OK; } NS_ASSERTION(!mDidInitialize, "Why are we being called?"); nsCOMPtr<nsIPresShell> kungFuDeathGrip(this); mDidInitialize = true; #ifdef DEBUG if (VERIFY_REFLOW_NOISY_RC & gVerifyReflowFlags) { if (mDocument) { nsIURI *uri = mDocument->GetDocumentURI(); if (uri) { printf("*** PresShell::Initialize (this=%p, url='%s')\n", (void*)this, uri->GetSpecOrDefault().get()); } } } #endif // XXX Do a full invalidate at the beginning so that invalidates along // the way don't have region accumulation issues? mPresContext->SetVisibleArea(nsRect(0, 0, aWidth, aHeight)); if (mStyleSet->IsServo() && mDocument->GetRootElement()) { // If we have the root element already, go ahead style it along with any // descendants. // // Some things, like nsDocumentViewer::GetPageMode, recreate the PresShell // while keeping the content tree alive (see bug 1292280) - so we // unconditionally mark the root as dirty. mDocument->GetRootElement()->SetIsDirtyForServo(); mStyleSet->AsServo()->StyleDocument(/* aLeaveDirtyBits = */ false); } // Get the root frame from the frame manager // XXXbz it would be nice to move this somewhere else... like frame manager // Init(), say. But we need to make sure our views are all set up by the // time we do this! nsIFrame* rootFrame = mFrameConstructor->GetRootFrame(); NS_ASSERTION(!rootFrame, "How did that happen, exactly?"); if (!rootFrame) { nsAutoScriptBlocker scriptBlocker; mFrameConstructor->BeginUpdate(); rootFrame = mFrameConstructor->ConstructRootFrame(); mFrameConstructor->SetRootFrame(rootFrame); mFrameConstructor->EndUpdate(); } NS_ENSURE_STATE(!mHaveShutDown); if (!rootFrame) { return NS_ERROR_OUT_OF_MEMORY; } nsIFrame* invalidateFrame = nullptr; for (nsIFrame* f = rootFrame; f; f = nsLayoutUtils::GetCrossDocParentFrame(f)) { if (f->GetStateBits() & NS_FRAME_NO_COMPONENT_ALPHA) { invalidateFrame = f; f->RemoveStateBits(NS_FRAME_NO_COMPONENT_ALPHA); } nsCOMPtr<nsIPresShell> shell; if (f->GetType() == nsGkAtoms::subDocumentFrame && (shell = static_cast<nsSubDocumentFrame*>(f)->GetSubdocumentPresShellForPainting(0)) && shell->GetPresContext()->IsRootContentDocument()) { // Root content documents build a 'force active' layer, and component alpha flattening // can't be propagated across that so no need to invalidate above this frame. break; } } if (invalidateFrame) { invalidateFrame->InvalidateFrameSubtree(); } Element *root = mDocument->GetRootElement(); if (root) { { nsAutoCauseReflowNotifier reflowNotifier(this); mFrameConstructor->BeginUpdate(); // Have the style sheet processor construct frame for the root // content object down mFrameConstructor->ContentInserted(nullptr, root, nullptr, false); VERIFY_STYLE_TREE; // Something in mFrameConstructor->ContentInserted may have caused // Destroy() to get called, bug 337586. NS_ENSURE_STATE(!mHaveShutDown); mFrameConstructor->EndUpdate(); } // nsAutoCauseReflowNotifier (which sets up a script blocker) going out of // scope may have killed us too NS_ENSURE_STATE(!mHaveShutDown); // Run the XBL binding constructors for any new frames we've constructed. // (Do this in a script runner, since our caller might have a script // blocker on the stack.) nsContentUtils::AddScriptRunner(new XBLConstructorRunner(mDocument)); } NS_ASSERTION(rootFrame, "How did that happen?"); // Note: when the frame was created above it had the NS_FRAME_IS_DIRTY bit // set, but XBL processing could have caused a reflow which clears it. if (MOZ_LIKELY(rootFrame->GetStateBits() & NS_FRAME_IS_DIRTY)) { // Unset the DIRTY bits so that FrameNeedsReflow() will work right. rootFrame->RemoveStateBits(NS_FRAME_IS_DIRTY | NS_FRAME_HAS_DIRTY_CHILDREN); NS_ASSERTION(!mDirtyRoots.Contains(rootFrame), "Why is the root in mDirtyRoots already?"); FrameNeedsReflow(rootFrame, nsIPresShell::eResize, NS_FRAME_IS_DIRTY); NS_ASSERTION(mDirtyRoots.Contains(rootFrame), "Should be in mDirtyRoots now"); NS_ASSERTION(mReflowScheduled, "Why no reflow scheduled?"); } // Restore our root scroll position now if we're getting here after EndLoad // got called, since this is our one chance to do it. Note that we need not // have reflowed for this to work; when the scrollframe is finally reflowed // it'll pick up the position we store in it here. if (!mDocumentLoading) { RestoreRootScrollPosition(); } // For printing, we just immediately unsuppress. if (!mPresContext->IsPaginated()) { // Kick off a one-shot timer based off our pref value. When this timer // fires, if painting is still locked down, then we will go ahead and // trigger a full invalidate and allow painting to proceed normally. mPaintingSuppressed = true; // Don't suppress painting if the document isn't loading. nsIDocument::ReadyState readyState = mDocument->GetReadyStateEnum(); if (readyState != nsIDocument::READYSTATE_COMPLETE) { mPaintSuppressionTimer = do_CreateInstance("@mozilla.org/timer;1"); } if (!mPaintSuppressionTimer) { mPaintingSuppressed = false; } else { // Initialize the timer. // Default to PAINTLOCK_EVENT_DELAY if we can't get the pref value. int32_t delay = Preferences::GetInt("nglayout.initialpaint.delay", PAINTLOCK_EVENT_DELAY); mPaintSuppressionTimer->InitWithNamedFuncCallback( sPaintSuppressionCallback, this, delay, nsITimer::TYPE_ONE_SHOT, "PresShell::sPaintSuppressionCallback"); } } // If we get here and painting is not suppressed, then we can paint anytime // and we should fire the before-first-paint notification if (!mPaintingSuppressed) { ScheduleBeforeFirstPaint(); } return NS_OK; //XXX this needs to be real. MMP } void PresShell::sPaintSuppressionCallback(nsITimer *aTimer, void* aPresShell) { RefPtr<PresShell> self = static_cast<PresShell*>(aPresShell); if (self) self->UnsuppressPainting(); } void PresShell::AsyncResizeEventCallback(nsITimer* aTimer, void* aPresShell) { static_cast<PresShell*>(aPresShell)->FireResizeEvent(); } nsresult PresShell::ResizeReflow(nscoord aWidth, nscoord aHeight, nscoord aOldWidth, nscoord aOldHeight) { if (mZoomConstraintsClient) { // If we have a ZoomConstraintsClient and the available screen area // changed, then we might need to disable double-tap-to-zoom, so notify // the ZCC to update itself. mZoomConstraintsClient->ScreenSizeChanged(); } if (mMobileViewportManager) { // If we have a mobile viewport manager, request a reflow from it. It can // recompute the final CSS viewport and trigger a call to // ResizeReflowIgnoreOverride if it changed. mMobileViewportManager->RequestReflow(); return NS_OK; } return ResizeReflowIgnoreOverride(aWidth, aHeight, aOldWidth, aOldHeight); } nsresult PresShell::ResizeReflowIgnoreOverride(nscoord aWidth, nscoord aHeight, nscoord aOldWidth, nscoord aOldHeight) { NS_PRECONDITION(!mIsReflowing, "Shouldn't be in reflow here!"); // If we don't have a root frame yet, that means we haven't had our initial // reflow... If that's the case, and aWidth or aHeight is unconstrained, // ignore them altogether. nsIFrame* rootFrame = mFrameConstructor->GetRootFrame(); if (!rootFrame && aHeight == NS_UNCONSTRAINEDSIZE) { // We can't do the work needed for SizeToContent without a root // frame, and we want to return before setting the visible area. return NS_ERROR_NOT_AVAILABLE; } mPresContext->SetVisibleArea(nsRect(0, 0, aWidth, aHeight)); // There isn't anything useful we can do if the initial reflow hasn't happened. if (!rootFrame) { return NS_OK; } WritingMode wm = rootFrame->GetWritingMode(); NS_PRECONDITION((wm.IsVertical() ? aHeight : aWidth) != NS_UNCONSTRAINEDSIZE, "shouldn't use unconstrained isize anymore"); const bool isBSizeChanging = wm.IsVertical() ? aOldWidth != aWidth : aOldHeight != aHeight; RefPtr<nsViewManager> viewManager = mViewManager; // Take this ref after viewManager so it'll make sure to go away first. nsCOMPtr<nsIPresShell> kungFuDeathGrip(this); if (!GetPresContext()->SuppressingResizeReflow()) { // Have to make sure that the content notifications are flushed before we // start messing with the frame model; otherwise we can get content doubling. mDocument->FlushPendingNotifications(Flush_ContentAndNotify); // Make sure style is up to date { nsAutoScriptBlocker scriptBlocker; mPresContext->RestyleManager()->ProcessPendingRestyles(); } rootFrame = mFrameConstructor->GetRootFrame(); if (!mIsDestroying && rootFrame) { // XXX Do a full invalidate at the beginning so that invalidates along // the way don't have region accumulation issues? if (isBSizeChanging) { // For BSize changes driven by style, RestyleManager handles this. // For height:auto BSizes (i.e. layout-controlled), descendant // intrinsic sizes can't depend on them. So the only other case is // viewport-controlled BSizes which we handle here. nsLayoutUtils::MarkIntrinsicISizesDirtyIfDependentOnBSize(rootFrame); } { nsAutoCauseReflowNotifier crNotifier(this); WillDoReflow(); // Kick off a top-down reflow AUTO_LAYOUT_PHASE_ENTRY_POINT(GetPresContext(), Reflow); nsViewManager::AutoDisableRefresh refreshBlocker(viewManager); mDirtyRoots.RemoveElement(rootFrame); DoReflow(rootFrame, true); } DidDoReflow(true); } } rootFrame = mFrameConstructor->GetRootFrame(); if (rootFrame) { wm = rootFrame->GetWritingMode(); if (wm.IsVertical()) { if (aWidth == NS_UNCONSTRAINEDSIZE) { mPresContext->SetVisibleArea( nsRect(0, 0, rootFrame->GetRect().width, aHeight)); } } else { if (aHeight == NS_UNCONSTRAINEDSIZE) { mPresContext->SetVisibleArea( nsRect(0, 0, aWidth, rootFrame->GetRect().height)); } } } if (!mIsDestroying && !mResizeEvent.IsPending() && !mAsyncResizeTimerIsActive) { if (mInResize) { if (!mAsyncResizeEventTimer) { mAsyncResizeEventTimer = do_CreateInstance("@mozilla.org/timer;1"); } if (mAsyncResizeEventTimer) { mAsyncResizeTimerIsActive = true; mAsyncResizeEventTimer->InitWithFuncCallback(AsyncResizeEventCallback, this, 15, nsITimer::TYPE_ONE_SHOT); } } else { RefPtr<nsRunnableMethod<PresShell> > resizeEvent = NewRunnableMethod(this, &PresShell::FireResizeEvent); if (NS_SUCCEEDED(NS_DispatchToCurrentThread(resizeEvent))) { mResizeEvent = resizeEvent; mDocument->SetNeedStyleFlush(); } } } return NS_OK; //XXX this needs to be real. MMP } void PresShell::FireResizeEvent() { if (mAsyncResizeTimerIsActive) { mAsyncResizeTimerIsActive = false; mAsyncResizeEventTimer->Cancel(); } mResizeEvent.Revoke(); if (mIsDocumentGone) return; //Send resize event from here. WidgetEvent event(true, mozilla::eResize); nsEventStatus status = nsEventStatus_eIgnore; if (nsPIDOMWindowOuter* window = mDocument->GetWindow()) { nsCOMPtr<nsIPresShell> kungFuDeathGrip(this); mInResize = true; EventDispatcher::Dispatch(window, mPresContext, &event, nullptr, &status); mInResize = false; } } void PresShell::SetIgnoreFrameDestruction(bool aIgnore) { if (mDocument) { // We need to tell the ImageLoader to drop all its references to frames // because they're about to go away and it won't get notifications of that. mDocument->StyleImageLoader()->ClearFrames(mPresContext); } mIgnoreFrameDestruction = aIgnore; } void PresShell::NotifyDestroyingFrame(nsIFrame* aFrame) { if (!mIgnoreFrameDestruction) { mDocument->StyleImageLoader()->DropRequestsForFrame(aFrame); mFrameConstructor->NotifyDestroyingFrame(aFrame); for (int32_t idx = mDirtyRoots.Length(); idx; ) { --idx; if (mDirtyRoots[idx] == aFrame) { mDirtyRoots.RemoveElementAt(idx); } } // Remove frame properties mPresContext->NotifyDestroyingFrame(aFrame); if (aFrame == mCurrentEventFrame) { mCurrentEventContent = aFrame->GetContent(); mCurrentEventFrame = nullptr; } #ifdef DEBUG if (aFrame == mDrawEventTargetFrame) { mDrawEventTargetFrame = nullptr; } #endif for (unsigned int i=0; i < mCurrentEventFrameStack.Length(); i++) { if (aFrame == mCurrentEventFrameStack.ElementAt(i)) { //One of our stack frames was deleted. Get its content so that when we //pop it we can still get its new frame from its content nsIContent *currentEventContent = aFrame->GetContent(); mCurrentEventContentStack.ReplaceObjectAt(currentEventContent, i); mCurrentEventFrameStack[i] = nullptr; } } mFramesToDirty.RemoveEntry(aFrame); } else { // We must delete this property in situ so that its destructor removes the // frame from FrameLayerBuilder::DisplayItemData::mFrameList -- otherwise // the DisplayItemData destructor will use the destroyed frame when it // tries to remove it from the (array) value of this property. mPresContext->PropertyTable()-> Delete(aFrame, FrameLayerBuilder::LayerManagerDataProperty()); } } already_AddRefed<nsCaret> PresShell::GetCaret() const { RefPtr<nsCaret> caret = mCaret; return caret.forget(); } already_AddRefed<AccessibleCaretEventHub> PresShell::GetAccessibleCaretEventHub() const { RefPtr<AccessibleCaretEventHub> eventHub = mAccessibleCaretEventHub; return eventHub.forget(); } void PresShell::SetCaret(nsCaret *aNewCaret) { mCaret = aNewCaret; } void PresShell::RestoreCaret() { mCaret = mOriginalCaret; } NS_IMETHODIMP PresShell::SetCaretEnabled(bool aInEnable) { bool oldEnabled = mCaretEnabled; mCaretEnabled = aInEnable; if (mCaretEnabled != oldEnabled) { MOZ_ASSERT(mCaret); if (mCaret) { mCaret->SetVisible(mCaretEnabled); } } return NS_OK; } NS_IMETHODIMP PresShell::SetCaretReadOnly(bool aReadOnly) { if (mCaret) mCaret->SetCaretReadOnly(aReadOnly); return NS_OK; } NS_IMETHODIMP PresShell::GetCaretEnabled(bool *aOutEnabled) { NS_ENSURE_ARG_POINTER(aOutEnabled); *aOutEnabled = mCaretEnabled; return NS_OK; } NS_IMETHODIMP PresShell::SetCaretVisibilityDuringSelection(bool aVisibility) { if (mCaret) mCaret->SetVisibilityDuringSelection(aVisibility); return NS_OK; } NS_IMETHODIMP PresShell::GetCaretVisible(bool *aOutIsVisible) { *aOutIsVisible = false; if (mCaret) { *aOutIsVisible = mCaret->IsVisible(); } return NS_OK; } NS_IMETHODIMP PresShell::SetSelectionFlags(int16_t aInEnable) { mSelectionFlags = aInEnable; return NS_OK; } NS_IMETHODIMP PresShell::GetSelectionFlags(int16_t *aOutEnable) { if (!aOutEnable) return NS_ERROR_INVALID_ARG; *aOutEnable = mSelectionFlags; return NS_OK; } //implementation of nsISelectionController NS_IMETHODIMP PresShell::PhysicalMove(int16_t aDirection, int16_t aAmount, bool aExtend) { RefPtr<nsFrameSelection> frameSelection = mSelection; return frameSelection->PhysicalMove(aDirection, aAmount, aExtend); } NS_IMETHODIMP PresShell::CharacterMove(bool aForward, bool aExtend) { RefPtr<nsFrameSelection> frameSelection = mSelection; return frameSelection->CharacterMove(aForward, aExtend); } NS_IMETHODIMP PresShell::CharacterExtendForDelete() { RefPtr<nsFrameSelection> frameSelection = mSelection; return frameSelection->CharacterExtendForDelete(); } NS_IMETHODIMP PresShell::CharacterExtendForBackspace() { RefPtr<nsFrameSelection> frameSelection = mSelection; return frameSelection->CharacterExtendForBackspace(); } NS_IMETHODIMP PresShell::WordMove(bool aForward, bool aExtend) { RefPtr<nsFrameSelection> frameSelection = mSelection; nsresult result = frameSelection->WordMove(aForward, aExtend); // if we can't go down/up any more we must then move caret completely to // end/beginning respectively. if (NS_FAILED(result)) result = CompleteMove(aForward, aExtend); return result; } NS_IMETHODIMP PresShell::WordExtendForDelete(bool aForward) { RefPtr<nsFrameSelection> frameSelection = mSelection; return frameSelection->WordExtendForDelete(aForward); } NS_IMETHODIMP PresShell::LineMove(bool aForward, bool aExtend) { RefPtr<nsFrameSelection> frameSelection = mSelection; nsresult result = frameSelection->LineMove(aForward, aExtend); // if we can't go down/up any more we must then move caret completely to // end/beginning respectively. if (NS_FAILED(result)) result = CompleteMove(aForward,aExtend); return result; } NS_IMETHODIMP PresShell::IntraLineMove(bool aForward, bool aExtend) { RefPtr<nsFrameSelection> frameSelection = mSelection; return frameSelection->IntraLineMove(aForward, aExtend); } NS_IMETHODIMP PresShell::PageMove(bool aForward, bool aExtend) { nsIScrollableFrame *scrollableFrame = GetFrameToScrollAsScrollable(nsIPresShell::eVertical); if (!scrollableFrame) return NS_OK; RefPtr<nsFrameSelection> frameSelection = mSelection; frameSelection->CommonPageMove(aForward, aExtend, scrollableFrame); // After ScrollSelectionIntoView(), the pending notifications might be // flushed and PresShell/PresContext/Frames may be dead. See bug 418470. return ScrollSelectionIntoView(nsISelectionController::SELECTION_NORMAL, nsISelectionController::SELECTION_FOCUS_REGION, nsISelectionController::SCROLL_SYNCHRONOUS | nsISelectionController::SCROLL_FOR_CARET_MOVE); } NS_IMETHODIMP PresShell::ScrollPage(bool aForward) { nsIScrollableFrame* scrollFrame = GetFrameToScrollAsScrollable(nsIPresShell::eVertical); if (scrollFrame) { mozilla::Telemetry::Accumulate(mozilla::Telemetry::SCROLL_INPUT_METHODS, (uint32_t) ScrollInputMethod::MainThreadScrollPage); scrollFrame->ScrollBy(nsIntPoint(0, aForward ? 1 : -1), nsIScrollableFrame::PAGES, nsIScrollableFrame::SMOOTH, nullptr, nullptr, nsIScrollableFrame::NOT_MOMENTUM, nsIScrollableFrame::ENABLE_SNAP); } return NS_OK; } NS_IMETHODIMP PresShell::ScrollLine(bool aForward) { nsIScrollableFrame* scrollFrame = GetFrameToScrollAsScrollable(nsIPresShell::eVertical); if (scrollFrame) { mozilla::Telemetry::Accumulate(mozilla::Telemetry::SCROLL_INPUT_METHODS, (uint32_t) ScrollInputMethod::MainThreadScrollLine); int32_t lineCount = Preferences::GetInt("toolkit.scrollbox.verticalScrollDistance", NS_DEFAULT_VERTICAL_SCROLL_DISTANCE); scrollFrame->ScrollBy(nsIntPoint(0, aForward ? lineCount : -lineCount), nsIScrollableFrame::LINES, nsIScrollableFrame::SMOOTH, nullptr, nullptr, nsIScrollableFrame::NOT_MOMENTUM, nsIScrollableFrame::ENABLE_SNAP); } return NS_OK; } NS_IMETHODIMP PresShell::ScrollCharacter(bool aRight) { nsIScrollableFrame* scrollFrame = GetFrameToScrollAsScrollable(nsIPresShell::eHorizontal); if (scrollFrame) { mozilla::Telemetry::Accumulate(mozilla::Telemetry::SCROLL_INPUT_METHODS, (uint32_t) ScrollInputMethod::MainThreadScrollCharacter); int32_t h = Preferences::GetInt("toolkit.scrollbox.horizontalScrollDistance", NS_DEFAULT_HORIZONTAL_SCROLL_DISTANCE); scrollFrame->ScrollBy(nsIntPoint(aRight ? h : -h, 0), nsIScrollableFrame::LINES, nsIScrollableFrame::SMOOTH, nullptr, nullptr, nsIScrollableFrame::NOT_MOMENTUM, nsIScrollableFrame::ENABLE_SNAP); } return NS_OK; } NS_IMETHODIMP PresShell::CompleteScroll(bool aForward) { nsIScrollableFrame* scrollFrame = GetFrameToScrollAsScrollable(nsIPresShell::eVertical); if (scrollFrame) { mozilla::Telemetry::Accumulate(mozilla::Telemetry::SCROLL_INPUT_METHODS, (uint32_t) ScrollInputMethod::MainThreadCompleteScroll); scrollFrame->ScrollBy(nsIntPoint(0, aForward ? 1 : -1), nsIScrollableFrame::WHOLE, nsIScrollableFrame::SMOOTH, nullptr, nullptr, nsIScrollableFrame::NOT_MOMENTUM, nsIScrollableFrame::ENABLE_SNAP); } return NS_OK; } NS_IMETHODIMP PresShell::CompleteMove(bool aForward, bool aExtend) { // Beware! This may flush notifications via synchronous // ScrollSelectionIntoView. RefPtr<nsFrameSelection> frameSelection = mSelection; nsIContent* limiter = frameSelection->GetAncestorLimiter(); nsIFrame* frame = limiter ? limiter->GetPrimaryFrame() : FrameConstructor()->GetRootElementFrame(); if (!frame) return NS_ERROR_FAILURE; nsIFrame::CaretPosition pos = frame->GetExtremeCaretPosition(!aForward); frameSelection->HandleClick(pos.mResultContent, pos.mContentOffset, pos.mContentOffset, aExtend, false, aForward ? CARET_ASSOCIATE_AFTER : CARET_ASSOCIATE_BEFORE); if (limiter) { // HandleClick resets ancestorLimiter, so set it again. frameSelection->SetAncestorLimiter(limiter); } // After ScrollSelectionIntoView(), the pending notifications might be // flushed and PresShell/PresContext/Frames may be dead. See bug 418470. return ScrollSelectionIntoView(nsISelectionController::SELECTION_NORMAL, nsISelectionController::SELECTION_FOCUS_REGION, nsISelectionController::SCROLL_SYNCHRONOUS | nsISelectionController::SCROLL_FOR_CARET_MOVE); } NS_IMETHODIMP PresShell::SelectAll() { RefPtr<nsFrameSelection> frameSelection = mSelection; return frameSelection->SelectAll(); } static void DoCheckVisibility(nsPresContext* aPresContext, nsIContent* aNode, int16_t aStartOffset, int16_t aEndOffset, bool* aRetval) { nsIFrame* frame = aNode->GetPrimaryFrame(); if (!frame) { // No frame to look at so it must not be visible. return; } // Start process now to go through all frames to find startOffset. Then check // chars after that to see if anything until EndOffset is visible. bool finished = false; frame->CheckVisibility(aPresContext, aStartOffset, aEndOffset, true, &finished, aRetval); // Don't worry about other return value. } NS_IMETHODIMP PresShell::CheckVisibility(nsIDOMNode *node, int16_t startOffset, int16_t EndOffset, bool *_retval) { if (!node || startOffset>EndOffset || !_retval || startOffset<0 || EndOffset<0) return NS_ERROR_INVALID_ARG; *_retval = false; //initialize return parameter nsCOMPtr<nsIContent> content(do_QueryInterface(node)); if (!content) return NS_ERROR_FAILURE; DoCheckVisibility(mPresContext, content, startOffset, EndOffset, _retval); return NS_OK; } nsresult PresShell::CheckVisibilityContent(nsIContent* aNode, int16_t aStartOffset, int16_t aEndOffset, bool* aRetval) { if (!aNode || aStartOffset > aEndOffset || !aRetval || aStartOffset < 0 || aEndOffset < 0) { return NS_ERROR_INVALID_ARG; } *aRetval = false; DoCheckVisibility(mPresContext, aNode, aStartOffset, aEndOffset, aRetval); return NS_OK; } //end implementations nsISelectionController nsIFrame* nsIPresShell::GetRootFrameExternal() const { return mFrameConstructor->GetRootFrame(); } nsIFrame* nsIPresShell::GetRootScrollFrame() const { nsIFrame* rootFrame = mFrameConstructor->GetRootFrame(); // Ensure root frame is a viewport frame if (!rootFrame || nsGkAtoms::viewportFrame != rootFrame->GetType()) return nullptr; nsIFrame* theFrame = rootFrame->PrincipalChildList().FirstChild(); if (!theFrame || nsGkAtoms::scrollFrame != theFrame->GetType()) return nullptr; return theFrame; } nsIScrollableFrame* nsIPresShell::GetRootScrollFrameAsScrollable() const { nsIFrame* frame = GetRootScrollFrame(); if (!frame) return nullptr; nsIScrollableFrame* scrollableFrame = do_QueryFrame(frame); NS_ASSERTION(scrollableFrame, "All scroll frames must implement nsIScrollableFrame"); return scrollableFrame; } nsIScrollableFrame* nsIPresShell::GetRootScrollFrameAsScrollableExternal() const { return GetRootScrollFrameAsScrollable(); } nsIPageSequenceFrame* PresShell::GetPageSequenceFrame() const { nsIFrame* frame = mFrameConstructor->GetPageSequenceFrame(); return do_QueryFrame(frame); } nsCanvasFrame* PresShell::GetCanvasFrame() const { nsIFrame* frame = mFrameConstructor->GetDocElementContainingBlock(); return do_QueryFrame(frame); } void PresShell::BeginUpdate(nsIDocument *aDocument, nsUpdateType aUpdateType) { #ifdef DEBUG mUpdateCount++; #endif mFrameConstructor->BeginUpdate(); if (aUpdateType & UPDATE_STYLE) mStyleSet->BeginUpdate(); } void PresShell::EndUpdate(nsIDocument *aDocument, nsUpdateType aUpdateType) { #ifdef DEBUG NS_PRECONDITION(0 != mUpdateCount, "too many EndUpdate's"); --mUpdateCount; #endif if (aUpdateType & UPDATE_STYLE) { mStyleSet->EndUpdate(); if (mStylesHaveChanged || !mChangedScopeStyleRoots.IsEmpty()) RestyleForCSSRuleChanges(); } mFrameConstructor->EndUpdate(); } void PresShell::RestoreRootScrollPosition() { nsIScrollableFrame* scrollableFrame = GetRootScrollFrameAsScrollable(); if (scrollableFrame) { scrollableFrame->ScrollToRestoredPosition(); } } void PresShell::MaybeReleaseCapturingContent() { RefPtr<nsFrameSelection> frameSelection = FrameSelection(); if (frameSelection) { frameSelection->SetDragState(false); } if (gCaptureInfo.mContent && gCaptureInfo.mContent->OwnerDoc() == mDocument) { SetCapturingContent(nullptr, 0); } } void PresShell::BeginLoad(nsIDocument *aDocument) { mDocumentLoading = true; gfxTextPerfMetrics *tp = nullptr; if (mPresContext) { tp = mPresContext->GetTextPerfMetrics(); } bool shouldLog = MOZ_LOG_TEST(gLog, LogLevel::Debug); if (shouldLog || tp) { mLoadBegin = TimeStamp::Now(); } if (shouldLog) { nsIURI* uri = mDocument->GetDocumentURI(); MOZ_LOG(gLog, LogLevel::Debug, ("(presshell) %p load begin [%s]\n", this, uri ? uri->GetSpecOrDefault().get() : "")); } } void PresShell::EndLoad(nsIDocument *aDocument) { NS_PRECONDITION(aDocument == mDocument, "Wrong document"); RestoreRootScrollPosition(); mDocumentLoading = false; } void PresShell::LoadComplete() { gfxTextPerfMetrics *tp = nullptr; if (mPresContext) { tp = mPresContext->GetTextPerfMetrics(); } // log load bool shouldLog = MOZ_LOG_TEST(gLog, LogLevel::Debug); if (shouldLog || tp) { TimeDuration loadTime = TimeStamp::Now() - mLoadBegin; nsIURI* uri = mDocument->GetDocumentURI(); nsAutoCString spec; if (uri) { spec = uri->GetSpecOrDefault(); } if (shouldLog) { MOZ_LOG(gLog, LogLevel::Debug, ("(presshell) %p load done time-ms: %9.2f [%s]\n", this, loadTime.ToMilliseconds(), spec.get())); } if (tp) { tp->Accumulate(); if (tp->cumulative.numChars > 0) { LogTextPerfStats(tp, this, tp->cumulative, loadTime.ToMilliseconds(), eLog_loaddone, spec.get()); } } } } #ifdef DEBUG void PresShell::VerifyHasDirtyRootAncestor(nsIFrame* aFrame) { // XXXbz due to bug 372769, can't actually assert anything here... return; // XXXbz shouldn't need this part; remove it once FrameNeedsReflow // handles the root frame correctly. if (!aFrame->GetParent()) { return; } // Make sure that there is a reflow root ancestor of |aFrame| that's // in mDirtyRoots already. while (aFrame && (aFrame->GetStateBits() & NS_FRAME_HAS_DIRTY_CHILDREN)) { if (((aFrame->GetStateBits() & NS_FRAME_REFLOW_ROOT) || !aFrame->GetParent()) && mDirtyRoots.Contains(aFrame)) { return; } aFrame = aFrame->GetParent(); } NS_NOTREACHED("Frame has dirty bits set but isn't scheduled to be " "reflowed?"); } #endif void PresShell::FrameNeedsReflow(nsIFrame *aFrame, IntrinsicDirty aIntrinsicDirty, nsFrameState aBitToAdd, ReflowRootHandling aRootHandling) { NS_PRECONDITION(aBitToAdd == NS_FRAME_IS_DIRTY || aBitToAdd == NS_FRAME_HAS_DIRTY_CHILDREN || !aBitToAdd, "Unexpected bits being added"); NS_PRECONDITION(!(aIntrinsicDirty == eStyleChange && aBitToAdd == NS_FRAME_HAS_DIRTY_CHILDREN), "bits don't correspond to style change reason"); NS_ASSERTION(!mIsReflowing, "can't mark frame dirty during reflow"); // If we've not yet done the initial reflow, then don't bother // enqueuing a reflow command yet. if (! mDidInitialize) return; // If we're already destroying, don't bother with this either. if (mIsDestroying) return; #ifdef DEBUG //printf("gShellCounter: %d\n", gShellCounter++); if (mInVerifyReflow) return; if (VERIFY_REFLOW_NOISY_RC & gVerifyReflowFlags) { printf("\nPresShell@%p: frame %p needs reflow\n", (void*)this, (void*)aFrame); if (VERIFY_REFLOW_REALLY_NOISY_RC & gVerifyReflowFlags) { printf("Current content model:\n"); Element *rootElement = mDocument->GetRootElement(); if (rootElement) { rootElement->List(stdout, 0); } } } #endif AutoTArray<nsIFrame*, 4> subtrees; subtrees.AppendElement(aFrame); do { nsIFrame *subtreeRoot = subtrees.ElementAt(subtrees.Length() - 1); subtrees.RemoveElementAt(subtrees.Length() - 1); // Grab |wasDirty| now so we can go ahead and update the bits on // subtreeRoot. bool wasDirty = NS_SUBTREE_DIRTY(subtreeRoot); subtreeRoot->AddStateBits(aBitToAdd); // Determine whether we need to keep looking for the next ancestor // reflow root if subtreeRoot itself is a reflow root. bool targetNeedsReflowFromParent; switch (aRootHandling) { case ePositionOrSizeChange: targetNeedsReflowFromParent = true; break; case eNoPositionOrSizeChange: targetNeedsReflowFromParent = false; break; case eInferFromBitToAdd: targetNeedsReflowFromParent = (aBitToAdd == NS_FRAME_IS_DIRTY); break; } #define FRAME_IS_REFLOW_ROOT(_f) \ ((_f->GetStateBits() & NS_FRAME_REFLOW_ROOT) && \ (_f != subtreeRoot || !targetNeedsReflowFromParent)) // Mark the intrinsic widths as dirty on the frame, all of its ancestors, // and all of its descendants, if needed: if (aIntrinsicDirty != nsIPresShell::eResize) { // Mark argument and all ancestors dirty. (Unless we hit a reflow // root that should contain the reflow. That root could be // subtreeRoot itself if it's not dirty, or it could be some // ancestor of subtreeRoot.) for (nsIFrame *a = subtreeRoot; a && !FRAME_IS_REFLOW_ROOT(a); a = a->GetParent()) a->MarkIntrinsicISizesDirty(); } if (aIntrinsicDirty == eStyleChange) { // Mark all descendants dirty (using an nsTArray stack rather than // recursion). // Note that ReflowInput::InitResizeFlags has some similar // code; see comments there for how and why it differs. AutoTArray<nsIFrame*, 32> stack; stack.AppendElement(subtreeRoot); do { nsIFrame *f = stack.ElementAt(stack.Length() - 1); stack.RemoveElementAt(stack.Length() - 1); if (f->GetType() == nsGkAtoms::placeholderFrame) { nsIFrame *oof = nsPlaceholderFrame::GetRealFrameForPlaceholder(f); if (!nsLayoutUtils::IsProperAncestorFrame(subtreeRoot, oof)) { // We have another distinct subtree we need to mark. subtrees.AppendElement(oof); } } nsIFrame::ChildListIterator lists(f); for (; !lists.IsDone(); lists.Next()) { for (nsIFrame* kid : lists.CurrentList()) { kid->MarkIntrinsicISizesDirty(); stack.AppendElement(kid); } } } while (stack.Length() != 0); } // Skip setting dirty bits up the tree if we weren't given a bit to add. if (!aBitToAdd) { continue; } // Set NS_FRAME_HAS_DIRTY_CHILDREN bits (via nsIFrame::ChildIsDirty) // up the tree until we reach either a frame that's already dirty or // a reflow root. nsIFrame *f = subtreeRoot; for (;;) { if (FRAME_IS_REFLOW_ROOT(f) || !f->GetParent()) { // we've hit a reflow root or the root frame if (!wasDirty) { mDirtyRoots.AppendElement(f); mDocument->SetNeedLayoutFlush(); } #ifdef DEBUG else { VerifyHasDirtyRootAncestor(f); } #endif break; } nsIFrame *child = f; f = f->GetParent(); wasDirty = NS_SUBTREE_DIRTY(f); f->ChildIsDirty(child); NS_ASSERTION(f->GetStateBits() & NS_FRAME_HAS_DIRTY_CHILDREN, "ChildIsDirty didn't do its job"); if (wasDirty) { // This frame was already marked dirty. #ifdef DEBUG VerifyHasDirtyRootAncestor(f); #endif break; } } } while (subtrees.Length() != 0); MaybeScheduleReflow(); } void PresShell::FrameNeedsToContinueReflow(nsIFrame *aFrame) { NS_ASSERTION(mIsReflowing, "Must be in reflow when marking path dirty."); NS_PRECONDITION(mCurrentReflowRoot, "Must have a current reflow root here"); NS_ASSERTION(aFrame == mCurrentReflowRoot || nsLayoutUtils::IsProperAncestorFrame(mCurrentReflowRoot, aFrame), "Frame passed in is not the descendant of mCurrentReflowRoot"); NS_ASSERTION(aFrame->GetStateBits() & NS_FRAME_IN_REFLOW, "Frame passed in not in reflow?"); mFramesToDirty.PutEntry(aFrame); } nsIScrollableFrame* nsIPresShell::GetFrameToScrollAsScrollable( nsIPresShell::ScrollDirection aDirection) { nsIScrollableFrame* scrollFrame = nullptr; nsCOMPtr<nsIContent> focusedContent; nsIFocusManager* fm = nsFocusManager::GetFocusManager(); if (fm && mDocument) { nsCOMPtr<nsIDOMElement> focusedElement; fm->GetFocusedElementForWindow(mDocument->GetWindow(), false, nullptr, getter_AddRefs(focusedElement)); focusedContent = do_QueryInterface(focusedElement); } if (!focusedContent && mSelection) { nsISelection* domSelection = mSelection->GetSelection(SelectionType::eNormal); if (domSelection) { nsCOMPtr<nsIDOMNode> focusedNode; domSelection->GetFocusNode(getter_AddRefs(focusedNode)); focusedContent = do_QueryInterface(focusedNode); } } if (focusedContent) { nsIFrame* startFrame = focusedContent->GetPrimaryFrame(); if (startFrame) { scrollFrame = startFrame->GetScrollTargetFrame(); if (scrollFrame) { startFrame = scrollFrame->GetScrolledFrame(); } if (aDirection == nsIPresShell::eEither) { scrollFrame = nsLayoutUtils::GetNearestScrollableFrame(startFrame); } else { scrollFrame = nsLayoutUtils::GetNearestScrollableFrameForDirection(startFrame, aDirection == eVertical ? nsLayoutUtils::eVertical : nsLayoutUtils::eHorizontal); } } } if (!scrollFrame) { scrollFrame = GetRootScrollFrameAsScrollable(); } return scrollFrame; } void PresShell::CancelAllPendingReflows() { mDirtyRoots.Clear(); if (mReflowScheduled) { GetPresContext()->RefreshDriver()->RemoveLayoutFlushObserver(this); mReflowScheduled = false; } ASSERT_REFLOW_SCHEDULED_STATE(); } void PresShell::DestroyFramesFor(nsIContent* aContent, nsIContent** aDestroyedFramesFor) { MOZ_ASSERT(aContent); NS_ENSURE_TRUE_VOID(mPresContext); if (!mDidInitialize) { return; } nsAutoScriptBlocker scriptBlocker; // Mark ourselves as not safe to flush while we're doing frame destruction. ++mChangeNestCount; nsCSSFrameConstructor* fc = FrameConstructor(); fc->BeginUpdate(); fc->DestroyFramesFor(aContent, aDestroyedFramesFor); fc->EndUpdate(); --mChangeNestCount; } void PresShell::CreateFramesFor(nsIContent* aContent) { NS_ENSURE_TRUE_VOID(mPresContext); if (!mDidInitialize) { // Nothing to do here. In fact, if we proceed and aContent is the // root we will crash. return; } // Don't call RecreateFramesForContent since that is not exported and we want // to keep the number of entrypoints down. NS_ASSERTION(mViewManager, "Should have view manager"); MOZ_ASSERT(aContent); // Have to make sure that the content notifications are flushed before we // start messing with the frame model; otherwise we can get content doubling. mDocument->FlushPendingNotifications(Flush_ContentAndNotify); nsAutoScriptBlocker scriptBlocker; // Mark ourselves as not safe to flush while we're doing frame construction. ++mChangeNestCount; nsCSSFrameConstructor* fc = FrameConstructor(); nsILayoutHistoryState* layoutState = fc->GetLastCapturedLayoutHistoryState(); fc->BeginUpdate(); fc->ContentInserted(aContent->GetParent(), aContent, layoutState, false); fc->EndUpdate(); --mChangeNestCount; } nsresult PresShell::RecreateFramesFor(nsIContent* aContent) { NS_ENSURE_TRUE(mPresContext, NS_ERROR_FAILURE); if (!mDidInitialize) { // Nothing to do here. In fact, if we proceed and aContent is the // root we will crash. return NS_OK; } // Don't call RecreateFramesForContent since that is not exported and we want // to keep the number of entrypoints down. NS_ASSERTION(mViewManager, "Should have view manager"); // Have to make sure that the content notifications are flushed before we // start messing with the frame model; otherwise we can get content doubling. mDocument->FlushPendingNotifications(Flush_ContentAndNotify); nsAutoScriptBlocker scriptBlocker; nsStyleChangeList changeList; changeList.AppendChange(nullptr, aContent, nsChangeHint_ReconstructFrame); // Mark ourselves as not safe to flush while we're doing frame construction. ++mChangeNestCount; RestyleManagerHandle restyleManager = mPresContext->RestyleManager(); nsresult rv = restyleManager->ProcessRestyledFrames(changeList); restyleManager->FlushOverflowChangedTracker(); --mChangeNestCount; return rv; } void nsIPresShell::PostRecreateFramesFor(Element* aElement) { mPresContext->RestyleManager()->PostRestyleEvent(aElement, nsRestyleHint(0), nsChangeHint_ReconstructFrame); } void nsIPresShell::RestyleForAnimation(Element* aElement, nsRestyleHint aHint) { // Now that we no longer have separate non-animation and animation // restyles, this method having a distinct identity is less important, // but it still seems useful to offer as a "more public" API and as a // chokepoint for these restyles to go through. mPresContext->RestyleManager()->PostRestyleEvent(aElement, aHint, nsChangeHint(0)); } void nsIPresShell::SetForwardingContainer(const WeakPtr<nsDocShell> &aContainer) { mForwardingContainer = aContainer; } void PresShell::ClearFrameRefs(nsIFrame* aFrame) { mPresContext->EventStateManager()->ClearFrameRefs(aFrame); nsWeakFrame* weakFrame = mWeakFrames; while (weakFrame) { nsWeakFrame* prev = weakFrame->GetPreviousWeakFrame(); if (weakFrame->GetFrame() == aFrame) { // This removes weakFrame from mWeakFrames. weakFrame->Clear(this); } weakFrame = prev; } } already_AddRefed<gfxContext> PresShell::CreateReferenceRenderingContext() { nsDeviceContext* devCtx = mPresContext->DeviceContext(); RefPtr<gfxContext> rc; if (mPresContext->IsScreen()) { rc = gfxContext::CreateOrNull(gfxPlatform::GetPlatform()->ScreenReferenceDrawTarget()); } else { // We assume the devCtx has positive width and height for this call. // However, width and height, may be outside of the reasonable range // so rc may still be null. rc = devCtx->CreateReferenceRenderingContext(); } return rc ? rc.forget() : nullptr; } nsresult PresShell::GoToAnchor(const nsAString& aAnchorName, bool aScroll, uint32_t aAdditionalScrollFlags) { if (!mDocument) { return NS_ERROR_FAILURE; } const Element *root = mDocument->GetRootElement(); if (root && root->IsSVGElement(nsGkAtoms::svg)) { // We need to execute this even if there is an empty anchor name // so that any existing SVG fragment identifier effect is removed if (SVGFragmentIdentifier::ProcessFragmentIdentifier(mDocument, aAnchorName)) { return NS_OK; } } // Hold a reference to the ESM in case event dispatch tears us down. RefPtr<EventStateManager> esm = mPresContext->EventStateManager(); if (aAnchorName.IsEmpty()) { NS_ASSERTION(!aScroll, "can't scroll to empty anchor name"); esm->SetContentState(nullptr, NS_EVENT_STATE_URLTARGET); return NS_OK; } nsCOMPtr<nsIDOMHTMLDocument> htmlDoc = do_QueryInterface(mDocument); nsresult rv = NS_OK; nsCOMPtr<nsIContent> content; // Search for an element with a matching "id" attribute if (mDocument) { content = mDocument->GetElementById(aAnchorName); } // Search for an anchor element with a matching "name" attribute if (!content && htmlDoc) { nsCOMPtr<nsIDOMNodeList> list; // Find a matching list of named nodes rv = htmlDoc->GetElementsByName(aAnchorName, getter_AddRefs(list)); if (NS_SUCCEEDED(rv) && list) { uint32_t i; // Loop through the named nodes looking for the first anchor for (i = 0; true; i++) { nsCOMPtr<nsIDOMNode> node; rv = list->Item(i, getter_AddRefs(node)); if (!node) { // End of list break; } // Ensure it's an anchor element content = do_QueryInterface(node); if (content) { if (content->IsHTMLElement(nsGkAtoms::a)) { break; } content = nullptr; } } } } // Search for anchor in the HTML namespace with a matching name if (!content && !htmlDoc) { nsCOMPtr<nsIDOMDocument> doc = do_QueryInterface(mDocument); nsCOMPtr<nsIDOMNodeList> list; NS_NAMED_LITERAL_STRING(nameSpace, "http://www.w3.org/1999/xhtml"); // Get the list of anchor elements rv = doc->GetElementsByTagNameNS(nameSpace, NS_LITERAL_STRING("a"), getter_AddRefs(list)); if (NS_SUCCEEDED(rv) && list) { uint32_t i; // Loop through the named nodes looking for the first anchor for (i = 0; true; i++) { nsCOMPtr<nsIDOMNode> node; rv = list->Item(i, getter_AddRefs(node)); if (!node) { // End of list break; } // Compare the name attribute nsCOMPtr<nsIDOMElement> element = do_QueryInterface(node); nsAutoString value; if (element && NS_SUCCEEDED(element->GetAttribute(NS_LITERAL_STRING("name"), value))) { if (value.Equals(aAnchorName)) { content = do_QueryInterface(element); break; } } } } } esm->SetContentState(content, NS_EVENT_STATE_URLTARGET); #ifdef ACCESSIBILITY nsIContent *anchorTarget = content; #endif nsIScrollableFrame* rootScroll = GetRootScrollFrameAsScrollable(); if (rootScroll && rootScroll->DidHistoryRestore()) { // Scroll position restored from history trumps scrolling to anchor. aScroll = false; rootScroll->ClearDidHistoryRestore(); } if (content) { if (aScroll) { rv = ScrollContentIntoView(content, ScrollAxis(SCROLL_TOP, SCROLL_ALWAYS), ScrollAxis(), ANCHOR_SCROLL_FLAGS | aAdditionalScrollFlags); NS_ENSURE_SUCCESS(rv, rv); nsIScrollableFrame* rootScroll = GetRootScrollFrameAsScrollable(); if (rootScroll) { mLastAnchorScrolledTo = content; mLastAnchorScrollPositionY = rootScroll->GetScrollPosition().y; } } // Should we select the target? This action is controlled by a // preference: the default is to not select. bool selectAnchor = Preferences::GetBool("layout.selectanchor"); // Even if select anchor pref is false, we must still move the // caret there. That way tabbing will start from the new // location RefPtr<nsIDOMRange> jumpToRange = new nsRange(mDocument); while (content && content->GetFirstChild()) { content = content->GetFirstChild(); } nsCOMPtr<nsIDOMNode> node(do_QueryInterface(content)); NS_ASSERTION(node, "No nsIDOMNode for descendant of anchor"); jumpToRange->SelectNodeContents(node); // Select the anchor RefPtr<Selection> sel = mSelection->GetSelection(SelectionType::eNormal); if (sel) { sel->RemoveAllRanges(); sel->AddRange(jumpToRange); if (!selectAnchor) { // Use a caret (collapsed selection) at the start of the anchor sel->CollapseToStart(); } } // Selection is at anchor. // Now focus the document itself if focus is on an element within it. nsPIDOMWindowOuter *win = mDocument->GetWindow(); nsIFocusManager* fm = nsFocusManager::GetFocusManager(); if (fm && win) { nsCOMPtr<mozIDOMWindowProxy> focusedWindow; fm->GetFocusedWindow(getter_AddRefs(focusedWindow)); if (SameCOMIdentity(win, focusedWindow)) { fm->ClearFocus(focusedWindow); } } // If the target is an animation element, activate the animation if (content->IsNodeOfType(nsINode::eANIMATION)) { SVGContentUtils::ActivateByHyperlink(content.get()); } } else { rv = NS_ERROR_FAILURE; NS_NAMED_LITERAL_STRING(top, "top"); if (nsContentUtils::EqualsIgnoreASCIICase(aAnchorName, top)) { // Scroll to the top/left if aAnchorName is "top" and there is no element // with such a name or id. rv = NS_OK; nsIScrollableFrame* sf = GetRootScrollFrameAsScrollable(); // Check |aScroll| after setting |rv| so we set |rv| to the same // thing whether or not |aScroll| is true. if (aScroll && sf) { // Scroll to the top of the page sf->ScrollTo(nsPoint(0, 0), nsIScrollableFrame::INSTANT); } } } #ifdef ACCESSIBILITY if (anchorTarget) { nsAccessibilityService* accService = AccService(); if (accService) accService->NotifyOfAnchorJumpTo(anchorTarget); } #endif return rv; } nsresult PresShell::ScrollToAnchor() { if (!mLastAnchorScrolledTo) { return NS_OK; } NS_ASSERTION(mDidInitialize, "should have done initial reflow by now"); nsIScrollableFrame* rootScroll = GetRootScrollFrameAsScrollable(); if (!rootScroll || mLastAnchorScrollPositionY != rootScroll->GetScrollPosition().y) { return NS_OK; } nsresult rv = ScrollContentIntoView(mLastAnchorScrolledTo, ScrollAxis(SCROLL_TOP, SCROLL_ALWAYS), ScrollAxis(), ANCHOR_SCROLL_FLAGS); mLastAnchorScrolledTo = nullptr; return rv; } /* * Helper (per-continuation) for ScrollContentIntoView. * * @param aContainerFrame [in] the frame which aRect is relative to * @param aFrame [in] Frame whose bounds should be unioned * @param aUseWholeLineHeightForInlines [in] if true, then for inline frames * we should include the top of the line in the added rectangle * @param aRect [inout] rect into which its bounds should be unioned * @param aHaveRect [inout] whether aRect contains data yet * @param aPrevBlock [inout] the block aLines is a line iterator for * @param aLines [inout] the line iterator we're using * @param aCurLine [inout] the line to start looking from in this iterator */ static void AccumulateFrameBounds(nsIFrame* aContainerFrame, nsIFrame* aFrame, bool aUseWholeLineHeightForInlines, nsRect& aRect, bool& aHaveRect, nsIFrame*& aPrevBlock, nsAutoLineIterator& aLines, int32_t& aCurLine) { nsIFrame* frame = aFrame; nsRect frameBounds = nsRect(nsPoint(0, 0), aFrame->GetSize()); // If this is an inline frame and either the bounds height is 0 (quirks // layout model) or aUseWholeLineHeightForInlines is set, we need to // change the top of the bounds to include the whole line. if (frameBounds.height == 0 || aUseWholeLineHeightForInlines) { nsIFrame *prevFrame = aFrame; nsIFrame *f = aFrame; while (f && f->IsFrameOfType(nsIFrame::eLineParticipant) && !f->IsTransformed() && !f->IsAbsPosContainingBlock()) { prevFrame = f; f = prevFrame->GetParent(); } if (f != aFrame && f && f->GetType() == nsGkAtoms::blockFrame) { // find the line containing aFrame and increase the top of |offset|. if (f != aPrevBlock) { aLines = f->GetLineIterator(); aPrevBlock = f; aCurLine = 0; } if (aLines) { int32_t index = aLines->FindLineContaining(prevFrame, aCurLine); if (index >= 0) { aCurLine = index; nsIFrame *trash1; int32_t trash2; nsRect lineBounds; if (NS_SUCCEEDED(aLines->GetLine(index, &trash1, &trash2, lineBounds))) { frameBounds += frame->GetOffsetTo(f); frame = f; if (lineBounds.y < frameBounds.y) { frameBounds.height = frameBounds.YMost() - lineBounds.y; frameBounds.y = lineBounds.y; } } } } } } nsRect transformedBounds = nsLayoutUtils::TransformFrameRectToAncestor(frame, frameBounds, aContainerFrame); if (aHaveRect) { // We can't use nsRect::UnionRect since it drops empty rects on // the floor, and we need to include them. (Thus we need // aHaveRect to know when to drop the initial value on the floor.) aRect.UnionRectEdges(aRect, transformedBounds); } else { aHaveRect = true; aRect = transformedBounds; } } static bool ComputeNeedToScroll(nsIPresShell::WhenToScroll aWhenToScroll, nscoord aLineSize, nscoord aRectMin, nscoord aRectMax, nscoord aViewMin, nscoord aViewMax) { // See how the rect should be positioned vertically if (nsIPresShell::SCROLL_ALWAYS == aWhenToScroll) { // The caller wants the frame as visible as possible return true; } else if (nsIPresShell::SCROLL_IF_NOT_VISIBLE == aWhenToScroll) { // Scroll only if no part of the frame is visible in this view return aRectMax - aLineSize <= aViewMin || aRectMin + aLineSize >= aViewMax; } else if (nsIPresShell::SCROLL_IF_NOT_FULLY_VISIBLE == aWhenToScroll) { // Scroll only if part of the frame is hidden and more can fit in view return !(aRectMin >= aViewMin && aRectMax <= aViewMax) && std::min(aViewMax, aRectMax) - std::max(aRectMin, aViewMin) < aViewMax - aViewMin; } return false; } static nscoord ComputeWhereToScroll(int16_t aWhereToScroll, nscoord aOriginalCoord, nscoord aRectMin, nscoord aRectMax, nscoord aViewMin, nscoord aViewMax, nscoord* aRangeMin, nscoord* aRangeMax) { nscoord resultCoord = aOriginalCoord; // Allow the scroll operation to land anywhere that // makes the whole rectangle visible. if (nsIPresShell::SCROLL_MINIMUM == aWhereToScroll) { if (aRectMin < aViewMin) { // Scroll up so the frame's top edge is visible resultCoord = aRectMin; } else if (aRectMax > aViewMax) { // Scroll down so the frame's bottom edge is visible. Make sure the // frame's top edge is still visible resultCoord = aOriginalCoord + aRectMax - aViewMax; if (resultCoord > aRectMin) { resultCoord = aRectMin; } } } else { nscoord frameAlignCoord = NSToCoordRound(aRectMin + (aRectMax - aRectMin) * (aWhereToScroll / 100.0f)); resultCoord = NSToCoordRound(frameAlignCoord - (aViewMax - aViewMin) * ( aWhereToScroll / 100.0f)); } nscoord scrollPortLength = aViewMax - aViewMin; // Force the scroll range to extend to include resultCoord. *aRangeMin = std::min(resultCoord, aRectMax - scrollPortLength); *aRangeMax = std::max(resultCoord, aRectMin); return resultCoord; } /** * This function takes a scrollable frame, a rect in the coordinate system * of the scrolled frame, and a desired percentage-based scroll * position and attempts to scroll the rect to that position in the * scrollport. * * This needs to work even if aRect has a width or height of zero. */ static void ScrollToShowRect(nsIScrollableFrame* aFrameAsScrollable, const nsRect& aRect, nsIPresShell::ScrollAxis aVertical, nsIPresShell::ScrollAxis aHorizontal, uint32_t aFlags) { nsPoint scrollPt = aFrameAsScrollable->GetScrollPosition(); nsRect visibleRect(scrollPt, aFrameAsScrollable->GetScrollPositionClampingScrollPortSize()); nsSize lineSize; // Don't call GetLineScrollAmount unless we actually need it. Not only // does this save time, but it's not safe to call GetLineScrollAmount // during reflow (because it depends on font size inflation and doesn't // use the in-reflow-safe font-size inflation path). If we did call it, // it would assert and possible give the wrong result. if (aVertical.mWhenToScroll == nsIPresShell::SCROLL_IF_NOT_VISIBLE || aHorizontal.mWhenToScroll == nsIPresShell::SCROLL_IF_NOT_VISIBLE) { lineSize = aFrameAsScrollable->GetLineScrollAmount(); } ScrollbarStyles ss = aFrameAsScrollable->GetScrollbarStyles(); nsRect allowedRange(scrollPt, nsSize(0, 0)); bool needToScroll = false; uint32_t directions = aFrameAsScrollable->GetPerceivedScrollingDirections(); if (((aFlags & nsIPresShell::SCROLL_OVERFLOW_HIDDEN) || ss.mVertical != NS_STYLE_OVERFLOW_HIDDEN) && (!aVertical.mOnlyIfPerceivedScrollableDirection || (directions & nsIScrollableFrame::VERTICAL))) { if (ComputeNeedToScroll(aVertical.mWhenToScroll, lineSize.height, aRect.y, aRect.YMost(), visibleRect.y, visibleRect.YMost())) { nscoord maxHeight; scrollPt.y = ComputeWhereToScroll(aVertical.mWhereToScroll, scrollPt.y, aRect.y, aRect.YMost(), visibleRect.y, visibleRect.YMost(), &allowedRange.y, &maxHeight); allowedRange.height = maxHeight - allowedRange.y; needToScroll = true; } } if (((aFlags & nsIPresShell::SCROLL_OVERFLOW_HIDDEN) || ss.mHorizontal != NS_STYLE_OVERFLOW_HIDDEN) && (!aHorizontal.mOnlyIfPerceivedScrollableDirection || (directions & nsIScrollableFrame::HORIZONTAL))) { if (ComputeNeedToScroll(aHorizontal.mWhenToScroll, lineSize.width, aRect.x, aRect.XMost(), visibleRect.x, visibleRect.XMost())) { nscoord maxWidth; scrollPt.x = ComputeWhereToScroll(aHorizontal.mWhereToScroll, scrollPt.x, aRect.x, aRect.XMost(), visibleRect.x, visibleRect.XMost(), &allowedRange.x, &maxWidth); allowedRange.width = maxWidth - allowedRange.x; needToScroll = true; } } // If we don't need to scroll, then don't try since it might cancel // a current smooth scroll operation. if (needToScroll) { nsIScrollableFrame::ScrollMode scrollMode = nsIScrollableFrame::INSTANT; bool autoBehaviorIsSmooth = (aFrameAsScrollable->GetScrollbarStyles().mScrollBehavior == NS_STYLE_SCROLL_BEHAVIOR_SMOOTH); bool smoothScroll = (aFlags & nsIPresShell::SCROLL_SMOOTH) || ((aFlags & nsIPresShell::SCROLL_SMOOTH_AUTO) && autoBehaviorIsSmooth); if (gfxPrefs::ScrollBehaviorEnabled() && smoothScroll) { scrollMode = nsIScrollableFrame::SMOOTH_MSD; } aFrameAsScrollable->ScrollTo(scrollPt, scrollMode, &allowedRange); } } nsresult PresShell::ScrollContentIntoView(nsIContent* aContent, nsIPresShell::ScrollAxis aVertical, nsIPresShell::ScrollAxis aHorizontal, uint32_t aFlags) { NS_ENSURE_TRUE(aContent, NS_ERROR_NULL_POINTER); nsCOMPtr<nsIDocument> composedDoc = aContent->GetComposedDoc(); NS_ENSURE_STATE(composedDoc); NS_ASSERTION(mDidInitialize, "should have done initial reflow by now"); if (mContentToScrollTo) { mContentToScrollTo->DeleteProperty(nsGkAtoms::scrolling); } mContentToScrollTo = aContent; ScrollIntoViewData* data = new ScrollIntoViewData(); data->mContentScrollVAxis = aVertical; data->mContentScrollHAxis = aHorizontal; data->mContentToScrollToFlags = aFlags; if (NS_FAILED(mContentToScrollTo->SetProperty(nsGkAtoms::scrolling, data, nsINode::DeleteProperty<PresShell::ScrollIntoViewData>))) { mContentToScrollTo = nullptr; } // Flush layout and attempt to scroll in the process. composedDoc->SetNeedLayoutFlush(); composedDoc->FlushPendingNotifications(Flush_InterruptibleLayout); // If mContentToScrollTo is non-null, that means we interrupted the reflow // (or suppressed it altogether because we're suppressing interruptible // flushes right now) and won't necessarily get the position correct, but do // a best-effort scroll here. The other option would be to do this inside // FlushPendingNotifications, but I'm not sure the repeated scrolling that // could trigger if reflows keep getting interrupted would be more desirable // than a single best-effort scroll followed by one final scroll on the first // completed reflow. if (mContentToScrollTo) { DoScrollContentIntoView(); } return NS_OK; } void PresShell::DoScrollContentIntoView() { NS_ASSERTION(mDidInitialize, "should have done initial reflow by now"); nsIFrame* frame = mContentToScrollTo->GetPrimaryFrame(); if (!frame) { mContentToScrollTo->DeleteProperty(nsGkAtoms::scrolling); mContentToScrollTo = nullptr; return; } if (frame->GetStateBits() & NS_FRAME_FIRST_REFLOW) { // The reflow flush before this scroll got interrupted, and this frame's // coords and size are all zero, and it has no content showing anyway. // Don't bother scrolling to it. We'll try again when we finish up layout. return; } // Make sure we skip 'frame' ... if it's scrollable, we should use its // scrollable ancestor as the container. nsIFrame* container = nsLayoutUtils::GetClosestFrameOfType(frame->GetParent(), nsGkAtoms::scrollFrame); if (!container) { // nothing can be scrolled return; } ScrollIntoViewData* data = static_cast<ScrollIntoViewData*>( mContentToScrollTo->GetProperty(nsGkAtoms::scrolling)); if (MOZ_UNLIKELY(!data)) { mContentToScrollTo = nullptr; return; } // This is a two-step process. // Step 1: Find the bounds of the rect we want to scroll into view. For // example, for an inline frame we may want to scroll in the whole // line, or we may want to scroll multiple lines into view. // Step 2: Walk container frame and its ancestors and scroll them // appropriately. // frameBounds is relative to container. We're assuming // that scrollframes don't split so every continuation of frame will // be a descendant of container. (Things would still mostly work // even if that assumption was false.) nsRect frameBounds; bool haveRect = false; bool useWholeLineHeightForInlines = data->mContentScrollVAxis.mWhenToScroll != nsIPresShell::SCROLL_IF_NOT_FULLY_VISIBLE; // Reuse the same line iterator across calls to AccumulateFrameBounds. We set // it every time we detect a new block (stored in prevBlock). nsIFrame* prevBlock = nullptr; nsAutoLineIterator lines; // The last line we found a continuation on in |lines|. We assume that later // continuations cannot come on earlier lines. int32_t curLine = 0; do { AccumulateFrameBounds(container, frame, useWholeLineHeightForInlines, frameBounds, haveRect, prevBlock, lines, curLine); } while ((frame = frame->GetNextContinuation())); ScrollFrameRectIntoView(container, frameBounds, data->mContentScrollVAxis, data->mContentScrollHAxis, data->mContentToScrollToFlags); } bool PresShell::ScrollFrameRectIntoView(nsIFrame* aFrame, const nsRect& aRect, nsIPresShell::ScrollAxis aVertical, nsIPresShell::ScrollAxis aHorizontal, uint32_t aFlags) { bool didScroll = false; // This function needs to work even if rect has a width or height of 0. nsRect rect = aRect; nsIFrame* container = aFrame; // Walk up the frame hierarchy scrolling the rect into view and // keeping rect relative to container do { nsIScrollableFrame* sf = do_QueryFrame(container); if (sf) { nsPoint oldPosition = sf->GetScrollPosition(); nsRect targetRect = rect; if (container->StyleDisplay()->mOverflowClipBox == NS_STYLE_OVERFLOW_CLIP_BOX_CONTENT_BOX) { nsMargin padding = container->GetUsedPadding(); targetRect.Inflate(padding); } ScrollToShowRect(sf, targetRect - sf->GetScrolledFrame()->GetPosition(), aVertical, aHorizontal, aFlags); nsPoint newPosition = sf->LastScrollDestination(); // If the scroll position increased, that means our content moved up, // so our rect's offset should decrease rect += oldPosition - newPosition; if (oldPosition != newPosition) { didScroll = true; } // only scroll one container when this flag is set if (aFlags & nsIPresShell::SCROLL_FIRST_ANCESTOR_ONLY) { break; } } nsIFrame* parent; if (container->IsTransformed()) { container->GetTransformMatrix(nullptr, &parent); rect = nsLayoutUtils::TransformFrameRectToAncestor(container, rect, parent); } else { rect += container->GetPosition(); parent = container->GetParent(); } if (!parent && !(aFlags & nsIPresShell::SCROLL_NO_PARENT_FRAMES)) { nsPoint extraOffset(0,0); parent = nsLayoutUtils::GetCrossDocParentFrame(container, &extraOffset); if (parent) { int32_t APD = container->PresContext()->AppUnitsPerDevPixel(); int32_t parentAPD = parent->PresContext()->AppUnitsPerDevPixel(); rect = rect.ScaleToOtherAppUnitsRoundOut(APD, parentAPD); rect += extraOffset; } } container = parent; } while (container); return didScroll; } nsRectVisibility PresShell::GetRectVisibility(nsIFrame* aFrame, const nsRect &aRect, nscoord aMinTwips) const { NS_ASSERTION(aFrame->PresContext() == GetPresContext(), "prescontext mismatch?"); nsIFrame* rootFrame = mFrameConstructor->GetRootFrame(); NS_ASSERTION(rootFrame, "How can someone have a frame for this presshell when there's no root?"); nsIScrollableFrame* sf = GetRootScrollFrameAsScrollable(); nsRect scrollPortRect; if (sf) { scrollPortRect = sf->GetScrollPortRect(); nsIFrame* f = do_QueryFrame(sf); scrollPortRect += f->GetOffsetTo(rootFrame); } else { scrollPortRect = nsRect(nsPoint(0,0), rootFrame->GetSize()); } nsRect r = aRect + aFrame->GetOffsetTo(rootFrame); // If aRect is entirely visible then we don't need to ensure that // at least aMinTwips of it is visible if (scrollPortRect.Contains(r)) return nsRectVisibility_kVisible; nsRect insetRect = scrollPortRect; insetRect.Deflate(aMinTwips, aMinTwips); if (r.YMost() <= insetRect.y) return nsRectVisibility_kAboveViewport; if (r.y >= insetRect.YMost()) return nsRectVisibility_kBelowViewport; if (r.XMost() <= insetRect.x) return nsRectVisibility_kLeftOfViewport; if (r.x >= insetRect.XMost()) return nsRectVisibility_kRightOfViewport; return nsRectVisibility_kVisible; } class PaintTimerCallBack final : public nsITimerCallback { public: explicit PaintTimerCallBack(PresShell* aShell) : mShell(aShell) {} NS_DECL_ISUPPORTS NS_IMETHOD Notify(nsITimer* aTimer) final { mShell->SetNextPaintCompressed(); mShell->AddInvalidateHiddenPresShellObserver(mShell->GetPresContext()->RefreshDriver()); mShell->ScheduleViewManagerFlush(); return NS_OK; } private: ~PaintTimerCallBack() {} PresShell* mShell; }; NS_IMPL_ISUPPORTS(PaintTimerCallBack, nsITimerCallback) void PresShell::ScheduleViewManagerFlush(PaintType aType) { if (aType == PAINT_DELAYED_COMPRESS) { // Delay paint for 1 second. static const uint32_t kPaintDelayPeriod = 1000; if (!mDelayedPaintTimer) { mDelayedPaintTimer = do_CreateInstance(NS_TIMER_CONTRACTID); RefPtr<PaintTimerCallBack> cb = new PaintTimerCallBack(this); mDelayedPaintTimer->InitWithCallback(cb, kPaintDelayPeriod, nsITimer::TYPE_ONE_SHOT); } return; } nsPresContext* presContext = GetPresContext(); if (presContext) { presContext->RefreshDriver()->ScheduleViewManagerFlush(); } if (mDocument) { mDocument->SetNeedLayoutFlush(); } } bool FlushLayoutRecursive(nsIDocument* aDocument, void* aData = nullptr) { MOZ_ASSERT(!aData); nsCOMPtr<nsIDocument> kungFuDeathGrip(aDocument); aDocument->EnumerateSubDocuments(FlushLayoutRecursive, nullptr); aDocument->FlushPendingNotifications(Flush_Layout); return true; } void PresShell::DispatchSynthMouseMove(WidgetGUIEvent* aEvent, bool aFlushOnHoverChange) { RestyleManagerHandle restyleManager = mPresContext->RestyleManager(); uint32_t hoverGenerationBefore = restyleManager->GetHoverGeneration(); nsEventStatus status; nsView* targetView = nsView::GetViewFor(aEvent->mWidget); if (!targetView) return; targetView->GetViewManager()->DispatchEvent(aEvent, targetView, &status); if (MOZ_UNLIKELY(mIsDestroying)) { return; } if (aFlushOnHoverChange && hoverGenerationBefore != restyleManager->GetHoverGeneration()) { // Flush so that the resulting reflow happens now so that our caller // can suppress any synthesized mouse moves caused by that reflow. // This code only ever runs for the root document, but :hover changes // can happen in descendant documents too, so make sure we flush // all of them. FlushLayoutRecursive(mDocument); } } void PresShell::ClearMouseCaptureOnView(nsView* aView) { if (gCaptureInfo.mContent) { if (aView) { // if a view was specified, ensure that the captured content is within // this view. nsIFrame* frame = gCaptureInfo.mContent->GetPrimaryFrame(); if (frame) { nsView* view = frame->GetClosestView(); // if there is no view, capturing won't be handled any more, so // just release the capture. if (view) { do { if (view == aView) { gCaptureInfo.mContent = nullptr; // the view containing the captured content likely disappeared so // disable capture for now. gCaptureInfo.mAllowed = false; break; } view = view->GetParent(); } while (view); // return if the view wasn't found return; } } } gCaptureInfo.mContent = nullptr; } // disable mouse capture until the next mousedown as a dialog has opened // or a drag has started. Otherwise, someone could start capture during // the modal dialog or drag. gCaptureInfo.mAllowed = false; } void nsIPresShell::ClearMouseCapture(nsIFrame* aFrame) { if (!gCaptureInfo.mContent) { gCaptureInfo.mAllowed = false; return; } // null frame argument means clear the capture if (!aFrame) { gCaptureInfo.mContent = nullptr; gCaptureInfo.mAllowed = false; return; } nsIFrame* capturingFrame = gCaptureInfo.mContent->GetPrimaryFrame(); if (!capturingFrame) { gCaptureInfo.mContent = nullptr; gCaptureInfo.mAllowed = false; return; } if (nsLayoutUtils::IsAncestorFrameCrossDoc(aFrame, capturingFrame)) { gCaptureInfo.mContent = nullptr; gCaptureInfo.mAllowed = false; } } nsresult PresShell::CaptureHistoryState(nsILayoutHistoryState** aState) { NS_PRECONDITION(nullptr != aState, "null state pointer"); // We actually have to mess with the docshell here, since we want to // store the state back in it. // XXXbz this isn't really right, since this is being called in the // content viewer's Hide() method... by that point the docshell's // state could be wrong. We should sort out a better ownership // model for the layout history state. nsCOMPtr<nsIDocShell> docShell(mPresContext->GetDocShell()); if (!docShell) return NS_ERROR_FAILURE; nsCOMPtr<nsILayoutHistoryState> historyState; docShell->GetLayoutHistoryState(getter_AddRefs(historyState)); if (!historyState) { // Create the document state object historyState = NS_NewLayoutHistoryState(); docShell->SetLayoutHistoryState(historyState); } *aState = historyState; NS_IF_ADDREF(*aState); // Capture frame state for the entire frame hierarchy nsIFrame* rootFrame = mFrameConstructor->GetRootFrame(); if (!rootFrame) return NS_OK; mFrameConstructor->CaptureFrameState(rootFrame, historyState); return NS_OK; } void PresShell::ScheduleBeforeFirstPaint() { if (!mDocument->IsResourceDoc()) { // Notify observers that a new page is about to be drawn. Execute this // as soon as it is safe to run JS, which is guaranteed to be before we // go back to the event loop and actually draw the page. nsContentUtils::AddScriptRunner(new nsBeforeFirstPaintDispatcher(mDocument)); } } void PresShell::UnsuppressAndInvalidate() { // Note: We ignore the EnsureVisible check for resource documents, because // they won't have a docshell, so they'll always fail EnsureVisible. if ((!mDocument->IsResourceDoc() && !mPresContext->EnsureVisible()) || mHaveShutDown) { // No point; we're about to be torn down anyway. return; } ScheduleBeforeFirstPaint(); mPaintingSuppressed = false; nsIFrame* rootFrame = mFrameConstructor->GetRootFrame(); if (rootFrame) { // let's assume that outline on a root frame is not supported rootFrame->InvalidateFrame(); } // now that painting is unsuppressed, focus may be set on the document if (nsPIDOMWindowOuter* win = mDocument->GetWindow()) win->SetReadyForFocus(); if (!mHaveShutDown) { SynthesizeMouseMove(false); ScheduleApproximateFrameVisibilityUpdateNow(); } } void PresShell::UnsuppressPainting() { if (mPaintSuppressionTimer) { mPaintSuppressionTimer->Cancel(); mPaintSuppressionTimer = nullptr; } if (mIsDocumentGone || !mPaintingSuppressed) return; // If we have reflows pending, just wait until we process // the reflows and get all the frames where we want them // before actually unlocking the painting. Otherwise // go ahead and unlock now. if (!mDirtyRoots.IsEmpty()) mShouldUnsuppressPainting = true; else UnsuppressAndInvalidate(); } // Post a request to handle an arbitrary callback after reflow has finished. nsresult PresShell::PostReflowCallback(nsIReflowCallback* aCallback) { void* result = AllocateMisc(sizeof(nsCallbackEventRequest)); nsCallbackEventRequest* request = (nsCallbackEventRequest*)result; request->callback = aCallback; request->next = nullptr; if (mLastCallbackEventRequest) { mLastCallbackEventRequest = mLastCallbackEventRequest->next = request; } else { mFirstCallbackEventRequest = request; mLastCallbackEventRequest = request; } return NS_OK; } void PresShell::CancelReflowCallback(nsIReflowCallback* aCallback) { nsCallbackEventRequest* before = nullptr; nsCallbackEventRequest* node = mFirstCallbackEventRequest; while(node) { nsIReflowCallback* callback = node->callback; if (callback == aCallback) { nsCallbackEventRequest* toFree = node; if (node == mFirstCallbackEventRequest) { node = node->next; mFirstCallbackEventRequest = node; NS_ASSERTION(before == nullptr, "impossible"); } else { node = node->next; before->next = node; } if (toFree == mLastCallbackEventRequest) { mLastCallbackEventRequest = before; } FreeMisc(sizeof(nsCallbackEventRequest), toFree); } else { before = node; node = node->next; } } } void PresShell::CancelPostedReflowCallbacks() { while (mFirstCallbackEventRequest) { nsCallbackEventRequest* node = mFirstCallbackEventRequest; mFirstCallbackEventRequest = node->next; if (!mFirstCallbackEventRequest) { mLastCallbackEventRequest = nullptr; } nsIReflowCallback* callback = node->callback; FreeMisc(sizeof(nsCallbackEventRequest), node); if (callback) { callback->ReflowCallbackCanceled(); } } } void PresShell::HandlePostedReflowCallbacks(bool aInterruptible) { bool shouldFlush = false; while (mFirstCallbackEventRequest) { nsCallbackEventRequest* node = mFirstCallbackEventRequest; mFirstCallbackEventRequest = node->next; if (!mFirstCallbackEventRequest) { mLastCallbackEventRequest = nullptr; } nsIReflowCallback* callback = node->callback; FreeMisc(sizeof(nsCallbackEventRequest), node); if (callback) { if (callback->ReflowFinished()) { shouldFlush = true; } } } mozFlushType flushType = aInterruptible ? Flush_InterruptibleLayout : Flush_Layout; if (shouldFlush && !mIsDestroying) { FlushPendingNotifications(flushType); } } bool PresShell::IsSafeToFlush() const { // Not safe if we are reflowing or in the middle of frame construction bool isSafeToFlush = !mIsReflowing && !mChangeNestCount; if (isSafeToFlush) { // Not safe if we are painting nsViewManager* viewManager = GetViewManager(); if (viewManager) { bool isPainting = false; viewManager->IsPainting(isPainting); if (isPainting) { isSafeToFlush = false; } } } return isSafeToFlush; } void PresShell::FlushPendingNotifications(mozFlushType aType) { // by default, flush animations if aType >= Flush_Style mozilla::ChangesToFlush flush(aType, aType >= Flush_Style); FlushPendingNotifications(flush); } void PresShell::FlushPendingNotifications(mozilla::ChangesToFlush aFlush) { if (mIsZombie) { return; } /** * VERY IMPORTANT: If you add some sort of new flushing to this * method, make sure to add the relevant SetNeedLayoutFlush or * SetNeedStyleFlush calls on the document. */ mozFlushType flushType = aFlush.mFlushType; #ifdef MOZ_ENABLE_PROFILER_SPS static const char flushTypeNames[][20] = { "Content", "ContentAndNotify", "Style", "InterruptibleLayout", "Layout", "Display" }; // Make sure that we don't miss things added to mozFlushType! MOZ_ASSERT(static_cast<uint32_t>(flushType) <= ArrayLength(flushTypeNames)); PROFILER_LABEL_PRINTF("PresShell", "Flush", js::ProfileEntry::Category::GRAPHICS, "(Flush_%s)", flushTypeNames[flushType - 1]); #endif #ifdef ACCESSIBILITY #ifdef DEBUG nsAccessibilityService* accService = GetAccService(); if (accService) { NS_ASSERTION(!accService->IsProcessingRefreshDriverNotification(), "Flush during accessible tree update!"); } #endif #endif NS_ASSERTION(flushType >= Flush_Frames, "Why did we get called?"); bool isSafeToFlush = IsSafeToFlush(); // If layout could possibly trigger scripts, then it's only safe to flush if // it's safe to run script. bool hasHadScriptObject; if (mDocument->GetScriptHandlingObject(hasHadScriptObject) || hasHadScriptObject) { isSafeToFlush = isSafeToFlush && nsContentUtils::IsSafeToRunScript(); } NS_ASSERTION(!isSafeToFlush || mViewManager, "Must have view manager"); // Make sure the view manager stays alive. RefPtr<nsViewManager> viewManager = mViewManager; bool didStyleFlush = false; bool didLayoutFlush = false; nsCOMPtr<nsIPresShell> kungFuDeathGrip; if (isSafeToFlush && viewManager) { // Processing pending notifications can kill us, and some callers only // hold weak refs when calling FlushPendingNotifications(). :( kungFuDeathGrip = this; if (mResizeEvent.IsPending()) { FireResizeEvent(); if (mIsDestroying) { return; } } // We need to make sure external resource documents are flushed too (for // example, svg filters that reference a filter in an external document // need the frames in the external document to be constructed for the // filter to work). We only need external resources to be flushed when the // main document is flushing >= Flush_Frames, so we flush external // resources here instead of nsDocument::FlushPendingNotifications. mDocument->FlushExternalResources(flushType); // Force flushing of any pending content notifications that might have // queued up while our event was pending. That will ensure that we don't // construct frames for content right now that's still waiting to be // notified on, mDocument->FlushPendingNotifications(Flush_ContentAndNotify); // Process pending restyles, since any flush of the presshell wants // up-to-date style data. if (!mIsDestroying) { viewManager->FlushDelayedResize(false); mPresContext->FlushPendingMediaFeatureValuesChanged(); // Flush any pending update of the user font set, since that could // cause style changes (for updating ex/ch units, and to cause a // reflow). mDocument->FlushUserFontSet(); mPresContext->FlushCounterStyles(); // Flush any requested SMIL samples. if (mDocument->HasAnimationController()) { mDocument->GetAnimationController()->FlushResampleRequests(); } if (aFlush.mFlushAnimations && mPresContext->EffectCompositor()) { mPresContext->EffectCompositor()->PostRestyleForThrottledAnimations(); } // The FlushResampleRequests() above flushed style changes. if (!mIsDestroying) { nsAutoScriptBlocker scriptBlocker; mPresContext->RestyleManager()->ProcessPendingRestyles(); } } // Process whatever XBL constructors those restyles queued up. This // ensures that onload doesn't fire too early and that we won't do extra // reflows after those constructors run. if (!mIsDestroying) { mDocument->BindingManager()->ProcessAttachedQueue(); } // Now those constructors or events might have posted restyle // events. At the same time, we still need up-to-date style data. // In particular, reflow depends on style being completely up to // date. If it's not, then style context reparenting, which can // happen during reflow, might suddenly pick up the new rules and // we'll end up with frames whose style doesn't match the frame // type. if (!mIsDestroying) { nsAutoScriptBlocker scriptBlocker; mPresContext->RestyleManager()->ProcessPendingRestyles(); } didStyleFlush = true; // There might be more pending constructors now, but we're not going to // worry about them. They can't be triggered during reflow, so we should // be good. if (flushType >= (mSuppressInterruptibleReflows ? Flush_Layout : Flush_InterruptibleLayout) && !mIsDestroying) { didLayoutFlush = true; mFrameConstructor->RecalcQuotesAndCounters(); viewManager->FlushDelayedResize(true); if (ProcessReflowCommands(flushType < Flush_Layout) && mContentToScrollTo) { // We didn't get interrupted. Go ahead and scroll to our content DoScrollContentIntoView(); if (mContentToScrollTo) { mContentToScrollTo->DeleteProperty(nsGkAtoms::scrolling); mContentToScrollTo = nullptr; } } } if (flushType >= Flush_Layout) { if (!mIsDestroying) { viewManager->UpdateWidgetGeometry(); } } } if (!didStyleFlush && flushType >= Flush_Style && !mIsDestroying) { mDocument->SetNeedStyleFlush(); } if (!didLayoutFlush && !mIsDestroying && (flushType >= (mSuppressInterruptibleReflows ? Flush_Layout : Flush_InterruptibleLayout))) { // We suppressed this flush due to mSuppressInterruptibleReflows or // !isSafeToFlush, but the document thinks it doesn't // need to flush anymore. Let it know what's really going on. mDocument->SetNeedLayoutFlush(); } } void PresShell::CharacterDataChanged(nsIDocument *aDocument, nsIContent* aContent, CharacterDataChangeInfo* aInfo) { NS_PRECONDITION(!mIsDocumentGone, "Unexpected CharacterDataChanged"); NS_PRECONDITION(aDocument == mDocument, "Unexpected aDocument"); nsAutoCauseReflowNotifier crNotifier(this); // Call this here so it only happens for real content mutations and // not cases when the frame constructor calls its own methods to force // frame reconstruction. nsIContent *container = aContent->GetParent(); uint32_t selectorFlags = container ? (container->GetFlags() & NODE_ALL_SELECTOR_FLAGS) : 0; if (selectorFlags != 0 && !aContent->IsRootOfAnonymousSubtree()) { Element* element = container->AsElement(); if (aInfo->mAppend && !aContent->GetNextSibling()) mPresContext->RestyleManager()->RestyleForAppend(element, aContent); else mPresContext->RestyleManager()->RestyleForInsertOrChange(element, aContent); } mFrameConstructor->CharacterDataChanged(aContent, aInfo); VERIFY_STYLE_TREE; } void PresShell::ContentStateChanged(nsIDocument* aDocument, nsIContent* aContent, EventStates aStateMask) { NS_PRECONDITION(!mIsDocumentGone, "Unexpected ContentStateChanged"); NS_PRECONDITION(aDocument == mDocument, "Unexpected aDocument"); if (mDidInitialize) { nsAutoCauseReflowNotifier crNotifier(this); mPresContext->RestyleManager()->ContentStateChanged(aContent, aStateMask); VERIFY_STYLE_TREE; } } void PresShell::DocumentStatesChanged(nsIDocument* aDocument, EventStates aStateMask) { NS_PRECONDITION(!mIsDocumentGone, "Unexpected DocumentStatesChanged"); NS_PRECONDITION(aDocument == mDocument, "Unexpected aDocument"); nsStyleSet* styleSet = mStyleSet->GetAsGecko(); if (!styleSet) { // XXXheycam ServoStyleSets don't support document state selectors, // but these are only used in chrome documents, which we are not // aiming to support yet. NS_WARNING("stylo: ServoStyleSets cannot respond to document state " "changes yet (only matters for chrome documents). See bug 1290285."); return; } if (mDidInitialize && styleSet->HasDocumentStateDependentStyle(mDocument->GetRootElement(), aStateMask)) { mPresContext->RestyleManager()->PostRestyleEvent(mDocument->GetRootElement(), eRestyle_Subtree, nsChangeHint(0)); VERIFY_STYLE_TREE; } if (aStateMask.HasState(NS_DOCUMENT_STATE_WINDOW_INACTIVE)) { nsIFrame* root = mFrameConstructor->GetRootFrame(); if (root) { root->SchedulePaint(); } } } void PresShell::AttributeWillChange(nsIDocument* aDocument, Element* aElement, int32_t aNameSpaceID, nsIAtom* aAttribute, int32_t aModType, const nsAttrValue* aNewValue) { NS_PRECONDITION(!mIsDocumentGone, "Unexpected AttributeWillChange"); NS_PRECONDITION(aDocument == mDocument, "Unexpected aDocument"); // XXXwaterson it might be more elegant to wait until after the // initial reflow to begin observing the document. That would // squelch any other inappropriate notifications as well. if (mDidInitialize) { nsAutoCauseReflowNotifier crNotifier(this); mPresContext->RestyleManager()->AttributeWillChange(aElement, aNameSpaceID, aAttribute, aModType, aNewValue); VERIFY_STYLE_TREE; } } void PresShell::AttributeChanged(nsIDocument* aDocument, Element* aElement, int32_t aNameSpaceID, nsIAtom* aAttribute, int32_t aModType, const nsAttrValue* aOldValue) { NS_PRECONDITION(!mIsDocumentGone, "Unexpected AttributeChanged"); NS_PRECONDITION(aDocument == mDocument, "Unexpected aDocument"); // XXXwaterson it might be more elegant to wait until after the // initial reflow to begin observing the document. That would // squelch any other inappropriate notifications as well. if (mDidInitialize) { nsAutoCauseReflowNotifier crNotifier(this); mPresContext->RestyleManager()->AttributeChanged(aElement, aNameSpaceID, aAttribute, aModType, aOldValue); VERIFY_STYLE_TREE; } } // nsIMutationObserver callbacks have this terrible API where aContainer is // null in the case that the container is the document (since nsIDocument is // not an nsIContent), and callees are supposed to figure this out and use the // document instead. It would be nice to fix that API to just pass a single // nsINode* parameter in place of the nsIDocument*, nsIContent* pair, but // there are quite a lot of consumers. So we fix things up locally with this // routine for now. static inline nsINode* RealContainer(nsIDocument* aDocument, nsIContent* aContainer, nsIContent* aContent) { MOZ_ASSERT(aDocument); MOZ_ASSERT_IF(aContainer, aContainer->OwnerDoc() == aDocument); MOZ_ASSERT(aContent->OwnerDoc() == aDocument); MOZ_ASSERT_IF(!aContainer, aContent->GetParentNode() == aDocument); if (!aContainer) { return aDocument; } return aContainer; } void PresShell::ContentAppended(nsIDocument *aDocument, nsIContent* aContainer, nsIContent* aFirstNewContent, int32_t aNewIndexInContainer) { NS_PRECONDITION(!mIsDocumentGone, "Unexpected ContentAppended"); NS_PRECONDITION(aDocument == mDocument, "Unexpected aDocument"); // We never call ContentAppended with a document as the container, so we can // assert that we have an nsIContent container. MOZ_ASSERT(aContainer); MOZ_ASSERT(aContainer->IsElement() || aContainer->IsNodeOfType(nsINode::eDOCUMENT_FRAGMENT)); if (!mDidInitialize) { return; } nsAutoCauseReflowNotifier crNotifier(this); // Call this here so it only happens for real content mutations and // not cases when the frame constructor calls its own methods to force // frame reconstruction. mPresContext->RestyleManager()->ContentAppended(aContainer, aFirstNewContent); mFrameConstructor->ContentAppended(aContainer, aFirstNewContent, true); VERIFY_STYLE_TREE; } void PresShell::ContentInserted(nsIDocument* aDocument, nsIContent* aMaybeContainer, nsIContent* aChild, int32_t aIndexInContainer) { NS_PRECONDITION(!mIsDocumentGone, "Unexpected ContentInserted"); NS_PRECONDITION(aDocument == mDocument, "Unexpected aDocument"); nsINode* container = RealContainer(aDocument, aMaybeContainer, aChild); if (!mDidInitialize) { return; } nsAutoCauseReflowNotifier crNotifier(this); // Call this here so it only happens for real content mutations and // not cases when the frame constructor calls its own methods to force // frame reconstruction. mPresContext->RestyleManager()->ContentInserted(container, aChild); mFrameConstructor->ContentInserted(aMaybeContainer, aChild, nullptr, true); if (aChild->NodeType() == nsIDOMNode::DOCUMENT_TYPE_NODE) { MOZ_ASSERT(container == aDocument); NotifyFontSizeInflationEnabledIsDirty(); } VERIFY_STYLE_TREE; } void PresShell::ContentRemoved(nsIDocument *aDocument, nsIContent* aMaybeContainer, nsIContent* aChild, int32_t aIndexInContainer, nsIContent* aPreviousSibling) { NS_PRECONDITION(!mIsDocumentGone, "Unexpected ContentRemoved"); NS_PRECONDITION(aDocument == mDocument, "Unexpected aDocument"); nsINode* container = RealContainer(aDocument, aMaybeContainer, aChild); // Notify the ESM that the content has been removed, so that // it can clean up any state related to the content. // XXX_jwir3: There is no null check for aDocument necessary, since, even // though by nsIMutationObserver, aDocument could be null, the // precondition check that mDocument == aDocument ensures that // aDocument will not be null (since mDocument can't be null unless // we're still intializing). mPresContext->EventStateManager()->ContentRemoved(aDocument, aChild); nsAutoCauseReflowNotifier crNotifier(this); // Call this here so it only happens for real content mutations and // not cases when the frame constructor calls its own methods to force // frame reconstruction. nsIContent* oldNextSibling = container->GetChildAt(aIndexInContainer); mPresContext->RestyleManager()->ContentRemoved(container, aChild, oldNextSibling); // After removing aChild from tree we should save information about live ancestor if (mPointerEventTarget) { if (nsContentUtils::ContentIsDescendantOf(mPointerEventTarget, aChild)) { mPointerEventTarget = aMaybeContainer; } } // We should check that aChild does not contain pointer capturing elements. // If it does we should release the pointer capture for the elements. for (auto iter = sPointerCaptureList->Iter(); !iter.Done(); iter.Next()) { nsIPresShell::PointerCaptureInfo* data = iter.UserData(); if (data && data->mPendingContent && nsContentUtils::ContentIsDescendantOf(data->mPendingContent, aChild)) { nsIPresShell::ReleasePointerCapturingContent(iter.Key()); } } bool didReconstruct; mFrameConstructor->ContentRemoved(aMaybeContainer, aChild, oldNextSibling, nsCSSFrameConstructor::REMOVE_CONTENT, &didReconstruct); if (aChild->NodeType() == nsIDOMNode::DOCUMENT_TYPE_NODE) { MOZ_ASSERT(container == aDocument); NotifyFontSizeInflationEnabledIsDirty(); } VERIFY_STYLE_TREE; } void PresShell::NotifyCounterStylesAreDirty() { nsAutoCauseReflowNotifier reflowNotifier(this); mFrameConstructor->BeginUpdate(); mFrameConstructor->NotifyCounterStylesAreDirty(); mFrameConstructor->EndUpdate(); } nsresult PresShell::ReconstructFrames(void) { NS_PRECONDITION(!mFrameConstructor->GetRootFrame() || mDidInitialize, "Must not have root frame before initial reflow"); if (!mDidInitialize || mIsDestroying) { // Nothing to do here return NS_OK; } nsCOMPtr<nsIPresShell> kungFuDeathGrip(this); // Have to make sure that the content notifications are flushed before we // start messing with the frame model; otherwise we can get content doubling. mDocument->FlushPendingNotifications(Flush_ContentAndNotify); if (mIsDestroying) { return NS_OK; } nsAutoCauseReflowNotifier crNotifier(this); mFrameConstructor->BeginUpdate(); nsresult rv = mFrameConstructor->ReconstructDocElementHierarchy(); VERIFY_STYLE_TREE; mFrameConstructor->EndUpdate(); return rv; } void nsIPresShell::RestyleForCSSRuleChanges() { AutoTArray<RefPtr<mozilla::dom::Element>,1> scopeRoots; mChangedScopeStyleRoots.SwapElements(scopeRoots); if (mStylesHaveChanged) { // If we need to restyle everything, no need to restyle individual // scoped style roots. scopeRoots.Clear(); } mStylesHaveChanged = false; if (mIsDestroying) { // We don't want to mess with restyles at this point return; } mDocument->RebuildUserFontSet(); if (mPresContext) { mPresContext->RebuildCounterStyles(); } Element* root = mDocument->GetRootElement(); if (!mDidInitialize) { // Nothing to do here, since we have no frames yet return; } if (!root) { // No content to restyle return; } RestyleManagerHandle restyleManager = mPresContext->RestyleManager(); if (scopeRoots.IsEmpty()) { // If scopeRoots is empty, we know that mStylesHaveChanged was true at // the beginning of this function, and that we need to restyle the whole // document. restyleManager->PostRestyleEvent(root, eRestyle_Subtree, nsChangeHint(0)); } else { for (Element* scopeRoot : scopeRoots) { restyleManager->PostRestyleEvent(scopeRoot, eRestyle_Subtree, nsChangeHint(0)); } } } void PresShell::RecordStyleSheetChange(StyleSheet* aStyleSheet) { // too bad we can't check that the update is UPDATE_STYLE NS_ASSERTION(mUpdateCount != 0, "must be in an update"); if (mStylesHaveChanged) return; if (aStyleSheet->IsGecko()) { // XXXheycam ServoStyleSheets don't support <style scoped> yet. Element* scopeElement = aStyleSheet->AsGecko()->GetScopeElement(); if (scopeElement) { mChangedScopeStyleRoots.AppendElement(scopeElement); return; } } else { NS_WARNING("stylo: ServoStyleSheets don't support <style scoped>"); return; } mStylesHaveChanged = true; } void PresShell::StyleSheetAdded(StyleSheet* aStyleSheet, bool aDocumentSheet) { // We only care when enabled sheets are added NS_PRECONDITION(aStyleSheet, "Must have a style sheet!"); if (aStyleSheet->IsApplicable() && aStyleSheet->HasRules()) { RecordStyleSheetChange(aStyleSheet); } } void PresShell::StyleSheetRemoved(StyleSheet* aStyleSheet, bool aDocumentSheet) { // We only care when enabled sheets are removed NS_PRECONDITION(aStyleSheet, "Must have a style sheet!"); if (aStyleSheet->IsApplicable() && aStyleSheet->HasRules()) { RecordStyleSheetChange(aStyleSheet); } } void PresShell::StyleSheetApplicableStateChanged(StyleSheet* aStyleSheet) { if (aStyleSheet->HasRules()) { RecordStyleSheetChange(aStyleSheet); } } void PresShell::StyleRuleChanged(StyleSheet* aStyleSheet) { RecordStyleSheetChange(aStyleSheet); } void PresShell::StyleRuleAdded(StyleSheet* aStyleSheet) { RecordStyleSheetChange(aStyleSheet); } void PresShell::StyleRuleRemoved(StyleSheet* aStyleSheet) { RecordStyleSheetChange(aStyleSheet); } nsIFrame* PresShell::GetPlaceholderFrameFor(nsIFrame* aFrame) const { return mFrameConstructor->GetPlaceholderFrameFor(aFrame); } nsresult PresShell::RenderDocument(const nsRect& aRect, uint32_t aFlags, nscolor aBackgroundColor, gfxContext* aThebesContext) { NS_ENSURE_TRUE(!(aFlags & RENDER_IS_UNTRUSTED), NS_ERROR_NOT_IMPLEMENTED); nsRootPresContext* rootPresContext = mPresContext->GetRootPresContext(); if (rootPresContext) { rootPresContext->FlushWillPaintObservers(); if (mIsDestroying) return NS_OK; } nsAutoScriptBlocker blockScripts; // Set up the rectangle as the path in aThebesContext gfxRect r(0, 0, nsPresContext::AppUnitsToFloatCSSPixels(aRect.width), nsPresContext::AppUnitsToFloatCSSPixels(aRect.height)); aThebesContext->NewPath(); #ifdef MOZ_GFX_OPTIMIZE_MOBILE aThebesContext->Rectangle(r, true); #else aThebesContext->Rectangle(r); #endif nsIFrame* rootFrame = mFrameConstructor->GetRootFrame(); if (!rootFrame) { // Nothing to paint, just fill the rect aThebesContext->SetColor(Color::FromABGR(aBackgroundColor)); aThebesContext->Fill(); return NS_OK; } gfxContextAutoSaveRestore save(aThebesContext); MOZ_ASSERT(aThebesContext->CurrentOp() == CompositionOp::OP_OVER); aThebesContext->Clip(); nsDeviceContext* devCtx = mPresContext->DeviceContext(); gfxPoint offset(-nsPresContext::AppUnitsToFloatCSSPixels(aRect.x), -nsPresContext::AppUnitsToFloatCSSPixels(aRect.y)); gfxFloat scale = gfxFloat(devCtx->AppUnitsPerDevPixel())/nsPresContext::AppUnitsPerCSSPixel(); // Since canvas APIs use floats to set up their matrices, we may have some // slight rounding errors here. We use NudgeToIntegers() here to adjust // matrix components that are integers up to the accuracy of floats to be // those integers. gfxMatrix newTM = aThebesContext->CurrentMatrix().Translate(offset). Scale(scale, scale). NudgeToIntegers(); aThebesContext->SetMatrix(newTM); AutoSaveRestoreRenderingState _(this); nsRenderingContext rc(aThebesContext); bool wouldFlushRetainedLayers = false; PaintFrameFlags flags = PaintFrameFlags::PAINT_IGNORE_SUPPRESSION; if (aThebesContext->CurrentMatrix().HasNonIntegerTranslation()) { flags |= PaintFrameFlags::PAINT_IN_TRANSFORM; } if (!(aFlags & RENDER_ASYNC_DECODE_IMAGES)) { flags |= PaintFrameFlags::PAINT_SYNC_DECODE_IMAGES; } if (aFlags & RENDER_USE_WIDGET_LAYERS) { // We only support using widget layers on display root's with widgets. nsView* view = rootFrame->GetView(); if (view && view->GetWidget() && nsLayoutUtils::GetDisplayRootFrame(rootFrame) == rootFrame) { LayerManager* layerManager = view->GetWidget()->GetLayerManager(); // ClientLayerManagers in content processes don't support // taking snapshots. if (layerManager && (!layerManager->AsClientLayerManager() || XRE_IsParentProcess())) { flags |= PaintFrameFlags::PAINT_WIDGET_LAYERS; } } } if (!(aFlags & RENDER_CARET)) { wouldFlushRetainedLayers = true; flags |= PaintFrameFlags::PAINT_HIDE_CARET; } if (aFlags & RENDER_IGNORE_VIEWPORT_SCROLLING) { wouldFlushRetainedLayers = !IgnoringViewportScrolling(); mRenderFlags = ChangeFlag(mRenderFlags, true, STATE_IGNORING_VIEWPORT_SCROLLING); } if (aFlags & RENDER_DRAWWINDOW_NOT_FLUSHING) { mRenderFlags = ChangeFlag(mRenderFlags, true, STATE_DRAWWINDOW_NOT_FLUSHING); } if (aFlags & RENDER_DOCUMENT_RELATIVE) { // XXX be smarter about this ... drawWindow might want a rect // that's "pretty close" to what our retained layer tree covers. // In that case, it wouldn't disturb normal rendering too much, // and we should allow it. wouldFlushRetainedLayers = true; flags |= PaintFrameFlags::PAINT_DOCUMENT_RELATIVE; } // Don't let drawWindow blow away our retained layer tree if ((flags & PaintFrameFlags::PAINT_WIDGET_LAYERS) && wouldFlushRetainedLayers) { flags &= ~PaintFrameFlags::PAINT_WIDGET_LAYERS; } nsLayoutUtils::PaintFrame(&rc, rootFrame, nsRegion(aRect), aBackgroundColor, nsDisplayListBuilderMode::PAINTING, flags); return NS_OK; } /* * Clip the display list aList to a range. Returns the clipped * rectangle surrounding the range. */ nsRect PresShell::ClipListToRange(nsDisplayListBuilder *aBuilder, nsDisplayList* aList, nsRange* aRange) { // iterate though the display items and add up the bounding boxes of each. // This will allow the total area of the frames within the range to be // determined. To do this, remove an item from the bottom of the list, check // whether it should be part of the range, and if so, append it to the top // of the temporary list tmpList. If the item is a text frame at the end of // the selection range, clip it to the portion of the text frame that is // part of the selection. Then, append the wrapper to the top of the list. // Otherwise, just delete the item and don't append it. nsRect surfaceRect; nsDisplayList tmpList; nsDisplayItem* i; while ((i = aList->RemoveBottom())) { // itemToInsert indiciates the item that should be inserted into the // temporary list. If null, no item should be inserted. nsDisplayItem* itemToInsert = nullptr; nsIFrame* frame = i->Frame(); nsIContent* content = frame->GetContent(); if (content) { bool atStart = (content == aRange->GetStartParent()); bool atEnd = (content == aRange->GetEndParent()); if ((atStart || atEnd) && frame->GetType() == nsGkAtoms::textFrame) { int32_t frameStartOffset, frameEndOffset; frame->GetOffsets(frameStartOffset, frameEndOffset); int32_t hilightStart = atStart ? std::max(aRange->StartOffset(), frameStartOffset) : frameStartOffset; int32_t hilightEnd = atEnd ? std::min(aRange->EndOffset(), frameEndOffset) : frameEndOffset; if (hilightStart < hilightEnd) { // determine the location of the start and end edges of the range. nsPoint startPoint, endPoint; frame->GetPointFromOffset(hilightStart, &startPoint); frame->GetPointFromOffset(hilightEnd, &endPoint); // The clip rectangle is determined by taking the the start and // end points of the range, offset from the reference frame. // Because of rtl, the end point may be to the left of (or above, // in vertical mode) the start point, so x (or y) is set to the // lower of the values. nsRect textRect(aBuilder->ToReferenceFrame(frame), frame->GetSize()); if (frame->GetWritingMode().IsVertical()) { nscoord y = std::min(startPoint.y, endPoint.y); textRect.y += y; textRect.height = std::max(startPoint.y, endPoint.y) - y; } else { nscoord x = std::min(startPoint.x, endPoint.x); textRect.x += x; textRect.width = std::max(startPoint.x, endPoint.x) - x; } surfaceRect.UnionRect(surfaceRect, textRect); DisplayItemClip newClip; newClip.SetTo(textRect); newClip.IntersectWith(i->GetClip()); i->SetClip(aBuilder, newClip); itemToInsert = i; } } // Don't try to descend into subdocuments. // If this ever changes we'd need to add handling for subdocuments with // different zoom levels. else if (content->GetUncomposedDoc() == aRange->GetStartParent()->GetUncomposedDoc()) { // if the node is within the range, append it to the temporary list bool before, after; nsresult rv = nsRange::CompareNodeToRange(content, aRange, &before, &after); if (NS_SUCCEEDED(rv) && !before && !after) { itemToInsert = i; bool snap; surfaceRect.UnionRect(surfaceRect, i->GetBounds(aBuilder, &snap)); } } } // insert the item into the list if necessary. If the item has a child // list, insert that as well nsDisplayList* sublist = i->GetSameCoordinateSystemChildren(); if (itemToInsert || sublist) { tmpList.AppendToTop(itemToInsert ? itemToInsert : i); // if the item is a list, iterate over it as well if (sublist) surfaceRect.UnionRect(surfaceRect, ClipListToRange(aBuilder, sublist, aRange)); } else { // otherwise, just delete the item and don't readd it to the list i->~nsDisplayItem(); } } // now add all the items back onto the original list again aList->AppendToTop(&tmpList); return surfaceRect; } #ifdef DEBUG #include <stdio.h> static bool gDumpRangePaintList = false; #endif UniquePtr<RangePaintInfo> PresShell::CreateRangePaintInfo(nsIDOMRange* aRange, nsRect& aSurfaceRect, bool aForPrimarySelection) { nsRange* range = static_cast<nsRange*>(aRange); nsIFrame* ancestorFrame; nsIFrame* rootFrame = GetRootFrame(); // If the start or end of the range is the document, just use the root // frame, otherwise get the common ancestor of the two endpoints of the // range. nsINode* startParent = range->GetStartParent(); nsINode* endParent = range->GetEndParent(); nsIDocument* doc = startParent->GetComposedDoc(); if (startParent == doc || endParent == doc) { ancestorFrame = rootFrame; } else { nsINode* ancestor = nsContentUtils::GetCommonAncestor(startParent, endParent); NS_ASSERTION(!ancestor || ancestor->IsNodeOfType(nsINode::eCONTENT), "common ancestor is not content"); if (!ancestor || !ancestor->IsNodeOfType(nsINode::eCONTENT)) return nullptr; nsIContent* ancestorContent = static_cast<nsIContent*>(ancestor); ancestorFrame = ancestorContent->GetPrimaryFrame(); // XXX deal with ancestorFrame being null due to display:contents // use the nearest ancestor frame that includes all continuations as the // root for building the display list while (ancestorFrame && nsLayoutUtils::GetNextContinuationOrIBSplitSibling(ancestorFrame)) ancestorFrame = ancestorFrame->GetParent(); } if (!ancestorFrame) { return nullptr; } // get a display list containing the range auto info = MakeUnique<RangePaintInfo>(range, ancestorFrame); info->mBuilder.SetIncludeAllOutOfFlows(); if (aForPrimarySelection) { info->mBuilder.SetSelectedFramesOnly(); } info->mBuilder.EnterPresShell(ancestorFrame); nsCOMPtr<nsIContentIterator> iter = NS_NewContentSubtreeIterator(); nsresult rv = iter->Init(range); if (NS_FAILED(rv)) { return nullptr; } auto BuildDisplayListForNode = [&] (nsINode* aNode) { if (MOZ_UNLIKELY(!aNode->IsContent())) { return; } nsIFrame* frame = aNode->AsContent()->GetPrimaryFrame(); // XXX deal with frame being null due to display:contents for (; frame; frame = nsLayoutUtils::GetNextContinuationOrIBSplitSibling(frame)) { frame->BuildDisplayListForStackingContext(&info->mBuilder, frame->GetVisualOverflowRect(), &info->mList); } }; if (startParent->NodeType() == nsIDOMNode::TEXT_NODE) { BuildDisplayListForNode(startParent); } for (; !iter->IsDone(); iter->Next()) { nsCOMPtr<nsINode> node = iter->GetCurrentNode(); BuildDisplayListForNode(node); } if (endParent != startParent && endParent->NodeType() == nsIDOMNode::TEXT_NODE) { BuildDisplayListForNode(endParent); } #ifdef DEBUG if (gDumpRangePaintList) { fprintf(stderr, "CreateRangePaintInfo --- before ClipListToRange:\n"); nsFrame::PrintDisplayList(&(info->mBuilder), info->mList); } #endif nsRect rangeRect = ClipListToRange(&info->mBuilder, &info->mList, range); info->mBuilder.LeavePresShell(ancestorFrame, &info->mList); #ifdef DEBUG if (gDumpRangePaintList) { fprintf(stderr, "CreateRangePaintInfo --- after ClipListToRange:\n"); nsFrame::PrintDisplayList(&(info->mBuilder), info->mList); } #endif // determine the offset of the reference frame for the display list // to the root frame. This will allow the coordinates used when painting // to all be offset from the same point info->mRootOffset = ancestorFrame->GetOffsetTo(rootFrame); rangeRect.MoveBy(info->mRootOffset); aSurfaceRect.UnionRect(aSurfaceRect, rangeRect); return info; } already_AddRefed<SourceSurface> PresShell::PaintRangePaintInfo(const nsTArray<UniquePtr<RangePaintInfo>>& aItems, nsISelection* aSelection, nsIntRegion* aRegion, nsRect aArea, const LayoutDeviceIntPoint aPoint, LayoutDeviceIntRect* aScreenRect, uint32_t aFlags) { nsPresContext* pc = GetPresContext(); if (!pc || aArea.width == 0 || aArea.height == 0) return nullptr; // use the rectangle to create the surface nsIntRect pixelArea = aArea.ToOutsidePixels(pc->AppUnitsPerDevPixel()); // if the image should not be resized, the scale, relative to the original image, must be 1 float scale = 1.0; nsIntRect rootScreenRect = GetRootFrame()->GetScreenRectInAppUnits().ToNearestPixels( pc->AppUnitsPerDevPixel()); nsRect maxSize; pc->DeviceContext()->GetClientRect(maxSize); // check if the image should be resized bool resize = aFlags & RENDER_AUTO_SCALE; if (resize) { // check if image-resizing-algorithm should be used if (aFlags & RENDER_IS_IMAGE) { // get max screensize nscoord maxWidth = pc->AppUnitsToDevPixels(maxSize.width); nscoord maxHeight = pc->AppUnitsToDevPixels(maxSize.height); // resize image relative to the screensize // get best height/width relative to screensize float bestHeight = float(maxHeight)*RELATIVE_SCALEFACTOR; float bestWidth = float(maxWidth)*RELATIVE_SCALEFACTOR; // get scalefactor to reach bestWidth scale = bestWidth / float(pixelArea.width); // get the worst height (height when width is perfect) float worstHeight = float(pixelArea.height)*scale; // get the difference of best and worst height float difference = bestHeight - worstHeight; // half the difference and add it to worstHeight, // then get scalefactor to reach this scale = (worstHeight + difference / 2) / float(pixelArea.height); } else { // get half of max screensize nscoord maxWidth = pc->AppUnitsToDevPixels(maxSize.width >> 1); nscoord maxHeight = pc->AppUnitsToDevPixels(maxSize.height >> 1); if (pixelArea.width > maxWidth || pixelArea.height > maxHeight) { scale = 1.0; // divide the maximum size by the image size in both directions. Whichever // direction produces the smallest result determines how much should be // scaled. if (pixelArea.width > maxWidth) scale = std::min(scale, float(maxWidth) / pixelArea.width); if (pixelArea.height > maxHeight) scale = std::min(scale, float(maxHeight) / pixelArea.height); } } pixelArea.width = NSToIntFloor(float(pixelArea.width) * scale); pixelArea.height = NSToIntFloor(float(pixelArea.height) * scale); if (!pixelArea.width || !pixelArea.height) return nullptr; // adjust the screen position based on the rescaled size nscoord left = rootScreenRect.x + pixelArea.x; nscoord top = rootScreenRect.y + pixelArea.y; aScreenRect->x = NSToIntFloor(aPoint.x - float(aPoint.x - left) * scale); aScreenRect->y = NSToIntFloor(aPoint.y - float(aPoint.y - top) * scale); } else { // move aScreenRect to the position of the surface in screen coordinates aScreenRect->MoveTo(rootScreenRect.x + pixelArea.x, rootScreenRect.y + pixelArea.y); } aScreenRect->width = pixelArea.width; aScreenRect->height = pixelArea.height; RefPtr<DrawTarget> dt = gfxPlatform::GetPlatform()->CreateOffscreenContentDrawTarget( IntSize(pixelArea.width, pixelArea.height), SurfaceFormat::B8G8R8A8); if (!dt || !dt->IsValid()) { return nullptr; } RefPtr<gfxContext> ctx = gfxContext::CreateOrNull(dt); MOZ_ASSERT(ctx); // already checked the draw target above if (aRegion) { RefPtr<PathBuilder> builder = dt->CreatePathBuilder(FillRule::FILL_WINDING); // Convert aRegion from CSS pixels to dev pixels nsIntRegion region = aRegion->ToAppUnits(nsPresContext::AppUnitsPerCSSPixel()) .ToOutsidePixels(pc->AppUnitsPerDevPixel()); for (auto iter = region.RectIter(); !iter.Done(); iter.Next()) { const nsIntRect& rect = iter.Get(); builder->MoveTo(rect.TopLeft()); builder->LineTo(rect.TopRight()); builder->LineTo(rect.BottomRight()); builder->LineTo(rect.BottomLeft()); builder->LineTo(rect.TopLeft()); } RefPtr<Path> path = builder->Finish(); ctx->Clip(path); } nsRenderingContext rc(ctx); gfxMatrix initialTM = ctx->CurrentMatrix(); if (resize) initialTM.Scale(scale, scale); // translate so that points are relative to the surface area gfxPoint surfaceOffset = nsLayoutUtils::PointToGfxPoint(-aArea.TopLeft(), pc->AppUnitsPerDevPixel()); initialTM.Translate(surfaceOffset); // temporarily hide the selection so that text is drawn normally. If a // selection is being rendered, use that, otherwise use the presshell's // selection. RefPtr<nsFrameSelection> frameSelection; if (aSelection) { frameSelection = aSelection->AsSelection()->GetFrameSelection(); } else { frameSelection = FrameSelection(); } int16_t oldDisplaySelection = frameSelection->GetDisplaySelection(); frameSelection->SetDisplaySelection(nsISelectionController::SELECTION_HIDDEN); // next, paint each range in the selection for (const UniquePtr<RangePaintInfo>& rangeInfo : aItems) { // the display lists paint relative to the offset from the reference // frame, so account for that translation too: gfxPoint rootOffset = nsLayoutUtils::PointToGfxPoint(rangeInfo->mRootOffset, pc->AppUnitsPerDevPixel()); ctx->SetMatrix(gfxMatrix(initialTM).Translate(rootOffset)); aArea.MoveBy(-rangeInfo->mRootOffset.x, -rangeInfo->mRootOffset.y); nsRegion visible(aArea); RefPtr<LayerManager> layerManager = rangeInfo->mList.PaintRoot(&rangeInfo->mBuilder, &rc, nsDisplayList::PAINT_DEFAULT); aArea.MoveBy(rangeInfo->mRootOffset.x, rangeInfo->mRootOffset.y); } // restore the old selection display state frameSelection->SetDisplaySelection(oldDisplaySelection); return dt->Snapshot(); } already_AddRefed<SourceSurface> PresShell::RenderNode(nsIDOMNode* aNode, nsIntRegion* aRegion, const LayoutDeviceIntPoint aPoint, LayoutDeviceIntRect* aScreenRect, uint32_t aFlags) { // area will hold the size of the surface needed to draw the node, measured // from the root frame. nsRect area; nsTArray<UniquePtr<RangePaintInfo>> rangeItems; // nothing to draw if the node isn't in a document nsCOMPtr<nsINode> node = do_QueryInterface(aNode); if (!node->IsInUncomposedDoc()) return nullptr; RefPtr<nsRange> range = new nsRange(node); if (NS_FAILED(range->SelectNode(aNode))) return nullptr; UniquePtr<RangePaintInfo> info = CreateRangePaintInfo(range, area, false); if (info && !rangeItems.AppendElement(Move(info))) { return nullptr; } if (aRegion) { // combine the area with the supplied region nsIntRect rrectPixels = aRegion->GetBounds(); nsRect rrect = ToAppUnits(rrectPixels, nsPresContext::AppUnitsPerCSSPixel()); area.IntersectRect(area, rrect); nsPresContext* pc = GetPresContext(); if (!pc) return nullptr; // move the region so that it is offset from the topleft corner of the surface aRegion->MoveBy(-nsPresContext::AppUnitsToIntCSSPixels(area.x), -nsPresContext::AppUnitsToIntCSSPixels(area.y)); } return PaintRangePaintInfo(rangeItems, nullptr, aRegion, area, aPoint, aScreenRect, aFlags); } already_AddRefed<SourceSurface> PresShell::RenderSelection(nsISelection* aSelection, const LayoutDeviceIntPoint aPoint, LayoutDeviceIntRect* aScreenRect, uint32_t aFlags) { // area will hold the size of the surface needed to draw the selection, // measured from the root frame. nsRect area; nsTArray<UniquePtr<RangePaintInfo>> rangeItems; // iterate over each range and collect them into the rangeItems array. // This is done so that the size of selection can be determined so as // to allocate a surface area int32_t numRanges; aSelection->GetRangeCount(&numRanges); NS_ASSERTION(numRanges > 0, "RenderSelection called with no selection"); for (int32_t r = 0; r < numRanges; r++) { nsCOMPtr<nsIDOMRange> range; aSelection->GetRangeAt(r, getter_AddRefs(range)); UniquePtr<RangePaintInfo> info = CreateRangePaintInfo(range, area, true); if (info && !rangeItems.AppendElement(Move(info))) { return nullptr; } } return PaintRangePaintInfo(rangeItems, aSelection, nullptr, area, aPoint, aScreenRect, aFlags); } void PresShell::AddPrintPreviewBackgroundItem(nsDisplayListBuilder& aBuilder, nsDisplayList& aList, nsIFrame* aFrame, const nsRect& aBounds) { aList.AppendNewToBottom(new (&aBuilder) nsDisplaySolidColor(&aBuilder, aFrame, aBounds, NS_RGB(115, 115, 115))); } static bool AddCanvasBackgroundColor(const nsDisplayList& aList, nsIFrame* aCanvasFrame, nscolor aColor, bool aCSSBackgroundColor) { for (nsDisplayItem* i = aList.GetBottom(); i; i = i->GetAbove()) { if (i->Frame() == aCanvasFrame && i->GetType() == nsDisplayItem::TYPE_CANVAS_BACKGROUND_COLOR) { nsDisplayCanvasBackgroundColor* bg = static_cast<nsDisplayCanvasBackgroundColor*>(i); bg->SetExtraBackgroundColor(aColor); return true; } nsDisplayList* sublist = i->GetSameCoordinateSystemChildren(); if (sublist && !(i->GetType() == nsDisplayItem::TYPE_BLEND_CONTAINER && !aCSSBackgroundColor) && AddCanvasBackgroundColor(*sublist, aCanvasFrame, aColor, aCSSBackgroundColor)) return true; } return false; } void PresShell::AddCanvasBackgroundColorItem(nsDisplayListBuilder& aBuilder, nsDisplayList& aList, nsIFrame* aFrame, const nsRect& aBounds, nscolor aBackstopColor, uint32_t aFlags) { if (aBounds.IsEmpty()) { return; } // We don't want to add an item for the canvas background color if the frame // (sub)tree we are painting doesn't include any canvas frames. There isn't // an easy way to check this directly, but if we check if the root of the // (sub)tree we are painting is a canvas frame that should cover us in all // cases (it will usually be a viewport frame when we have a canvas frame in // the (sub)tree). if (!(aFlags & nsIPresShell::FORCE_DRAW) && !nsCSSRendering::IsCanvasFrame(aFrame)) { return; } nscolor bgcolor = NS_ComposeColors(aBackstopColor, mCanvasBackgroundColor); if (NS_GET_A(bgcolor) == 0) return; // To make layers work better, we want to avoid having a big non-scrolled // color background behind a scrolled transparent background. Instead, // we'll try to move the color background into the scrolled content // by making nsDisplayCanvasBackground paint it. if (!aFrame->GetParent()) { nsIScrollableFrame* sf = aFrame->PresContext()->PresShell()->GetRootScrollFrameAsScrollable(); if (sf) { nsCanvasFrame* canvasFrame = do_QueryFrame(sf->GetScrolledFrame()); if (canvasFrame && canvasFrame->IsVisibleForPainting(&aBuilder)) { if (AddCanvasBackgroundColor(aList, canvasFrame, bgcolor, mHasCSSBackgroundColor)) return; } } } aList.AppendNewToBottom( new (&aBuilder) nsDisplaySolidColor(&aBuilder, aFrame, aBounds, bgcolor)); } static bool IsTransparentContainerElement(nsPresContext* aPresContext) { nsCOMPtr<nsIDocShell> docShell = aPresContext->GetDocShell(); if (!docShell) { return false; } nsCOMPtr<nsPIDOMWindowOuter> pwin = docShell->GetWindow(); if (!pwin) return false; nsCOMPtr<Element> containerElement = pwin->GetFrameElementInternal(); TabChild* tab = TabChild::GetFrom(docShell); if (tab) { // Check if presShell is the top PresShell. Only the top can // influence the canvas background color. nsCOMPtr<nsIPresShell> presShell = aPresContext->GetPresShell(); nsCOMPtr<nsIPresShell> topPresShell = tab->GetPresShell(); if (presShell != topPresShell) { tab = nullptr; } } return (containerElement && containerElement->HasAttr(kNameSpaceID_None, nsGkAtoms::transparent)) || (tab && tab->IsTransparent()); } nscolor PresShell::GetDefaultBackgroundColorToDraw() { if (!mPresContext || !mPresContext->GetBackgroundColorDraw()) { return NS_RGB(255,255,255); } return mPresContext->DefaultBackgroundColor(); } void PresShell::UpdateCanvasBackground() { // If we have a frame tree and it has style information that // specifies the background color of the canvas, update our local // cache of that color. nsIFrame* rootStyleFrame = FrameConstructor()->GetRootElementStyleFrame(); if (rootStyleFrame) { nsStyleContext* bgStyle = nsCSSRendering::FindRootFrameBackground(rootStyleFrame); // XXX We should really be passing the canvasframe, not the root element // style frame but we don't have access to the canvasframe here. It isn't // a problem because only a few frames can return something other than true // and none of them would be a canvas frame or root element style frame. bool drawBackgroundImage; bool drawBackgroundColor; mCanvasBackgroundColor = nsCSSRendering::DetermineBackgroundColor(mPresContext, bgStyle, rootStyleFrame, drawBackgroundImage, drawBackgroundColor); mHasCSSBackgroundColor = drawBackgroundColor; if (mPresContext->IsRootContentDocument() && !IsTransparentContainerElement(mPresContext)) { mCanvasBackgroundColor = NS_ComposeColors(GetDefaultBackgroundColorToDraw(), mCanvasBackgroundColor); } } // If the root element of the document (ie html) has style 'display: none' // then the document's background color does not get drawn; cache the // color we actually draw. if (!FrameConstructor()->GetRootElementFrame()) { mCanvasBackgroundColor = GetDefaultBackgroundColorToDraw(); } } nscolor PresShell::ComputeBackstopColor(nsView* aDisplayRoot) { nsIWidget* widget = aDisplayRoot->GetWidget(); if (widget && (widget->GetTransparencyMode() != eTransparencyOpaque || widget->WidgetPaintsBackground())) { // Within a transparent widget, so the backstop color must be // totally transparent. return NS_RGBA(0,0,0,0); } // Within an opaque widget (or no widget at all), so the backstop // color must be totally opaque. The user's default background // as reported by the prescontext is guaranteed to be opaque. return GetDefaultBackgroundColorToDraw(); } struct PaintParams { nscolor mBackgroundColor; }; LayerManager* PresShell::GetLayerManager() { NS_ASSERTION(mViewManager, "Should have view manager"); nsView* rootView = mViewManager->GetRootView(); if (rootView) { if (nsIWidget* widget = rootView->GetWidget()) { return widget->GetLayerManager(); } } return nullptr; } bool PresShell::AsyncPanZoomEnabled() { NS_ASSERTION(mViewManager, "Should have view manager"); nsView* rootView = mViewManager->GetRootView(); if (rootView) { if (nsIWidget* widget = rootView->GetWidget()) { return widget->AsyncPanZoomEnabled(); } } return gfxPlatform::AsyncPanZoomEnabled(); } void PresShell::SetIgnoreViewportScrolling(bool aIgnore) { if (IgnoringViewportScrolling() == aIgnore) { return; } RenderingState state(this); state.mRenderFlags = ChangeFlag(state.mRenderFlags, aIgnore, STATE_IGNORING_VIEWPORT_SCROLLING); SetRenderingState(state); } nsresult PresShell::SetResolutionImpl(float aResolution, bool aScaleToResolution) { if (!(aResolution > 0.0)) { return NS_ERROR_ILLEGAL_VALUE; } if (aResolution == mResolution.valueOr(0.0)) { MOZ_ASSERT(mResolution.isSome()); return NS_OK; } RenderingState state(this); state.mResolution = Some(aResolution); SetRenderingState(state); mScaleToResolution = aScaleToResolution; if (mMobileViewportManager) { mMobileViewportManager->ResolutionUpdated(); } return NS_OK; } bool PresShell::ScaleToResolution() const { return mScaleToResolution; } float PresShell::GetCumulativeResolution() { float resolution = GetResolution(); nsPresContext* parentCtx = GetPresContext()->GetParentPresContext(); if (parentCtx) { resolution *= parentCtx->PresShell()->GetCumulativeResolution(); } return resolution; } float PresShell::GetCumulativeNonRootScaleResolution() { float resolution = 1.0; nsIPresShell* currentShell = this; while (currentShell) { nsPresContext* currentCtx = currentShell->GetPresContext(); if (currentCtx != currentCtx->GetRootPresContext()) { resolution *= currentShell->ScaleToResolution() ? currentShell->GetResolution() : 1.0f; } nsPresContext* parentCtx = currentCtx->GetParentPresContext(); if (parentCtx) { currentShell = parentCtx->PresShell(); } else { currentShell = nullptr; } } return resolution; } void PresShell::SetRestoreResolution(float aResolution, LayoutDeviceIntSize aDisplaySize) { if (mMobileViewportManager) { mMobileViewportManager->SetRestoreResolution(aResolution, aDisplaySize); } } void PresShell::SetRenderingState(const RenderingState& aState) { if (mRenderFlags != aState.mRenderFlags) { // Rendering state changed in a way that forces us to flush any // retained layers we might already have. LayerManager* manager = GetLayerManager(); if (manager) { FrameLayerBuilder::InvalidateAllLayers(manager); } } mRenderFlags = aState.mRenderFlags; mResolution = aState.mResolution; } void PresShell::SynthesizeMouseMove(bool aFromScroll) { if (!sSynthMouseMove) return; if (mPaintingSuppressed || !mIsActive || !mPresContext) { return; } if (!mPresContext->IsRoot()) { nsIPresShell* rootPresShell = GetRootPresShell(); if (rootPresShell) { rootPresShell->SynthesizeMouseMove(aFromScroll); } return; } if (mMouseLocation == nsPoint(NS_UNCONSTRAINEDSIZE, NS_UNCONSTRAINEDSIZE)) return; if (!mSynthMouseMoveEvent.IsPending()) { RefPtr<nsSynthMouseMoveEvent> ev = new nsSynthMouseMoveEvent(this, aFromScroll); if (!GetPresContext()->RefreshDriver()->AddRefreshObserver(ev, Flush_Display)) { NS_WARNING("failed to dispatch nsSynthMouseMoveEvent"); return; } mSynthMouseMoveEvent = ev; } } /** * Find the first floating view with a widget in a postorder traversal of the * view tree that contains the point. Thus more deeply nested floating views * are preferred over their ancestors, and floating views earlier in the * view hierarchy (i.e., added later) are preferred over their siblings. * This is adequate for finding the "topmost" floating view under a point, * given that floating views don't supporting having a specific z-index. * * We cannot exit early when aPt is outside the view bounds, because floating * views aren't necessarily included in their parent's bounds, so this could * traverse the entire view hierarchy --- use carefully. */ static nsView* FindFloatingViewContaining(nsView* aView, nsPoint aPt) { if (aView->GetVisibility() == nsViewVisibility_kHide) // No need to look into descendants. return nullptr; nsIFrame* frame = aView->GetFrame(); if (frame) { if (!frame->IsVisibleConsideringAncestors(nsIFrame::VISIBILITY_CROSS_CHROME_CONTENT_BOUNDARY) || !frame->PresContext()->PresShell()->IsActive()) { return nullptr; } } for (nsView* v = aView->GetFirstChild(); v; v = v->GetNextSibling()) { nsView* r = FindFloatingViewContaining(v, v->ConvertFromParentCoords(aPt)); if (r) return r; } if (aView->GetFloating() && aView->HasWidget() && aView->GetDimensions().Contains(aPt)) return aView; return nullptr; } /* * This finds the first view containing the given point in a postorder * traversal of the view tree that contains the point, assuming that the * point is not in a floating view. It assumes that only floating views * extend outside the bounds of their parents. * * This methods should only be called if FindFloatingViewContaining * returns null. */ static nsView* FindViewContaining(nsView* aView, nsPoint aPt) { if (!aView->GetDimensions().Contains(aPt) || aView->GetVisibility() == nsViewVisibility_kHide) { return nullptr; } nsIFrame* frame = aView->GetFrame(); if (frame) { if (!frame->IsVisibleConsideringAncestors(nsIFrame::VISIBILITY_CROSS_CHROME_CONTENT_BOUNDARY) || !frame->PresContext()->PresShell()->IsActive()) { return nullptr; } } for (nsView* v = aView->GetFirstChild(); v; v = v->GetNextSibling()) { nsView* r = FindViewContaining(v, v->ConvertFromParentCoords(aPt)); if (r) return r; } return aView; } void PresShell::ProcessSynthMouseMoveEvent(bool aFromScroll) { // If drag session has started, we shouldn't synthesize mousemove event. nsCOMPtr<nsIDragSession> dragSession = nsContentUtils::GetDragSession(); if (dragSession) { mSynthMouseMoveEvent.Forget(); return; } // allow new event to be posted while handling this one only if the // source of the event is a scroll (to prevent infinite reflow loops) if (aFromScroll) { mSynthMouseMoveEvent.Forget(); } nsView* rootView = mViewManager ? mViewManager->GetRootView() : nullptr; if (mMouseLocation == nsPoint(NS_UNCONSTRAINEDSIZE, NS_UNCONSTRAINEDSIZE) || !rootView || !rootView->HasWidget() || !mPresContext) { mSynthMouseMoveEvent.Forget(); return; } NS_ASSERTION(mPresContext->IsRoot(), "Only a root pres shell should be here"); // Hold a ref to ourselves so DispatchEvent won't destroy us (since // we need to access members after we call DispatchEvent). nsCOMPtr<nsIPresShell> kungFuDeathGrip(this); #ifdef DEBUG_MOUSE_LOCATION printf("[ps=%p]synthesizing mouse move to (%d,%d)\n", this, mMouseLocation.x, mMouseLocation.y); #endif int32_t APD = mPresContext->AppUnitsPerDevPixel(); // We need a widget to put in the event we are going to dispatch so we look // for a view that has a widget and the mouse location is over. We first look // for floating views, if there isn't one we use the root view. |view| holds // that view. nsView* view = nullptr; // The appunits per devpixel ratio of |view|. int32_t viewAPD; // mRefPoint will be mMouseLocation relative to the widget of |view|, the // widget we will put in the event we dispatch, in viewAPD appunits nsPoint refpoint(0, 0); // We always dispatch the event to the pres shell that contains the view that // the mouse is over. pointVM is the VM of that pres shell. nsViewManager *pointVM = nullptr; // This could be a bit slow (traverses entire view hierarchy) // but it's OK to do it once per synthetic mouse event view = FindFloatingViewContaining(rootView, mMouseLocation); if (!view) { view = rootView; nsView *pointView = FindViewContaining(rootView, mMouseLocation); // pointView can be null in situations related to mouse capture pointVM = (pointView ? pointView : view)->GetViewManager(); refpoint = mMouseLocation + rootView->ViewToWidgetOffset(); viewAPD = APD; } else { pointVM = view->GetViewManager(); nsIFrame* frame = view->GetFrame(); NS_ASSERTION(frame, "floating views can't be anonymous"); viewAPD = frame->PresContext()->AppUnitsPerDevPixel(); refpoint = mMouseLocation.ScaleToOtherAppUnits(APD, viewAPD); refpoint -= view->GetOffsetTo(rootView); refpoint += view->ViewToWidgetOffset(); } NS_ASSERTION(view->GetWidget(), "view should have a widget here"); WidgetMouseEvent event(true, eMouseMove, view->GetWidget(), WidgetMouseEvent::eSynthesized); event.mRefPoint = LayoutDeviceIntPoint::FromAppUnitsToNearest(refpoint, viewAPD); event.mTime = PR_IntervalNow(); // XXX set event.mModifiers ? // XXX mnakano I think that we should get the latest information from widget. nsCOMPtr<nsIPresShell> shell = pointVM->GetPresShell(); if (shell) { // Since this gets run in a refresh tick there isn't an InputAPZContext on // the stack from the nsBaseWidget. We need to simulate one with at least // the correct target guid, so that the correct callback transform gets // applied if this event goes to a child process. The input block id is set // to 0 because this is a synthetic event which doesn't really belong to any // input block. Same for the APZ response field. InputAPZContext apzContext(mMouseEventTargetGuid, 0, nsEventStatus_eIgnore); shell->DispatchSynthMouseMove(&event, !aFromScroll); } if (!aFromScroll) { mSynthMouseMoveEvent.Forget(); } } static void AddFrameToVisibleRegions(nsIFrame* aFrame, nsViewManager* aViewManager, Maybe<VisibleRegions>& aVisibleRegions) { if (!aVisibleRegions) { return; } MOZ_ASSERT(aFrame); MOZ_ASSERT(aViewManager); // Retrieve the view ID for this frame (which we obtain from the enclosing // scrollable frame). nsIScrollableFrame* scrollableFrame = nsLayoutUtils::GetNearestScrollableFrame(aFrame, nsLayoutUtils::SCROLLABLE_ONLY_ASYNC_SCROLLABLE | nsLayoutUtils::SCROLLABLE_ALWAYS_MATCH_ROOT); if (!scrollableFrame) { return; } nsIFrame* scrollableFrameAsFrame = do_QueryFrame(scrollableFrame); MOZ_ASSERT(scrollableFrameAsFrame); nsIContent* scrollableFrameContent = scrollableFrameAsFrame->GetContent(); if (!scrollableFrameContent) { return; } ViewID viewID; if (!nsLayoutUtils::FindIDFor(scrollableFrameContent, &viewID)) { return ; } // Update the visible region for the appropriate view ID. nsRect frameRectInScrolledFrameSpace = aFrame->GetVisualOverflowRect(); nsLayoutUtils::TransformResult result = nsLayoutUtils::TransformRect(aFrame, scrollableFrame->GetScrolledFrame(), frameRectInScrolledFrameSpace); if (result != nsLayoutUtils::TransformResult::TRANSFORM_SUCCEEDED) { return; } CSSIntRegion* regionForView = aVisibleRegions->LookupOrAdd(viewID); MOZ_ASSERT(regionForView); regionForView->OrWith(CSSPixel::FromAppUnitsRounded(frameRectInScrolledFrameSpace)); } /* static */ void PresShell::MarkFramesInListApproximatelyVisible(const nsDisplayList& aList, Maybe<VisibleRegions>& aVisibleRegions) { for (nsDisplayItem* item = aList.GetBottom(); item; item = item->GetAbove()) { nsDisplayList* sublist = item->GetChildren(); if (sublist) { MarkFramesInListApproximatelyVisible(*sublist, aVisibleRegions); continue; } nsIFrame* frame = item->Frame(); MOZ_ASSERT(frame); if (!frame->TrackingVisibility()) { continue; } // Use the presshell containing the frame. auto* presShell = static_cast<PresShell*>(frame->PresContext()->PresShell()); uint32_t count = presShell->mApproximatelyVisibleFrames.Count(); MOZ_ASSERT(!presShell->AssumeAllFramesVisible()); presShell->mApproximatelyVisibleFrames.PutEntry(frame); if (presShell->mApproximatelyVisibleFrames.Count() > count) { // The frame was added to mApproximatelyVisibleFrames, so increment its visible count. frame->IncApproximateVisibleCount(); } AddFrameToVisibleRegions(frame, presShell->mViewManager, aVisibleRegions); } } static void NotifyCompositorOfVisibleRegionsChange(PresShell* aPresShell, const Maybe<VisibleRegions>& aRegions) { if (!aRegions) { return; } MOZ_ASSERT(aPresShell); // Retrieve the layers ID and pres shell ID. TabChild* tabChild = TabChild::GetFrom(aPresShell); if (!tabChild) { return; } const uint64_t layersId = tabChild->LayersId(); const uint32_t presShellId = aPresShell->GetPresShellId(); // Retrieve the CompositorBridgeChild. LayerManager* layerManager = aPresShell->GetLayerManager(); if (!layerManager) { return; } ClientLayerManager* clientLayerManager = layerManager->AsClientLayerManager(); if (!clientLayerManager) { return; } CompositorBridgeChild* compositorChild = clientLayerManager->GetCompositorBridgeChild(); if (!compositorChild) { return; } // Clear the old approximately visible regions associated with this document. compositorChild->SendClearApproximatelyVisibleRegions(layersId, presShellId); // Send the new approximately visible regions to the compositor. for (auto iter = aRegions->ConstIter(); !iter.Done(); iter.Next()) { const ViewID viewId = iter.Key(); const CSSIntRegion* region = iter.UserData(); MOZ_ASSERT(region); const ScrollableLayerGuid guid(layersId, presShellId, viewId); compositorChild->SendNotifyApproximatelyVisibleRegion(guid, *region); } } /* static */ void PresShell::DecApproximateVisibleCount(VisibleFrames& aFrames, Maybe<OnNonvisible> aNonvisibleAction /* = Nothing() */) { for (auto iter = aFrames.Iter(); !iter.Done(); iter.Next()) { nsIFrame* frame = iter.Get()->GetKey(); // Decrement the frame's visible count if we're still tracking its // visibility. (We may not be, if the frame disabled visibility tracking // after we added it to the visible frames list.) if (frame->TrackingVisibility()) { frame->DecApproximateVisibleCount(aNonvisibleAction); } } } void PresShell::RebuildApproximateFrameVisibilityDisplayList(const nsDisplayList& aList) { MOZ_ASSERT(!mApproximateFrameVisibilityVisited, "already visited?"); mApproximateFrameVisibilityVisited = true; // Remove the entries of the mApproximatelyVisibleFrames hashtable and put // them in oldApproxVisibleFrames. VisibleFrames oldApproximatelyVisibleFrames; mApproximatelyVisibleFrames.SwapElements(oldApproximatelyVisibleFrames); // If we're visualizing visible regions, create a VisibleRegions object to // store information about them. The functions we call will populate this // object and send it to the compositor only if it's Some(), so we don't // need to check the prefs everywhere. Maybe<VisibleRegions> visibleRegions; if (gfxPrefs::APZMinimap() && gfxPrefs::APZMinimapVisibilityEnabled()) { visibleRegions.emplace(); } MarkFramesInListApproximatelyVisible(aList, visibleRegions); DecApproximateVisibleCount(oldApproximatelyVisibleFrames); NotifyCompositorOfVisibleRegionsChange(this, visibleRegions); } /* static */ void PresShell::ClearApproximateFrameVisibilityVisited(nsView* aView, bool aClear) { nsViewManager* vm = aView->GetViewManager(); if (aClear) { PresShell* presShell = static_cast<PresShell*>(vm->GetPresShell()); if (!presShell->mApproximateFrameVisibilityVisited) { presShell->ClearApproximatelyVisibleFramesList(); } presShell->mApproximateFrameVisibilityVisited = false; } for (nsView* v = aView->GetFirstChild(); v; v = v->GetNextSibling()) { ClearApproximateFrameVisibilityVisited(v, v->GetViewManager() != vm); } } void PresShell::ClearApproximatelyVisibleFramesList(Maybe<OnNonvisible> aNonvisibleAction /* = Nothing() */) { DecApproximateVisibleCount(mApproximatelyVisibleFrames, aNonvisibleAction); mApproximatelyVisibleFrames.Clear(); } void PresShell::MarkFramesInSubtreeApproximatelyVisible(nsIFrame* aFrame, const nsRect& aRect, Maybe<VisibleRegions>& aVisibleRegions, bool aRemoveOnly /* = false */) { MOZ_ASSERT(aFrame->PresContext()->PresShell() == this, "wrong presshell"); if (aFrame->TrackingVisibility() && aFrame->StyleVisibility()->IsVisible() && (!aRemoveOnly || aFrame->GetVisibility() == Visibility::APPROXIMATELY_VISIBLE)) { MOZ_ASSERT(!AssumeAllFramesVisible()); uint32_t count = mApproximatelyVisibleFrames.Count(); mApproximatelyVisibleFrames.PutEntry(aFrame); if (mApproximatelyVisibleFrames.Count() > count) { // The frame was added to mApproximatelyVisibleFrames, so increment its visible count. aFrame->IncApproximateVisibleCount(); } AddFrameToVisibleRegions(aFrame, mViewManager, aVisibleRegions); } nsSubDocumentFrame* subdocFrame = do_QueryFrame(aFrame); if (subdocFrame) { nsIPresShell* presShell = subdocFrame->GetSubdocumentPresShellForPainting( nsSubDocumentFrame::IGNORE_PAINT_SUPPRESSION); if (presShell && !presShell->AssumeAllFramesVisible()) { nsRect rect = aRect; nsIFrame* root = presShell->GetRootFrame(); if (root) { rect.MoveBy(aFrame->GetOffsetToCrossDoc(root)); } else { rect.MoveBy(-aFrame->GetContentRectRelativeToSelf().TopLeft()); } rect = rect.ScaleToOtherAppUnitsRoundOut( aFrame->PresContext()->AppUnitsPerDevPixel(), presShell->GetPresContext()->AppUnitsPerDevPixel()); presShell->RebuildApproximateFrameVisibility(&rect); } return; } nsRect rect = aRect; nsIScrollableFrame* scrollFrame = do_QueryFrame(aFrame); if (scrollFrame) { scrollFrame->NotifyApproximateFrameVisibilityUpdate(); nsRect displayPort; bool usingDisplayport = nsLayoutUtils::GetDisplayPortForVisibilityTesting( aFrame->GetContent(), &displayPort, RelativeTo::ScrollFrame); if (usingDisplayport) { rect = displayPort; } else { rect = rect.Intersect(scrollFrame->GetScrollPortRect()); } rect = scrollFrame->ExpandRectToNearlyVisible(rect); } bool preserves3DChildren = aFrame->Extend3DContext(); // We assume all frames in popups are visible, so we skip them here. const nsIFrame::ChildListIDs skip(nsIFrame::kPopupList | nsIFrame::kSelectPopupList); for (nsIFrame::ChildListIterator childLists(aFrame); !childLists.IsDone(); childLists.Next()) { if (skip.Contains(childLists.CurrentID())) { continue; } for (nsIFrame* child : childLists.CurrentList()) { nsRect r = rect - child->GetPosition(); if (!r.IntersectRect(r, child->GetVisualOverflowRect())) { continue; } if (child->IsTransformed()) { // for children of a preserve3d element we just pass down the same dirty rect if (!preserves3DChildren || !child->Combines3DTransformWithAncestors()) { const nsRect overflow = child->GetVisualOverflowRectRelativeToSelf(); nsRect out; if (nsDisplayTransform::UntransformRect(r, overflow, child, &out)) { r = out; } else { r.SetEmpty(); } } } MarkFramesInSubtreeApproximatelyVisible(child, r, aVisibleRegions); } } } void PresShell::RebuildApproximateFrameVisibility(nsRect* aRect, bool aRemoveOnly /* = false */) { MOZ_ASSERT(!mApproximateFrameVisibilityVisited, "already visited?"); mApproximateFrameVisibilityVisited = true; nsIFrame* rootFrame = GetRootFrame(); if (!rootFrame) { return; } // Remove the entries of the mApproximatelyVisibleFrames hashtable and put // them in oldApproximatelyVisibleFrames. VisibleFrames oldApproximatelyVisibleFrames; mApproximatelyVisibleFrames.SwapElements(oldApproximatelyVisibleFrames); // If we're visualizing visible regions, create a VisibleRegions object to // store information about them. The functions we call will populate this // object and send it to the compositor only if it's Some(), so we don't // need to check the prefs everywhere. Maybe<VisibleRegions> visibleRegions; if (gfxPrefs::APZMinimap() && gfxPrefs::APZMinimapVisibilityEnabled()) { visibleRegions.emplace(); } nsRect vis(nsPoint(0, 0), rootFrame->GetSize()); if (aRect) { vis = *aRect; } MarkFramesInSubtreeApproximatelyVisible(rootFrame, vis, visibleRegions, aRemoveOnly); DecApproximateVisibleCount(oldApproximatelyVisibleFrames); NotifyCompositorOfVisibleRegionsChange(this, visibleRegions); } void PresShell::UpdateApproximateFrameVisibility() { DoUpdateApproximateFrameVisibility(/* aRemoveOnly = */ false); } void PresShell::DoUpdateApproximateFrameVisibility(bool aRemoveOnly) { MOZ_ASSERT(!mPresContext || mPresContext->IsRootContentDocument(), "Updating approximate frame visibility on a non-root content document?"); mUpdateApproximateFrameVisibilityEvent.Revoke(); if (mHaveShutDown || mIsDestroying) { return; } // call update on that frame nsIFrame* rootFrame = GetRootFrame(); if (!rootFrame) { ClearApproximatelyVisibleFramesList(Some(OnNonvisible::DISCARD_IMAGES)); return; } RebuildApproximateFrameVisibility(/* aRect = */ nullptr, aRemoveOnly); ClearApproximateFrameVisibilityVisited(rootFrame->GetView(), true); #ifdef DEBUG_FRAME_VISIBILITY_DISPLAY_LIST // This can be used to debug the frame walker by comparing beforeFrameList // and mApproximatelyVisibleFrames in RebuildFrameVisibilityDisplayList to see if // they produce the same results (mApproximatelyVisibleFrames holds the frames the // display list thinks are visible, beforeFrameList holds the frames the // frame walker thinks are visible). nsDisplayListBuilder builder(rootFrame, nsDisplayListBuilderMode::FRAME_VISIBILITY, false); nsRect updateRect(nsPoint(0, 0), rootFrame->GetSize()); nsIFrame* rootScroll = GetRootScrollFrame(); if (rootScroll) { nsIContent* content = rootScroll->GetContent(); if (content) { Unused << nsLayoutUtils::GetDisplayPortForVisibilityTesting(content, &updateRect, RelativeTo::ScrollFrame); } if (IgnoringViewportScrolling()) { builder.SetIgnoreScrollFrame(rootScroll); } } builder.IgnorePaintSuppression(); builder.EnterPresShell(rootFrame); nsDisplayList list; rootFrame->BuildDisplayListForStackingContext(&builder, updateRect, &list); builder.LeavePresShell(rootFrame, &list); RebuildApproximateFrameVisibilityDisplayList(list); ClearApproximateFrameVisibilityVisited(rootFrame->GetView(), true); list.DeleteAll(); #endif } bool PresShell::AssumeAllFramesVisible() { static bool sFrameVisibilityEnabled = true; static bool sFrameVisibilityPrefCached = false; if (!sFrameVisibilityPrefCached) { Preferences::AddBoolVarCache(&sFrameVisibilityEnabled, "layout.framevisibility.enabled", true); sFrameVisibilityPrefCached = true; } if (!sFrameVisibilityEnabled || !mPresContext || !mDocument) { return true; } // We assume all frames are visible in print, print preview, chrome, and // resource docs and don't keep track of them. if (mPresContext->Type() == nsPresContext::eContext_PrintPreview || mPresContext->Type() == nsPresContext::eContext_Print || mPresContext->IsChrome() || mDocument->IsResourceDoc()) { return true; } // If we're assuming all frames are visible in the top level content // document, we need to in subdocuments as well. Otherwise we can get in a // situation where things like animations won't work in subdocuments because // their frames appear not to be visible, since we won't schedule an image // visibility update if the top level content document is assuming all // frames are visible. // // Note that it's not safe to call IsRootContentDocument() if we're // currently being destroyed, so we have to check that first. if (!mHaveShutDown && !mIsDestroying && !mPresContext->IsRootContentDocument()) { nsPresContext* presContext = mPresContext->GetToplevelContentDocumentPresContext(); if (presContext && presContext->PresShell()->AssumeAllFramesVisible()) { return true; } } return false; } void PresShell::ScheduleApproximateFrameVisibilityUpdateSoon() { if (AssumeAllFramesVisible()) { return; } if (!mPresContext) { return; } nsRefreshDriver* refreshDriver = mPresContext->RefreshDriver(); if (!refreshDriver) { return; } // Ask the refresh driver to update frame visibility soon. refreshDriver->ScheduleFrameVisibilityUpdate(); } void PresShell::ScheduleApproximateFrameVisibilityUpdateNow() { if (AssumeAllFramesVisible()) { return; } if (!mPresContext->IsRootContentDocument()) { nsPresContext* presContext = mPresContext->GetToplevelContentDocumentPresContext(); if (!presContext) return; MOZ_ASSERT(presContext->IsRootContentDocument(), "Didn't get a root prescontext from GetToplevelContentDocumentPresContext?"); presContext->PresShell()->ScheduleApproximateFrameVisibilityUpdateNow(); return; } if (mHaveShutDown || mIsDestroying) { return; } if (mUpdateApproximateFrameVisibilityEvent.IsPending()) { return; } RefPtr<nsRunnableMethod<PresShell> > ev = NewRunnableMethod(this, &PresShell::UpdateApproximateFrameVisibility); if (NS_SUCCEEDED(NS_DispatchToCurrentThread(ev))) { mUpdateApproximateFrameVisibilityEvent = ev; } } void PresShell::EnsureFrameInApproximatelyVisibleList(nsIFrame* aFrame) { if (!aFrame->TrackingVisibility()) { return; } if (AssumeAllFramesVisible()) { aFrame->IncApproximateVisibleCount(); return; } #ifdef DEBUG // Make sure it's in this pres shell. nsCOMPtr<nsIContent> content = aFrame->GetContent(); if (content) { PresShell* shell = static_cast<PresShell*>(content->OwnerDoc()->GetShell()); MOZ_ASSERT(!shell || shell == this, "wrong shell"); } #endif if (!mApproximatelyVisibleFrames.Contains(aFrame)) { MOZ_ASSERT(!AssumeAllFramesVisible()); mApproximatelyVisibleFrames.PutEntry(aFrame); aFrame->IncApproximateVisibleCount(); } } void PresShell::RemoveFrameFromApproximatelyVisibleList(nsIFrame* aFrame) { #ifdef DEBUG // Make sure it's in this pres shell. nsCOMPtr<nsIContent> content = aFrame->GetContent(); if (content) { PresShell* shell = static_cast<PresShell*>(content->OwnerDoc()->GetShell()); MOZ_ASSERT(!shell || shell == this, "wrong shell"); } #endif if (AssumeAllFramesVisible()) { MOZ_ASSERT(mApproximatelyVisibleFrames.Count() == 0, "Shouldn't have any frames in the table"); return; } uint32_t count = mApproximatelyVisibleFrames.Count(); mApproximatelyVisibleFrames.RemoveEntry(aFrame); if (aFrame->TrackingVisibility() && mApproximatelyVisibleFrames.Count() < count) { // aFrame was in the hashtable, and we're still tracking its visibility, // so we need to decrement its visible count. aFrame->DecApproximateVisibleCount(); } } class nsAutoNotifyDidPaint { public: nsAutoNotifyDidPaint(PresShell* aShell, uint32_t aFlags) : mShell(aShell), mFlags(aFlags) { } ~nsAutoNotifyDidPaint() { mShell->GetPresContext()->NotifyDidPaintForSubtree(mFlags); } private: PresShell* mShell; uint32_t mFlags; }; void PresShell::RecordShadowStyleChange(ShadowRoot* aShadowRoot) { mChangedScopeStyleRoots.AppendElement(aShadowRoot->GetHost()->AsElement()); } void PresShell::Paint(nsView* aViewToPaint, const nsRegion& aDirtyRegion, uint32_t aFlags) { PROFILER_LABEL("PresShell", "Paint", js::ProfileEntry::Category::GRAPHICS); NS_ASSERTION(!mIsDestroying, "painting a destroyed PresShell"); NS_ASSERTION(aViewToPaint, "null view"); MOZ_ASSERT(!mApproximateFrameVisibilityVisited, "Should have been cleared"); if (!mIsActive || mIsZombie) { return; } nsPresContext* presContext = GetPresContext(); AUTO_LAYOUT_PHASE_ENTRY_POINT(presContext, Paint); nsIFrame* frame = aViewToPaint->GetFrame(); LayerManager* layerManager = aViewToPaint->GetWidget()->GetLayerManager(); NS_ASSERTION(layerManager, "Must be in paint event"); bool shouldInvalidate = layerManager->NeedsWidgetInvalidation(); nsAutoNotifyDidPaint notifyDidPaint(this, aFlags); // Whether or not we should set first paint when painting is // suppressed is debatable. For now we'll do it because // B2G relies on first paint to configure the viewport and // we only want to do that when we have real content to paint. // See Bug 798245 if (mIsFirstPaint && !mPaintingSuppressed) { layerManager->SetIsFirstPaint(); mIsFirstPaint = false; } if (!layerManager->BeginTransaction()) { return; } if (frame) { // Try to do an empty transaction, if the frame tree does not // need to be updated. Do not try to do an empty transaction on // a non-retained layer manager (like the BasicLayerManager that // draws the window title bar on Mac), because a) it won't work // and b) below we don't want to clear NS_FRAME_UPDATE_LAYER_TREE, // that will cause us to forget to update the real layer manager! if (!(aFlags & PAINT_LAYERS)) { if (layerManager->EndEmptyTransaction()) { return; } NS_WARNING("Must complete empty transaction when compositing!"); } if (!(aFlags & PAINT_SYNC_DECODE_IMAGES) && !(frame->GetStateBits() & NS_FRAME_UPDATE_LAYER_TREE) && !mNextPaintCompressed) { NotifySubDocInvalidationFunc computeInvalidFunc = presContext->MayHavePaintEventListenerInSubDocument() ? nsPresContext::NotifySubDocInvalidation : 0; bool computeInvalidRect = computeInvalidFunc || (layerManager->GetBackendType() == LayersBackend::LAYERS_BASIC); UniquePtr<LayerProperties> props; if (computeInvalidRect) { props = Move(LayerProperties::CloneFrom(layerManager->GetRoot())); } MaybeSetupTransactionIdAllocator(layerManager, aViewToPaint); if (layerManager->EndEmptyTransaction((aFlags & PAINT_COMPOSITE) ? LayerManager::END_DEFAULT : LayerManager::END_NO_COMPOSITE)) { nsIntRegion invalid; if (props) { invalid = props->ComputeDifferences(layerManager->GetRoot(), computeInvalidFunc); } else { LayerProperties::ClearInvalidations(layerManager->GetRoot()); } if (props) { if (!invalid.IsEmpty()) { nsIntRect bounds = invalid.GetBounds(); nsRect rect(presContext->DevPixelsToAppUnits(bounds.x), presContext->DevPixelsToAppUnits(bounds.y), presContext->DevPixelsToAppUnits(bounds.width), presContext->DevPixelsToAppUnits(bounds.height)); if (shouldInvalidate) { aViewToPaint->GetViewManager()->InvalidateViewNoSuppression(aViewToPaint, rect); } presContext->NotifyInvalidation(bounds, 0); } } else if (shouldInvalidate) { aViewToPaint->GetViewManager()->InvalidateView(aViewToPaint); } frame->UpdatePaintCountForPaintedPresShells(); return; } } frame->RemoveStateBits(NS_FRAME_UPDATE_LAYER_TREE); } if (frame) { frame->ClearPresShellsFromLastPaint(); } nscolor bgcolor = ComputeBackstopColor(aViewToPaint); PaintFrameFlags flags = PaintFrameFlags::PAINT_WIDGET_LAYERS | PaintFrameFlags::PAINT_EXISTING_TRANSACTION; if (!(aFlags & PAINT_COMPOSITE)) { flags |= PaintFrameFlags::PAINT_NO_COMPOSITE; } if (aFlags & PAINT_SYNC_DECODE_IMAGES) { flags |= PaintFrameFlags::PAINT_SYNC_DECODE_IMAGES; } if (mNextPaintCompressed) { flags |= PaintFrameFlags::PAINT_COMPRESSED; mNextPaintCompressed = false; } if (frame) { // We can paint directly into the widget using its layer manager. nsLayoutUtils::PaintFrame(nullptr, frame, aDirtyRegion, bgcolor, nsDisplayListBuilderMode::PAINTING, flags); return; } RefPtr<ColorLayer> root = layerManager->CreateColorLayer(); if (root) { nsPresContext* pc = GetPresContext(); nsIntRect bounds = pc->GetVisibleArea().ToOutsidePixels(pc->AppUnitsPerDevPixel()); bgcolor = NS_ComposeColors(bgcolor, mCanvasBackgroundColor); root->SetColor(Color::FromABGR(bgcolor)); root->SetVisibleRegion(LayerIntRegion::FromUnknownRegion(bounds)); layerManager->SetRoot(root); } MaybeSetupTransactionIdAllocator(layerManager, aViewToPaint); layerManager->EndTransaction(nullptr, nullptr, (aFlags & PAINT_COMPOSITE) ? LayerManager::END_DEFAULT : LayerManager::END_NO_COMPOSITE); } // static void nsIPresShell::SetCapturingContent(nsIContent* aContent, uint8_t aFlags) { // If capture was set for pointer lock, don't unlock unless we are coming // out of pointer lock explicitly. if (!aContent && gCaptureInfo.mPointerLock && !(aFlags & CAPTURE_POINTERLOCK)) { return; } gCaptureInfo.mContent = nullptr; // only set capturing content if allowed or the CAPTURE_IGNOREALLOWED or // CAPTURE_POINTERLOCK flags are used. if ((aFlags & CAPTURE_IGNOREALLOWED) || gCaptureInfo.mAllowed || (aFlags & CAPTURE_POINTERLOCK)) { if (aContent) { gCaptureInfo.mContent = aContent; } // CAPTURE_POINTERLOCK is the same as CAPTURE_RETARGETTOELEMENT & CAPTURE_IGNOREALLOWED gCaptureInfo.mRetargetToElement = ((aFlags & CAPTURE_RETARGETTOELEMENT) != 0) || ((aFlags & CAPTURE_POINTERLOCK) != 0); gCaptureInfo.mPreventDrag = (aFlags & CAPTURE_PREVENTDRAG) != 0; gCaptureInfo.mPointerLock = (aFlags & CAPTURE_POINTERLOCK) != 0; } } /* static */ void nsIPresShell::SetPointerCapturingContent(uint32_t aPointerId, nsIContent* aContent) { MOZ_ASSERT(aContent != nullptr); if (nsIDOMMouseEvent::MOZ_SOURCE_MOUSE == GetPointerType(aPointerId)) { SetCapturingContent(aContent, CAPTURE_PREVENTDRAG); } PointerCaptureInfo* pointerCaptureInfo = GetPointerCaptureInfo(aPointerId); if (pointerCaptureInfo) { pointerCaptureInfo->mPendingContent = aContent; } else { sPointerCaptureList->Put(aPointerId, new PointerCaptureInfo(aContent)); } } /* static */ nsIPresShell::PointerCaptureInfo* nsIPresShell::GetPointerCaptureInfo(uint32_t aPointerId) { PointerCaptureInfo* pointerCaptureInfo = nullptr; sPointerCaptureList->Get(aPointerId, &pointerCaptureInfo); return pointerCaptureInfo; } /* static */ void nsIPresShell::ReleasePointerCapturingContent(uint32_t aPointerId) { if (nsIDOMMouseEvent::MOZ_SOURCE_MOUSE == GetPointerType(aPointerId)) { SetCapturingContent(nullptr, CAPTURE_PREVENTDRAG); } PointerCaptureInfo* pointerCaptureInfo = GetPointerCaptureInfo(aPointerId); if (pointerCaptureInfo) { pointerCaptureInfo->mPendingContent = nullptr; } } /* static */ nsIContent* nsIPresShell::GetPointerCapturingContent(uint32_t aPointerId) { PointerCaptureInfo* pointerCaptureInfo = GetPointerCaptureInfo(aPointerId); if (pointerCaptureInfo) { return pointerCaptureInfo->mOverrideContent; } return nullptr; } /* static */ void nsIPresShell::CheckPointerCaptureState(const WidgetPointerEvent* aPointerEvent) { PointerCaptureInfo* captureInfo = GetPointerCaptureInfo(aPointerEvent->pointerId); if (captureInfo && captureInfo->mPendingContent != captureInfo->mOverrideContent) { // cache captureInfo->mPendingContent since it may be changed in the pointer // event listener nsIContent* pendingContent = captureInfo->mPendingContent.get(); if (captureInfo->mOverrideContent) { DispatchGotOrLostPointerCaptureEvent(/* aIsGotCapture */ false, aPointerEvent, captureInfo->mOverrideContent); } if (pendingContent) { DispatchGotOrLostPointerCaptureEvent(/* aIsGotCapture */ true, aPointerEvent, pendingContent); } captureInfo->mOverrideContent = pendingContent; if (captureInfo->Empty()) { sPointerCaptureList->Remove(aPointerEvent->pointerId); } } } /* static */ uint16_t nsIPresShell::GetPointerType(uint32_t aPointerId) { PointerInfo* pointerInfo = nullptr; if (sActivePointersIds->Get(aPointerId, &pointerInfo) && pointerInfo) { return pointerInfo->mPointerType; } return nsIDOMMouseEvent::MOZ_SOURCE_UNKNOWN; } /* static */ bool nsIPresShell::GetPointerPrimaryState(uint32_t aPointerId) { PointerInfo* pointerInfo = nullptr; if (sActivePointersIds->Get(aPointerId, &pointerInfo) && pointerInfo) { return pointerInfo->mPrimaryState; } return false; } /* static */ bool nsIPresShell::GetPointerInfo(uint32_t aPointerId, bool& aActiveState) { PointerInfo* pointerInfo = nullptr; if (sActivePointersIds->Get(aPointerId, &pointerInfo) && pointerInfo) { aActiveState = pointerInfo->mActiveState; return true; } return false; } void PresShell::UpdateActivePointerState(WidgetGUIEvent* aEvent) { switch (aEvent->mMessage) { case eMouseEnterIntoWidget: // In this case we have to know information about available mouse pointers if (WidgetMouseEvent* mouseEvent = aEvent->AsMouseEvent()) { sActivePointersIds->Put(mouseEvent->pointerId, new PointerInfo(false, mouseEvent->inputSource, true)); } break; case ePointerDown: // In this case we switch pointer to active state if (WidgetPointerEvent* pointerEvent = aEvent->AsPointerEvent()) { sActivePointersIds->Put(pointerEvent->pointerId, new PointerInfo(true, pointerEvent->inputSource, pointerEvent->mIsPrimary)); } break; case ePointerUp: // In this case we remove information about pointer or turn off active state if (WidgetPointerEvent* pointerEvent = aEvent->AsPointerEvent()) { if(pointerEvent->inputSource != nsIDOMMouseEvent::MOZ_SOURCE_TOUCH) { sActivePointersIds->Put(pointerEvent->pointerId, new PointerInfo(false, pointerEvent->inputSource, pointerEvent->mIsPrimary)); } else { sActivePointersIds->Remove(pointerEvent->pointerId); } } break; case eMouseExitFromWidget: // In this case we have to remove information about disappeared mouse // pointers if (WidgetMouseEvent* mouseEvent = aEvent->AsMouseEvent()) { sActivePointersIds->Remove(mouseEvent->pointerId); } break; default: break; } } nsIContent* PresShell::GetCurrentEventContent() { if (mCurrentEventContent && mCurrentEventContent->GetComposedDoc() != mDocument) { mCurrentEventContent = nullptr; mCurrentEventFrame = nullptr; } return mCurrentEventContent; } nsIFrame* PresShell::GetCurrentEventFrame() { if (MOZ_UNLIKELY(mIsDestroying)) { return nullptr; } // GetCurrentEventContent() makes sure the content is still in the // same document that this pres shell belongs to. If not, then the // frame shouldn't get an event, nor should we even assume its safe // to try and find the frame. nsIContent* content = GetCurrentEventContent(); if (!mCurrentEventFrame && content) { mCurrentEventFrame = content->GetPrimaryFrame(); MOZ_ASSERT(!mCurrentEventFrame || mCurrentEventFrame->PresContext()->GetPresShell() == this); } return mCurrentEventFrame; } nsIFrame* PresShell::GetEventTargetFrame() { return GetCurrentEventFrame(); } already_AddRefed<nsIContent> PresShell::GetEventTargetContent(WidgetEvent* aEvent) { nsCOMPtr<nsIContent> content = GetCurrentEventContent(); if (!content) { nsIFrame* currentEventFrame = GetCurrentEventFrame(); if (currentEventFrame) { currentEventFrame->GetContentForEvent(aEvent, getter_AddRefs(content)); NS_ASSERTION(!content || content->GetComposedDoc() == mDocument, "handing out content from a different doc"); } } return content.forget(); } void PresShell::PushCurrentEventInfo(nsIFrame* aFrame, nsIContent* aContent) { if (mCurrentEventFrame || mCurrentEventContent) { mCurrentEventFrameStack.InsertElementAt(0, mCurrentEventFrame); mCurrentEventContentStack.InsertObjectAt(mCurrentEventContent, 0); } mCurrentEventFrame = aFrame; mCurrentEventContent = aContent; } void PresShell::PopCurrentEventInfo() { mCurrentEventFrame = nullptr; mCurrentEventContent = nullptr; if (0 != mCurrentEventFrameStack.Length()) { mCurrentEventFrame = mCurrentEventFrameStack.ElementAt(0); mCurrentEventFrameStack.RemoveElementAt(0); mCurrentEventContent = mCurrentEventContentStack.ObjectAt(0); mCurrentEventContentStack.RemoveObjectAt(0); // Don't use it if it has moved to a different document. if (mCurrentEventContent && mCurrentEventContent->GetComposedDoc() != mDocument) { mCurrentEventContent = nullptr; mCurrentEventFrame = nullptr; } } } bool PresShell::InZombieDocument(nsIContent *aContent) { // If a content node points to a null document, or the document is not // attached to a window, then it is possibly in a zombie document, // about to be replaced by a newly loading document. // Such documents cannot handle DOM events. // It might actually be in a node not attached to any document, // in which case there is not parent presshell to retarget it to. nsIDocument* doc = aContent->GetComposedDoc(); return !doc || !doc->GetWindow(); } already_AddRefed<nsPIDOMWindowOuter> PresShell::GetRootWindow() { nsCOMPtr<nsPIDOMWindowOuter> window = mDocument->GetWindow(); if (window) { nsCOMPtr<nsPIDOMWindowOuter> rootWindow = window->GetPrivateRoot(); NS_ASSERTION(rootWindow, "nsPIDOMWindow::GetPrivateRoot() returns NULL"); return rootWindow.forget(); } // If we don't have DOM window, we're zombie, we should find the root window // with our parent shell. nsCOMPtr<nsIPresShell> parent = GetParentPresShellForEventHandling(); NS_ENSURE_TRUE(parent, nullptr); return parent->GetRootWindow(); } already_AddRefed<nsIPresShell> PresShell::GetParentPresShellForEventHandling() { NS_ENSURE_TRUE(mPresContext, nullptr); // Now, find the parent pres shell and send the event there nsCOMPtr<nsIDocShellTreeItem> treeItem = mPresContext->GetDocShell(); if (!treeItem) { treeItem = mForwardingContainer.get(); } // Might have gone away, or never been around to start with NS_ENSURE_TRUE(treeItem, nullptr); nsCOMPtr<nsIDocShellTreeItem> parentTreeItem; treeItem->GetParent(getter_AddRefs(parentTreeItem)); nsCOMPtr<nsIDocShell> parentDocShell = do_QueryInterface(parentTreeItem); NS_ENSURE_TRUE(parentDocShell && treeItem != parentTreeItem, nullptr); nsCOMPtr<nsIPresShell> parentPresShell = parentDocShell->GetPresShell(); return parentPresShell.forget(); } nsresult PresShell::RetargetEventToParent(WidgetGUIEvent* aEvent, nsEventStatus* aEventStatus) { // Send this events straight up to the parent pres shell. // We do this for keystroke events in zombie documents or if either a frame // or a root content is not present. // That way at least the UI key bindings can work. nsCOMPtr<nsIPresShell> kungFuDeathGrip(this); nsCOMPtr<nsIPresShell> parentPresShell = GetParentPresShellForEventHandling(); NS_ENSURE_TRUE(parentPresShell, NS_ERROR_FAILURE); // Fake the event as though it's from the parent pres shell's root frame. return parentPresShell->HandleEvent(parentPresShell->GetRootFrame(), aEvent, true, aEventStatus); } void PresShell::DisableNonTestMouseEvents(bool aDisable) { sDisableNonTestMouseEvents = aDisable; } already_AddRefed<nsPIDOMWindowOuter> PresShell::GetFocusedDOMWindowInOurWindow() { nsCOMPtr<nsPIDOMWindowOuter> rootWindow = GetRootWindow(); NS_ENSURE_TRUE(rootWindow, nullptr); nsCOMPtr<nsPIDOMWindowOuter> focusedWindow; nsFocusManager::GetFocusedDescendant(rootWindow, true, getter_AddRefs(focusedWindow)); return focusedWindow.forget(); } void PresShell::RecordMouseLocation(WidgetGUIEvent* aEvent) { if (!mPresContext) return; if (!mPresContext->IsRoot()) { PresShell* rootPresShell = GetRootPresShell(); if (rootPresShell) { rootPresShell->RecordMouseLocation(aEvent); } return; } if ((aEvent->mMessage == eMouseMove && aEvent->AsMouseEvent()->mReason == WidgetMouseEvent::eReal) || aEvent->mMessage == eMouseEnterIntoWidget || aEvent->mMessage == eMouseDown || aEvent->mMessage == eMouseUp) { nsIFrame* rootFrame = GetRootFrame(); if (!rootFrame) { nsView* rootView = mViewManager->GetRootView(); mMouseLocation = nsLayoutUtils::TranslateWidgetToView(mPresContext, aEvent->mWidget, aEvent->mRefPoint, rootView); mMouseEventTargetGuid = InputAPZContext::GetTargetLayerGuid(); } else { mMouseLocation = nsLayoutUtils::GetEventCoordinatesRelativeTo(aEvent, rootFrame); mMouseEventTargetGuid = InputAPZContext::GetTargetLayerGuid(); } #ifdef DEBUG_MOUSE_LOCATION if (aEvent->mMessage == eMouseEnterIntoWidget) { printf("[ps=%p]got mouse enter for %p\n", this, aEvent->mWidget); } printf("[ps=%p]setting mouse location to (%d,%d)\n", this, mMouseLocation.x, mMouseLocation.y); #endif if (aEvent->mMessage == eMouseEnterIntoWidget) { SynthesizeMouseMove(false); } } else if (aEvent->mMessage == eMouseExitFromWidget) { // Although we only care about the mouse moving into an area for which this // pres shell doesn't receive mouse move events, we don't check which widget // the mouse exit was for since this seems to vary by platform. Hopefully // this won't matter at all since we'll get the mouse move or enter after // the mouse exit when the mouse moves from one of our widgets into another. mMouseLocation = nsPoint(NS_UNCONSTRAINEDSIZE, NS_UNCONSTRAINEDSIZE); mMouseEventTargetGuid = InputAPZContext::GetTargetLayerGuid(); #ifdef DEBUG_MOUSE_LOCATION printf("[ps=%p]got mouse exit for %p\n", this, aEvent->mWidget); printf("[ps=%p]clearing mouse location\n", this); #endif } } nsIFrame* GetNearestFrameContainingPresShell(nsIPresShell* aPresShell) { nsView* view = aPresShell->GetViewManager()->GetRootView(); while (view && !view->GetFrame()) { view = view->GetParent(); } nsIFrame* frame = nullptr; if (view) { frame = view->GetFrame(); } return frame; } static bool FlushThrottledStyles(nsIDocument *aDocument, void *aData) { nsIPresShell* shell = aDocument->GetShell(); if (shell && shell->IsVisible()) { nsPresContext* presContext = shell->GetPresContext(); if (presContext) { if (presContext->RestyleManager()->IsGecko()) { // XXX stylo: ServoRestyleManager doesn't support animations yet. presContext->RestyleManager()->AsGecko()->UpdateOnlyAnimationStyles(); } } } aDocument->EnumerateSubDocuments(FlushThrottledStyles, nullptr); return true; } static nsresult DispatchPointerFromMouseOrTouch(PresShell* aShell, nsIFrame* aFrame, WidgetGUIEvent* aEvent, bool aDontRetargetEvents, nsEventStatus* aStatus, nsIContent** aTargetContent) { EventMessage pointerMessage = eVoidEvent; if (aEvent->mClass == eMouseEventClass) { WidgetMouseEvent* mouseEvent = aEvent->AsMouseEvent(); // 1. If it is not mouse then it is likely will come as touch event // 2. We don't synthesize pointer events for those events that are not // dispatched to DOM. if (!mouseEvent->convertToPointer || !aEvent->IsAllowedToDispatchDOMEvent()) { return NS_OK; } int16_t button = mouseEvent->button; switch (mouseEvent->mMessage) { case eMouseMove: button = -1; pointerMessage = ePointerMove; break; case eMouseUp: pointerMessage = mouseEvent->buttons ? ePointerMove : ePointerUp; break; case eMouseDown: pointerMessage = mouseEvent->buttons & ~nsContentUtils::GetButtonsFlagForButton(button) ? ePointerMove : ePointerDown; break; default: return NS_OK; } WidgetPointerEvent event(*mouseEvent); event.pointerId = mouseEvent->pointerId; event.inputSource = mouseEvent->inputSource; event.mMessage = pointerMessage; event.button = button; event.buttons = mouseEvent->buttons; event.pressure = event.buttons ? mouseEvent->pressure ? mouseEvent->pressure : 0.5f : 0.0f; event.convertToPointer = mouseEvent->convertToPointer = false; aShell->HandleEvent(aFrame, &event, aDontRetargetEvents, aStatus, aTargetContent); } else if (aEvent->mClass == eTouchEventClass) { WidgetTouchEvent* touchEvent = aEvent->AsTouchEvent(); // loop over all touches and dispatch pointer events on each touch // copy the event switch (touchEvent->mMessage) { case eTouchMove: pointerMessage = ePointerMove; break; case eTouchEnd: pointerMessage = ePointerUp; break; case eTouchStart: pointerMessage = ePointerDown; break; case eTouchCancel: pointerMessage = ePointerCancel; break; default: return NS_OK; } for (uint32_t i = 0; i < touchEvent->mTouches.Length(); ++i) { mozilla::dom::Touch* touch = touchEvent->mTouches[i]; if (!touch || !touch->convertToPointer) { continue; } WidgetPointerEvent event(touchEvent->IsTrusted(), pointerMessage, touchEvent->mWidget); event.mIsPrimary = i == 0; event.pointerId = touch->Identifier(); event.mRefPoint = touch->mRefPoint; event.mModifiers = touchEvent->mModifiers; event.mWidth = touch->RadiusX(); event.mHeight = touch->RadiusY(); event.tiltX = touch->tiltX; event.tiltY = touch->tiltY; event.mTime = touchEvent->mTime; event.mTimeStamp = touchEvent->mTimeStamp; event.mFlags = touchEvent->mFlags; event.button = WidgetMouseEvent::eLeftButton; event.buttons = WidgetMouseEvent::eLeftButtonFlag; event.inputSource = nsIDOMMouseEvent::MOZ_SOURCE_TOUCH; event.convertToPointer = touch->convertToPointer = false; aShell->HandleEvent(aFrame, &event, aDontRetargetEvents, aStatus, aTargetContent); } } return NS_OK; } class ReleasePointerCaptureCaller final { public: ReleasePointerCaptureCaller() : mPointerEvent(nullptr) { } ~ReleasePointerCaptureCaller() { if (mPointerEvent) { nsIPresShell::ReleasePointerCapturingContent(mPointerEvent->pointerId); nsIPresShell::CheckPointerCaptureState(mPointerEvent); } } void SetTarget(const WidgetPointerEvent* aPointerEvent) { MOZ_ASSERT(aPointerEvent); mPointerEvent = aPointerEvent; } private: // This is synchronously used inside PresShell::HandleEvent. const WidgetPointerEvent* mPointerEvent; }; static bool CheckPermissionForBeforeAfterKeyboardEvent(Element* aElement) { // An element which is chrome-privileged should be able to handle before // events and after events. nsIPrincipal* principal = aElement->NodePrincipal(); if (nsContentUtils::IsSystemPrincipal(principal)) { return true; } // An element which has "before-after-keyboard-event" permission should be // able to handle before events and after events. nsCOMPtr<nsIPermissionManager> permMgr = services::GetPermissionManager(); uint32_t permission = nsIPermissionManager::DENY_ACTION; if (permMgr) { permMgr->TestPermissionFromPrincipal(principal, "before-after-keyboard-event", &permission); if (permission == nsIPermissionManager::ALLOW_ACTION) { return true; } // Check "embed-apps" permission for later use. permission = nsIPermissionManager::DENY_ACTION; permMgr->TestPermissionFromPrincipal(principal, "embed-apps", &permission); } // An element can handle before events and after events if the following // conditions are met: // 1) <iframe mozbrowser mozapp> // 2) it has "embed-apps" permission. nsCOMPtr<nsIMozBrowserFrame> browserFrame(do_QueryInterface(aElement)); if ((permission == nsIPermissionManager::ALLOW_ACTION) && browserFrame && browserFrame->GetReallyIsApp()) { return true; } return false; } static void BuildTargetChainForBeforeAfterKeyboardEvent(nsINode* aTarget, nsTArray<nsCOMPtr<Element> >& aChain, bool aTargetIsIframe) { Element* frameElement; // If event target is not an iframe, skip the event target and get its // parent frame. if (aTargetIsIframe) { frameElement = aTarget->AsElement(); } else { nsPIDOMWindowOuter* window = aTarget->OwnerDoc()->GetWindow(); frameElement = window ? window->GetFrameElementInternal() : nullptr; } // Check permission for all ancestors and add them into the target chain. while (frameElement) { if (CheckPermissionForBeforeAfterKeyboardEvent(frameElement)) { aChain.AppendElement(frameElement); } nsPIDOMWindowOuter* window = frameElement->OwnerDoc()->GetWindow(); frameElement = window ? window->GetFrameElementInternal() : nullptr; } } void PresShell::DispatchBeforeKeyboardEventInternal(const nsTArray<nsCOMPtr<Element> >& aChain, const WidgetKeyboardEvent& aEvent, size_t& aChainIndex, bool& aDefaultPrevented) { size_t length = aChain.Length(); if (!CanDispatchEvent(&aEvent) || !length) { return; } EventMessage message = (aEvent.mMessage == eKeyDown) ? eBeforeKeyDown : eBeforeKeyUp; nsCOMPtr<EventTarget> eventTarget; // Dispatch before events from the outermost element. for (int32_t i = length - 1; i >= 0; i--) { eventTarget = do_QueryInterface(aChain[i]->OwnerDoc()->GetWindow()); if (!eventTarget || !CanDispatchEvent(&aEvent)) { return; } aChainIndex = i; InternalBeforeAfterKeyboardEvent beforeEvent(aEvent.IsTrusted(), message, aEvent.mWidget); beforeEvent.AssignBeforeAfterKeyEventData(aEvent, false); EventDispatcher::Dispatch(eventTarget, mPresContext, &beforeEvent); if (beforeEvent.DefaultPrevented()) { aDefaultPrevented = true; return; } } } void PresShell::DispatchAfterKeyboardEventInternal(const nsTArray<nsCOMPtr<Element> >& aChain, const WidgetKeyboardEvent& aEvent, bool aEmbeddedCancelled, size_t aStartOffset) { size_t length = aChain.Length(); if (!CanDispatchEvent(&aEvent) || !length) { return; } EventMessage message = (aEvent.mMessage == eKeyDown) ? eAfterKeyDown : eAfterKeyUp; bool embeddedCancelled = aEmbeddedCancelled; nsCOMPtr<EventTarget> eventTarget; // Dispatch after events from the innermost element. for (uint32_t i = aStartOffset; i < length; i++) { eventTarget = do_QueryInterface(aChain[i]->OwnerDoc()->GetWindow()); if (!eventTarget || !CanDispatchEvent(&aEvent)) { return; } InternalBeforeAfterKeyboardEvent afterEvent(aEvent.IsTrusted(), message, aEvent.mWidget); afterEvent.AssignBeforeAfterKeyEventData(aEvent, false); afterEvent.mEmbeddedCancelled.SetValue(embeddedCancelled); EventDispatcher::Dispatch(eventTarget, mPresContext, &afterEvent); embeddedCancelled = afterEvent.DefaultPrevented(); } } void PresShell::DispatchAfterKeyboardEvent(nsINode* aTarget, const WidgetKeyboardEvent& aEvent, bool aEmbeddedCancelled) { MOZ_ASSERT(aTarget); MOZ_ASSERT(BeforeAfterKeyboardEventEnabled()); if (NS_WARN_IF(aEvent.mMessage != eKeyDown && aEvent.mMessage != eKeyUp)) { return; } // Build up a target chain. Each item in the chain will receive an after event. AutoTArray<nsCOMPtr<Element>, 5> chain; bool targetIsIframe = IsTargetIframe(aTarget); BuildTargetChainForBeforeAfterKeyboardEvent(aTarget, chain, targetIsIframe); DispatchAfterKeyboardEventInternal(chain, aEvent, aEmbeddedCancelled); } bool PresShell::CanDispatchEvent(const WidgetGUIEvent* aEvent) const { bool rv = mPresContext && !mHaveShutDown && nsContentUtils::IsSafeToRunScript(); if (aEvent) { rv &= (aEvent && aEvent->mWidget && !aEvent->mWidget->Destroyed()); } return rv; } void PresShell::HandleKeyboardEvent(nsINode* aTarget, WidgetKeyboardEvent& aEvent, bool aEmbeddedCancelled, nsEventStatus* aStatus, EventDispatchingCallback* aEventCB) { MOZ_ASSERT(aTarget); // return true if the event target is in its child process bool targetIsIframe = IsTargetIframe(aTarget); // Dispatch event directly if the event is a keypress event, a key event on // plugin, or there is no need to fire beforeKey* and afterKey* events. if (aEvent.mMessage == eKeyPress || aEvent.IsKeyEventOnPlugin() || !BeforeAfterKeyboardEventEnabled()) { ForwardKeyToInputMethodAppOrDispatch(targetIsIframe, aTarget, aEvent, aStatus, aEventCB); return; } MOZ_ASSERT(aEvent.mMessage == eKeyDown || aEvent.mMessage == eKeyUp); // Build up a target chain. Each item in the chain will receive a before event. AutoTArray<nsCOMPtr<Element>, 5> chain; BuildTargetChainForBeforeAfterKeyboardEvent(aTarget, chain, targetIsIframe); // Dispatch before events. If each item in the chain consumes the before // event and doesn't prevent the default action, we will go further to // dispatch the actual key event and after events in the reverse order. // Otherwise, only items which has handled the before event will receive an // after event. size_t chainIndex; bool defaultPrevented = false; DispatchBeforeKeyboardEventInternal(chain, aEvent, chainIndex, defaultPrevented); // Before event is default-prevented. Dispatch after events with // embeddedCancelled = false to partial items. if (defaultPrevented) { *aStatus = nsEventStatus_eConsumeNoDefault; DispatchAfterKeyboardEventInternal(chain, aEvent, false, chainIndex); // No need to forward the event to child process. aEvent.StopCrossProcessForwarding(); return; } // Event listeners may kill nsPresContext and nsPresShell. if (!CanDispatchEvent()) { return; } if (ForwardKeyToInputMethodAppOrDispatch(targetIsIframe, aTarget, aEvent, aStatus, aEventCB)) { return; } if (aEvent.DefaultPrevented()) { // When embedder prevents the default action of actual key event, attribute // 'embeddedCancelled' of after event is false, i.e. |!targetIsIframe|. // On the contrary, if the defult action is prevented by embedded iframe, // 'embeddedCancelled' is true which equals to |!targetIsIframe|. DispatchAfterKeyboardEventInternal(chain, aEvent, !targetIsIframe, chainIndex); return; } // Event listeners may kill nsPresContext and nsPresShell. if (targetIsIframe || !CanDispatchEvent()) { return; } // Dispatch after events to all items in the chain. DispatchAfterKeyboardEventInternal(chain, aEvent, aEvent.DefaultPrevented()); } #ifdef MOZ_B2G bool PresShell::ForwardKeyToInputMethodApp(nsINode* aTarget, WidgetKeyboardEvent& aEvent, nsEventStatus* aStatus) { if (!XRE_IsParentProcess() || aEvent.mIsSynthesizedByTIP || aEvent.IsKeyEventOnPlugin()) { return false; } if (!mHardwareKeyHandler) { nsresult rv; mHardwareKeyHandler = do_GetService("@mozilla.org/HardwareKeyHandler;1", &rv); if (!NS_SUCCEEDED(rv) || !mHardwareKeyHandler) { return false; } } if (mHardwareKeyHandler->ForwardKeyToInputMethodApp(aTarget, aEvent.AsKeyboardEvent(), aStatus)) { // No need to dispatch the forwarded keyboard event to it's child process aEvent.mFlags.mNoCrossProcessBoundaryForwarding = true; return true; } return false; } #endif // MOZ_B2G bool PresShell::ForwardKeyToInputMethodAppOrDispatch(bool aIsTargetRemote, nsINode* aTarget, WidgetKeyboardEvent& aEvent, nsEventStatus* aStatus, EventDispatchingCallback* aEventCB) { #ifndef MOZ_B2G // No need to forward to input-method-app if the platform isn't run on B2G. EventDispatcher::Dispatch(aTarget, mPresContext, &aEvent, nullptr, aStatus, aEventCB); return false; #else // In-process case: the event target is in the current process if (!aIsTargetRemote) { if(ForwardKeyToInputMethodApp(aTarget, aEvent, aStatus)) { return true; } // If the keyboard event isn't forwarded to the input-method-app, // then it should be dispatched to its event target directly. EventDispatcher::Dispatch(aTarget, mPresContext, &aEvent, nullptr, aStatus, aEventCB); return false; } // OOP case: the event target is in its child process. // Dispatch the keyboard event to the iframe that embeds the remote // event target first. EventDispatcher::Dispatch(aTarget, mPresContext, &aEvent, nullptr, aStatus, aEventCB); // If the event is defaultPrevented, then there is no need to forward it // to the input-method-app. if (aEvent.mFlags.mDefaultPrevented) { return false; } // Try forwarding to the input-method-app. return ForwardKeyToInputMethodApp(aTarget, aEvent, aStatus); #endif // MOZ_B2G } nsresult PresShell::HandleEvent(nsIFrame* aFrame, WidgetGUIEvent* aEvent, bool aDontRetargetEvents, nsEventStatus* aEventStatus, nsIContent** aTargetContent) { #ifdef MOZ_TASK_TRACER // Make touch events, mouse events and hardware key events to be the source // events of TaskTracer, and originate the rest correlation tasks from here. SourceEventType type = SourceEventType::Unknown; if (aEvent->AsTouchEvent()) { type = SourceEventType::Touch; } else if (aEvent->AsMouseEvent()) { type = SourceEventType::Mouse; } else if (aEvent->AsKeyboardEvent()) { type = SourceEventType::Key; } AutoSourceEvent taskTracerEvent(type); #endif NS_ASSERTION(aFrame, "aFrame should be not null"); if (sPointerEventEnabled) { nsWeakFrame weakFrame(aFrame); nsCOMPtr<nsIContent> targetContent; DispatchPointerFromMouseOrTouch(this, aFrame, aEvent, aDontRetargetEvents, aEventStatus, getter_AddRefs(targetContent)); if (!weakFrame.IsAlive()) { if (targetContent) { aFrame = targetContent->GetPrimaryFrame(); if (!aFrame) { PushCurrentEventInfo(aFrame, targetContent); nsresult rv = HandleEventInternal(aEvent, aEventStatus, true); PopCurrentEventInfo(); return rv; } } else { return NS_OK; } } } if (mIsDestroying || (sDisableNonTestMouseEvents && !aEvent->mFlags.mIsSynthesizedForTests && aEvent->HasMouseEventMessage())) { return NS_OK; } RecordMouseLocation(aEvent); if (AccessibleCaretEnabled(mDocument->GetDocShell())) { // We have to target the focus window because regardless of where the // touch goes, we want to access the copy paste manager. nsCOMPtr<nsPIDOMWindowOuter> window = GetFocusedDOMWindowInOurWindow(); nsCOMPtr<nsIDocument> retargetEventDoc = window ? window->GetExtantDoc() : nullptr; nsCOMPtr<nsIPresShell> presShell = retargetEventDoc ? retargetEventDoc->GetShell() : nullptr; RefPtr<AccessibleCaretEventHub> eventHub = presShell ? presShell->GetAccessibleCaretEventHub() : nullptr; if (eventHub) { *aEventStatus = eventHub->HandleEvent(aEvent); if (*aEventStatus == nsEventStatus_eConsumeNoDefault) { // If the event is consumed, cancel APZC panning by setting // mMultipleActionsPrevented. aEvent->mFlags.mMultipleActionsPrevented = true; return NS_OK; } } } if (sPointerEventEnabled) { UpdateActivePointerState(aEvent); } if (!nsContentUtils::IsSafeToRunScript() && aEvent->IsAllowedToDispatchDOMEvent()) { if (aEvent->mClass == eCompositionEventClass) { IMEStateManager::OnCompositionEventDiscarded( aEvent->AsCompositionEvent()); } #ifdef DEBUG if (aEvent->IsIMERelatedEvent()) { nsPrintfCString warning("%d event is discarded", aEvent->mMessage); NS_WARNING(warning.get()); } #endif nsContentUtils::WarnScriptWasIgnored(GetDocument()); return NS_OK; } nsIContent* capturingContent = ((aEvent->mClass == ePointerEventClass || aEvent->mClass == eWheelEventClass || aEvent->HasMouseEventMessage()) ? GetCapturingContent() : nullptr); nsCOMPtr<nsIDocument> retargetEventDoc; if (!aDontRetargetEvents) { // key and IME related events should not cross top level window boundary. // Basically, such input events should be fired only on focused widget. // However, some IMEs might need to clean up composition after focused // window is deactivated. And also some tests on MozMill want to test key // handling on deactivated window because MozMill window can be activated // during tests. So, there is no merit the events should be redirected to // active window. So, the events should be handled on the last focused // content in the last focused DOM window in same top level window. // Note, if no DOM window has been focused yet, we can discard the events. if (aEvent->IsTargetedAtFocusedWindow()) { nsCOMPtr<nsPIDOMWindowOuter> window = GetFocusedDOMWindowInOurWindow(); // No DOM window in same top level window has not been focused yet, // discard the events. if (!window) { return NS_OK; } retargetEventDoc = window->GetExtantDoc(); if (!retargetEventDoc) return NS_OK; } else if (capturingContent) { // if the mouse is being captured then retarget the mouse event at the // document that is being captured. retargetEventDoc = capturingContent->GetComposedDoc(); #ifdef ANDROID } else if ((aEvent->mClass == eTouchEventClass) || (aEvent->mClass == eMouseEventClass) || (aEvent->mClass == eWheelEventClass)) { retargetEventDoc = GetTouchEventTargetDocument(); #endif } if (retargetEventDoc) { nsCOMPtr<nsIPresShell> presShell = retargetEventDoc->GetShell(); if (!presShell) return NS_OK; if (presShell != this) { nsIFrame* frame = presShell->GetRootFrame(); if (!frame) { if (aEvent->mMessage == eQueryTextContent || aEvent->IsContentCommandEvent()) { return NS_OK; } frame = GetNearestFrameContainingPresShell(presShell); } if (!frame) return NS_OK; nsCOMPtr<nsIPresShell> shell = frame->PresContext()->GetPresShell(); return shell->HandleEvent(frame, aEvent, true, aEventStatus); } } } if (aEvent->mClass == eKeyboardEventClass && mDocument && mDocument->EventHandlingSuppressed()) { if (aEvent->mMessage == eKeyDown) { mNoDelayedKeyEvents = true; } else if (!mNoDelayedKeyEvents) { DelayedEvent* event = new DelayedKeyEvent(aEvent->AsKeyboardEvent()); if (!mDelayedEvents.AppendElement(event)) { delete event; } } aEvent->mFlags.mIsSuppressedOrDelayed = true; return NS_OK; } nsIFrame* frame = aFrame; if (aEvent->IsUsingCoordinates()) { ReleasePointerCaptureCaller releasePointerCaptureCaller; if (mDocument) { if (aEvent->mClass == eTouchEventClass) { nsIDocument::UnlockPointer(); } nsWeakFrame weakFrame(frame); { // scope for scriptBlocker. nsAutoScriptBlocker scriptBlocker; FlushThrottledStyles(GetRootPresShell()->GetDocument(), nullptr); } if (!weakFrame.IsAlive()) { frame = GetNearestFrameContainingPresShell(this); } } if (!frame) { NS_WARNING("Nothing to handle this event!"); return NS_OK; } nsPresContext* framePresContext = frame->PresContext(); nsPresContext* rootPresContext = framePresContext->GetRootPresContext(); NS_ASSERTION(rootPresContext == mPresContext->GetRootPresContext(), "How did we end up outside the connected prescontext/viewmanager hierarchy?"); // If we aren't starting our event dispatch from the root frame of the root prescontext, // then someone must be capturing the mouse. In that case we don't want to search the popup // list. if (framePresContext == rootPresContext && frame == mFrameConstructor->GetRootFrame()) { nsIFrame* popupFrame = nsLayoutUtils::GetPopupFrameForEventCoordinates(rootPresContext, aEvent); // If a remote browser is currently capturing input break out if we // detect a chrome generated popup. if (popupFrame && capturingContent && EventStateManager::IsRemoteTarget(capturingContent)) { capturingContent = nullptr; } // If the popupFrame is an ancestor of the 'frame', the frame should // handle the event, otherwise, the popup should handle it. if (popupFrame && !nsContentUtils::ContentIsCrossDocDescendantOf( framePresContext->GetPresShell()->GetDocument(), popupFrame->GetContent())) { frame = popupFrame; } } bool captureRetarget = false; if (capturingContent) { // If a capture is active, determine if the docshell is visible. If not, // clear the capture and target the mouse event normally instead. This // would occur if the mouse button is held down while a tab change occurs. // If the docshell is visible, look for a scrolling container. bool vis; nsCOMPtr<nsIBaseWindow> baseWin = do_QueryInterface(mPresContext->GetContainerWeak()); if (baseWin && NS_SUCCEEDED(baseWin->GetVisibility(&vis)) && vis) { captureRetarget = gCaptureInfo.mRetargetToElement; if (!captureRetarget) { // A check was already done above to ensure that capturingContent is // in this presshell. NS_ASSERTION(capturingContent->GetComposedDoc() == GetDocument(), "Unexpected document"); nsIFrame* captureFrame = capturingContent->GetPrimaryFrame(); if (captureFrame) { if (capturingContent->IsHTMLElement(nsGkAtoms::select)) { // a dropdown <select> has a child in its selectPopupList and we should // capture on that instead. nsIFrame* childFrame = captureFrame->GetChildList(nsIFrame::kSelectPopupList).FirstChild(); if (childFrame) { captureFrame = childFrame; } } // scrollable frames should use the scrolling container as // the root instead of the document nsIScrollableFrame* scrollFrame = do_QueryFrame(captureFrame); if (scrollFrame) { frame = scrollFrame->GetScrolledFrame(); } } } } else { ClearMouseCapture(nullptr); capturingContent = nullptr; } } // all touch events except for touchstart use a captured target if (aEvent->mClass == eTouchEventClass && aEvent->mMessage != eTouchStart) { captureRetarget = true; } WidgetMouseEvent* mouseEvent = aEvent->AsMouseEvent(); bool isWindowLevelMouseExit = (aEvent->mMessage == eMouseExitFromWidget) && (mouseEvent && mouseEvent->mExitFrom == WidgetMouseEvent::eTopLevel); // Get the frame at the event point. However, don't do this if we're // capturing and retargeting the event because the captured frame will // be used instead below. Also keep using the root frame if we're dealing // with a window-level mouse exit event since we want to start sending // mouse out events at the root EventStateManager. if (!captureRetarget && !isWindowLevelMouseExit) { nsPoint eventPoint; uint32_t flags = 0; if (aEvent->mMessage == eTouchStart) { flags |= INPUT_IGNORE_ROOT_SCROLL_FRAME; WidgetTouchEvent* touchEvent = aEvent->AsTouchEvent(); // if this is a continuing session, ensure that all these events are // in the same document by taking the target of the events already in // the capture list nsCOMPtr<nsIContent> anyTarget; if (touchEvent->mTouches.Length() > 1) { anyTarget = TouchManager::GetAnyCapturedTouchTarget(); } for (int32_t i = touchEvent->mTouches.Length(); i; ) { --i; dom::Touch* touch = touchEvent->mTouches[i]; int32_t id = touch->Identifier(); if (!TouchManager::HasCapturedTouch(id)) { // find the target for this touch eventPoint = nsLayoutUtils::GetEventCoordinatesRelativeTo(aEvent, touch->mRefPoint, frame); nsIFrame* target = FindFrameTargetedByInputEvent(aEvent, frame, eventPoint, flags); if (target && !anyTarget) { target->GetContentForEvent(aEvent, getter_AddRefs(anyTarget)); while (anyTarget && !anyTarget->IsElement()) { anyTarget = anyTarget->GetParent(); } touch->SetTarget(anyTarget); } else { nsIFrame* newTargetFrame = nullptr; for (nsIFrame* f = target; f; f = nsLayoutUtils::GetParentOrPlaceholderForCrossDoc(f)) { if (f->PresContext()->Document() == anyTarget->OwnerDoc()) { newTargetFrame = f; break; } // We must be in a subdocument so jump directly to the root frame. // GetParentOrPlaceholderForCrossDoc gets called immediately to // jump up to the containing document. f = f->PresContext()->GetPresShell()->GetRootFrame(); } // if we couldn't find a target frame in the same document as // anyTarget, remove the touch from the capture touch list, as // well as the event->mTouches array. touchmove events that aren't // in the captured touch list will be discarded if (!newTargetFrame) { touchEvent->mTouches.RemoveElementAt(i); } else { target = newTargetFrame; nsCOMPtr<nsIContent> targetContent; target->GetContentForEvent(aEvent, getter_AddRefs(targetContent)); while (targetContent && !targetContent->IsElement()) { targetContent = targetContent->GetParent(); } touch->SetTarget(targetContent); } } if (target) { frame = target; } } else { // This touch is an old touch, we need to ensure that is not // marked as changed and set its target correctly touch->mChanged = false; int32_t id = touch->Identifier(); RefPtr<dom::Touch> oldTouch = TouchManager::GetCapturedTouch(id); if (oldTouch) { touch->SetTarget(oldTouch->mTarget); } } } } else { eventPoint = nsLayoutUtils::GetEventCoordinatesRelativeTo(aEvent, frame); } if (mouseEvent && mouseEvent->mClass == eMouseEventClass && mouseEvent->mIgnoreRootScrollFrame) { flags |= INPUT_IGNORE_ROOT_SCROLL_FRAME; } nsIFrame* target = FindFrameTargetedByInputEvent(aEvent, frame, eventPoint, flags); if (target) { frame = target; } } // if a node is capturing the mouse, check if the event needs to be // retargeted at the capturing content instead. This will be the case when // capture retargeting is being used, no frame was found or the frame's // content is not a descendant of the capturing content. if (capturingContent && (gCaptureInfo.mRetargetToElement || !frame->GetContent() || !nsContentUtils::ContentIsCrossDocDescendantOf(frame->GetContent(), capturingContent))) { // A check was already done above to ensure that capturingContent is // in this presshell. NS_ASSERTION(capturingContent->GetComposedDoc() == GetDocument(), "Unexpected document"); nsIFrame* capturingFrame = capturingContent->GetPrimaryFrame(); if (capturingFrame) { frame = capturingFrame; } } if (aEvent->mClass == ePointerEventClass) { if (WidgetPointerEvent* pointerEvent = aEvent->AsPointerEvent()) { // Try to keep frame for following check, because // frame can be damaged during CheckPointerCaptureState. nsWeakFrame frameKeeper(frame); // Handle pending pointer capture before any pointer events except // gotpointercapture / lostpointercapture. CheckPointerCaptureState(pointerEvent); // Prevent application crashes, in case damaged frame. if (!frameKeeper.IsAlive()) { frame = nullptr; } // Implicit pointer capture for touch if (frame && sPointerEventImplicitCapture && pointerEvent->mMessage == ePointerDown && pointerEvent->inputSource == nsIDOMMouseEvent::MOZ_SOURCE_TOUCH) { nsCOMPtr<nsIContent> targetContent; frame->GetContentForEvent(aEvent, getter_AddRefs(targetContent)); while (targetContent && !targetContent->IsElement()) { targetContent = targetContent->GetParent(); } if (targetContent) { SetPointerCapturingContent(pointerEvent->pointerId, targetContent); } } } } // Mouse events should be fired to the same target as their mapped pointer // events if ((aEvent->mClass == ePointerEventClass || aEvent->mClass == eMouseEventClass) && aEvent->mMessage != ePointerDown && aEvent->mMessage != eMouseDown) { if (WidgetMouseEvent* mouseEvent = aEvent->AsMouseEvent()) { uint32_t pointerId = mouseEvent->pointerId; nsIContent* pointerCapturingContent = GetPointerCapturingContent(pointerId); if (pointerCapturingContent) { if (nsIFrame* capturingFrame = pointerCapturingContent->GetPrimaryFrame()) { frame = capturingFrame; } if (aEvent->mMessage == ePointerUp || aEvent->mMessage == ePointerCancel) { // Implicitly releasing capture for given pointer. // ePointerLostCapture should be send after ePointerUp or // ePointerCancel. WidgetPointerEvent* pointerEvent = aEvent->AsPointerEvent(); MOZ_ASSERT(pointerEvent); releasePointerCaptureCaller.SetTarget(pointerEvent); } } } } // Suppress mouse event if it's being targeted at an element inside // a document which needs events suppressed if (aEvent->mClass == eMouseEventClass && frame->PresContext()->Document()->EventHandlingSuppressed()) { if (aEvent->mMessage == eMouseDown) { mNoDelayedMouseEvents = true; } else if (!mNoDelayedMouseEvents && (aEvent->mMessage == eMouseUp || // contextmenu is triggered after right mouseup on Windows and right // mousedown on other platforms. aEvent->mMessage == eContextMenu)) { DelayedEvent* event = new DelayedMouseEvent(aEvent->AsMouseEvent()); if (!mDelayedEvents.AppendElement(event)) { delete event; } } return NS_OK; } if (!frame) { NS_WARNING("Nothing to handle this event!"); return NS_OK; } PresShell* shell = static_cast<PresShell*>(frame->PresContext()->PresShell()); switch (aEvent->mMessage) { case eTouchMove: case eTouchCancel: case eTouchEnd: { // get the correct shell to dispatch to WidgetTouchEvent* touchEvent = aEvent->AsTouchEvent(); for (dom::Touch* touch : touchEvent->mTouches) { if (!touch) { break; } RefPtr<dom::Touch> oldTouch = TouchManager::GetCapturedTouch(touch->Identifier()); if (!oldTouch) { break; } nsCOMPtr<nsIContent> content = do_QueryInterface(oldTouch->GetTarget()); if (!content) { break; } nsIFrame* contentFrame = content->GetPrimaryFrame(); if (!contentFrame) { break; } shell = static_cast<PresShell*>( contentFrame->PresContext()->PresShell()); if (shell) { break; } } break; } default: break; } // Check if we have an active EventStateManager which isn't the // EventStateManager of the current PresContext. // If that is the case, and mouse is over some ancestor document, // forward event handling to the active document. // This way content can get mouse events even when // mouse is over the chrome or outside the window. // // Note, currently for backwards compatibility we don't forward mouse events // to the active document when mouse is over some subdocument. if (EventStateManager* activeESM = EventStateManager::GetActiveEventStateManager()) { if (aEvent->mClass == ePointerEventClass || aEvent->HasMouseEventMessage()) { if (activeESM != shell->GetPresContext()->EventStateManager()) { if (nsPresContext* activeContext = activeESM->GetPresContext()) { if (nsIPresShell* activeShell = activeContext->GetPresShell()) { if (nsContentUtils::ContentIsCrossDocDescendantOf(activeShell->GetDocument(), shell->GetDocument())) { shell = static_cast<PresShell*>(activeShell); frame = shell->GetRootFrame(); } } } } } } // Before HandlePositionedEvent we should save mPointerEventTarget in some // cases nsWeakFrame weakFrame; if (sPointerEventEnabled && aTargetContent && ePointerEventClass == aEvent->mClass) { weakFrame = frame; shell->mPointerEventTarget = frame->GetContent(); MOZ_ASSERT(!frame->GetContent() || shell->GetDocument() == frame->GetContent()->OwnerDoc()); } // Prevent deletion until we're done with event handling (bug 336582) and // swap mPointerEventTarget to *aTargetContent nsCOMPtr<nsIPresShell> kungFuDeathGrip(shell); nsresult rv; if (shell != this) { // Handle the event in the correct shell. // We pass the subshell's root frame as the frame to start from. This is // the only correct alternative; if the event was captured then it // must have been captured by us or some ancestor shell and we // now ask the subshell to dispatch it normally. rv = shell->HandlePositionedEvent(frame, aEvent, aEventStatus); } else { rv = HandlePositionedEvent(frame, aEvent, aEventStatus); } // After HandlePositionedEvent we should reestablish // content (which still live in tree) in some cases if (sPointerEventEnabled && aTargetContent && ePointerEventClass == aEvent->mClass) { if (!weakFrame.IsAlive()) { shell->mPointerEventTarget.swap(*aTargetContent); } } return rv; } nsresult rv = NS_OK; if (frame) { PushCurrentEventInfo(nullptr, nullptr); // key and IME related events go to the focused frame in this DOM window. if (aEvent->IsTargetedAtFocusedContent()) { mCurrentEventContent = nullptr; nsCOMPtr<nsPIDOMWindowOuter> window = mDocument->GetWindow(); nsCOMPtr<nsPIDOMWindowOuter> focusedWindow; nsCOMPtr<nsIContent> eventTarget = nsFocusManager::GetFocusedDescendant(window, false, getter_AddRefs(focusedWindow)); // otherwise, if there is no focused content or the focused content has // no frame, just use the root content. This ensures that key events // still get sent to the window properly if nothing is focused or if a // frame goes away while it is focused. if (!eventTarget || !eventTarget->GetPrimaryFrame()) { nsCOMPtr<nsIDOMHTMLDocument> htmlDoc = do_QueryInterface(mDocument); if (htmlDoc) { nsCOMPtr<nsIDOMHTMLElement> body; htmlDoc->GetBody(getter_AddRefs(body)); eventTarget = do_QueryInterface(body); if (!eventTarget) { eventTarget = mDocument->GetRootElement(); } } else { eventTarget = mDocument->GetRootElement(); } } if (aEvent->mMessage == eKeyDown) { NS_IF_RELEASE(gKeyDownTarget); NS_IF_ADDREF(gKeyDownTarget = eventTarget); } else if ((aEvent->mMessage == eKeyPress || aEvent->mMessage == eKeyUp) && gKeyDownTarget) { // If a different element is now focused for the keypress/keyup event // than what was focused during the keydown event, check if the new // focused element is not in a chrome document any more, and if so, // retarget the event back at the keydown target. This prevents a // content area from grabbing the focus from chrome in-between key // events. if (eventTarget) { bool keyDownIsChrome = nsContentUtils::IsChromeDoc(gKeyDownTarget->GetComposedDoc()); if (keyDownIsChrome != nsContentUtils::IsChromeDoc(eventTarget->GetComposedDoc()) || (keyDownIsChrome && TabParent::GetFrom(eventTarget))) { eventTarget = gKeyDownTarget; } } if (aEvent->mMessage == eKeyUp) { NS_RELEASE(gKeyDownTarget); } } mCurrentEventFrame = nullptr; nsIDocument* targetDoc = eventTarget ? eventTarget->OwnerDoc() : nullptr; if (targetDoc && targetDoc != mDocument) { PopCurrentEventInfo(); nsCOMPtr<nsIPresShell> shell = targetDoc->GetShell(); if (shell) { rv = static_cast<PresShell*>(shell.get())-> HandleRetargetedEvent(aEvent, aEventStatus, eventTarget); } return rv; } else { mCurrentEventContent = eventTarget; } if (!GetCurrentEventContent() || !GetCurrentEventFrame() || InZombieDocument(mCurrentEventContent)) { rv = RetargetEventToParent(aEvent, aEventStatus); PopCurrentEventInfo(); return rv; } } else { mCurrentEventFrame = frame; } if (GetCurrentEventFrame()) { rv = HandleEventInternal(aEvent, aEventStatus, true); } #ifdef DEBUG ShowEventTargetDebug(); #endif PopCurrentEventInfo(); } else { // Activation events need to be dispatched even if no frame was found, since // we don't want the focus to be out of sync. if (!NS_EVENT_NEEDS_FRAME(aEvent)) { mCurrentEventFrame = nullptr; return HandleEventInternal(aEvent, aEventStatus, true); } else if (aEvent->HasKeyEventMessage()) { // Keypress events in new blank tabs should not be completely thrown away. // Retarget them -- the parent chrome shell might make use of them. return RetargetEventToParent(aEvent, aEventStatus); } } return rv; } #ifdef ANDROID nsIDocument* PresShell::GetTouchEventTargetDocument() { nsPresContext* context = GetPresContext(); if (!context || !context->IsRoot()) { return nullptr; } nsCOMPtr<nsIDocShellTreeItem> shellAsTreeItem = context->GetDocShell(); if (!shellAsTreeItem) { return nullptr; } nsCOMPtr<nsIDocShellTreeOwner> owner; shellAsTreeItem->GetTreeOwner(getter_AddRefs(owner)); if (!owner) { return nullptr; } // now get the primary content shell (active tab) nsCOMPtr<nsIDocShellTreeItem> item; owner->GetPrimaryContentShell(getter_AddRefs(item)); nsCOMPtr<nsIDocShell> childDocShell = do_QueryInterface(item); if (!childDocShell) { return nullptr; } return childDocShell->GetDocument(); } #endif #ifdef DEBUG void PresShell::ShowEventTargetDebug() { if (nsFrame::GetShowEventTargetFrameBorder() && GetCurrentEventFrame()) { if (mDrawEventTargetFrame) { mDrawEventTargetFrame->InvalidateFrame(); } mDrawEventTargetFrame = mCurrentEventFrame; mDrawEventTargetFrame->InvalidateFrame(); } } #endif nsresult PresShell::HandlePositionedEvent(nsIFrame* aTargetFrame, WidgetGUIEvent* aEvent, nsEventStatus* aEventStatus) { nsresult rv = NS_OK; PushCurrentEventInfo(nullptr, nullptr); mCurrentEventFrame = aTargetFrame; if (mCurrentEventFrame) { nsCOMPtr<nsIContent> targetElement; mCurrentEventFrame->GetContentForEvent(aEvent, getter_AddRefs(targetElement)); // If there is no content for this frame, target it anyway. Some // frames can be targeted but do not have content, particularly // windows with scrolling off. if (targetElement) { // Bug 103055, bug 185889: mouse events apply to *elements*, not all // nodes. Thus we get the nearest element parent here. // XXX we leave the frame the same even if we find an element // parent, so that the text frame will receive the event (selection // and friends are the ones who care about that anyway) // // We use weak pointers because during this tight loop, the node // will *not* go away. And this happens on every mousemove. while (targetElement && !targetElement->IsElement()) { targetElement = targetElement->GetFlattenedTreeParent(); } // If we found an element, target it. Otherwise, target *nothing*. if (!targetElement) { mCurrentEventContent = nullptr; mCurrentEventFrame = nullptr; } else if (targetElement != mCurrentEventContent) { mCurrentEventContent = targetElement; } } } if (GetCurrentEventFrame()) { rv = HandleEventInternal(aEvent, aEventStatus, true); } #ifdef DEBUG ShowEventTargetDebug(); #endif PopCurrentEventInfo(); return rv; } nsresult PresShell::HandleEventWithTarget(WidgetEvent* aEvent, nsIFrame* aFrame, nsIContent* aContent, nsEventStatus* aStatus) { #if DEBUG MOZ_ASSERT(!aFrame || aFrame->PresContext()->GetPresShell() == this, "wrong shell"); if (aContent) { nsIDocument* doc = aContent->GetComposedDoc(); NS_ASSERTION(doc, "event for content that isn't in a document"); NS_ASSERTION(!doc || doc->GetShell() == this, "wrong shell"); } #endif NS_ENSURE_STATE(!aContent || aContent->GetComposedDoc() == mDocument); PushCurrentEventInfo(aFrame, aContent); nsresult rv = HandleEventInternal(aEvent, aStatus, false); PopCurrentEventInfo(); return rv; } nsresult PresShell::HandleEventInternal(WidgetEvent* aEvent, nsEventStatus* aStatus, bool aIsHandlingNativeEvent) { RefPtr<EventStateManager> manager = mPresContext->EventStateManager(); nsresult rv = NS_OK; if (!NS_EVENT_NEEDS_FRAME(aEvent) || GetCurrentEventFrame() || GetCurrentEventContent()) { bool touchIsNew = false; bool isHandlingUserInput = false; // XXX How about IME events and input events for plugins? if (aEvent->IsTrusted()) { switch (aEvent->mMessage) { case eKeyPress: case eKeyDown: case eKeyUp: { nsIDocument* doc = GetCurrentEventContent() ? mCurrentEventContent->OwnerDoc() : nullptr; auto keyCode = aEvent->AsKeyboardEvent()->mKeyCode; if (keyCode == NS_VK_ESCAPE) { nsIDocument* root = nsContentUtils::GetRootDocument(doc); if (root && root->GetFullscreenElement()) { // Prevent default action on ESC key press when exiting // DOM fullscreen mode. This prevents the browser ESC key // handler from stopping all loads in the document, which // would cause <video> loads to stop. // XXX We need to claim the Escape key event which will be // dispatched only into chrome is already consumed by // content because we need to prevent its default here // for some reasons (not sure) but we need to detect // if a chrome event handler will call PreventDefault() // again and check it later. aEvent->PreventDefaultBeforeDispatch(); aEvent->mFlags.mOnlyChromeDispatch = true; // The event listeners in chrome can prevent this ESC behavior by // calling prevent default on the preceding keydown/press events. if (!mIsLastChromeOnlyEscapeKeyConsumed && aEvent->mMessage == eKeyUp) { // ESC key released while in DOM fullscreen mode. // Fully exit all browser windows and documents from // fullscreen mode. nsIDocument::AsyncExitFullscreen(nullptr); } } nsCOMPtr<nsIDocument> pointerLockedDoc = do_QueryReferent(EventStateManager::sPointerLockedDoc); if (!mIsLastChromeOnlyEscapeKeyConsumed && pointerLockedDoc) { // XXX See above comment to understand the reason why this needs // to claim that the Escape key event is consumed by content // even though it will be dispatched only into chrome. aEvent->PreventDefaultBeforeDispatch(); aEvent->mFlags.mOnlyChromeDispatch = true; if (aEvent->mMessage == eKeyUp) { nsIDocument::UnlockPointer(); } } } if (keyCode != NS_VK_ESCAPE && keyCode != NS_VK_SHIFT && keyCode != NS_VK_CONTROL && keyCode != NS_VK_ALT && keyCode != NS_VK_WIN && keyCode != NS_VK_META) { // Allow keys other than ESC and modifiers be marked as a // valid user input for triggering popup, fullscreen, and // pointer lock. isHandlingUserInput = true; } break; } case eMouseDown: case eMouseUp: case ePointerDown: case ePointerUp: isHandlingUserInput = true; break; case eDrop: { nsCOMPtr<nsIDragSession> session = nsContentUtils::GetDragSession(); if (session) { bool onlyChromeDrop = false; session->GetOnlyChromeDrop(&onlyChromeDrop); if (onlyChromeDrop) { aEvent->mFlags.mOnlyChromeDispatch = true; } } break; } default: break; } if (!mTouchManager.PreHandleEvent(aEvent, aStatus, touchIsNew, isHandlingUserInput, mCurrentEventContent)) { return NS_OK; } } if (aEvent->mMessage == eContextMenu) { WidgetMouseEvent* mouseEvent = aEvent->AsMouseEvent(); if (mouseEvent->IsContextMenuKeyEvent() && !AdjustContextMenuKeyEvent(mouseEvent)) { return NS_OK; } if (mouseEvent->IsShift()) { aEvent->mFlags.mOnlyChromeDispatch = true; aEvent->mFlags.mRetargetToNonNativeAnonymous = true; } } AutoHandlingUserInputStatePusher userInpStatePusher(isHandlingUserInput, aEvent, mDocument); if (aEvent->IsTrusted() && aEvent->mMessage == eMouseMove) { nsIPresShell::AllowMouseCapture( EventStateManager::GetActiveEventStateManager() == manager); } nsAutoPopupStatePusher popupStatePusher( Event::GetEventPopupControlState(aEvent)); // FIXME. If the event was reused, we need to clear the old target, // bug 329430 aEvent->mTarget = nullptr; // 1. Give event to event manager for pre event state changes and // generation of synthetic events. rv = manager->PreHandleEvent(mPresContext, aEvent, mCurrentEventFrame, mCurrentEventContent, aStatus); // 2. Give event to the DOM for third party and JS use. if (NS_SUCCEEDED(rv)) { bool wasHandlingKeyBoardEvent = nsContentUtils::IsHandlingKeyBoardEvent(); if (aEvent->mClass == eKeyboardEventClass) { nsContentUtils::SetIsHandlingKeyBoardEvent(true); } if (aEvent->IsAllowedToDispatchDOMEvent()) { MOZ_ASSERT(nsContentUtils::IsSafeToRunScript(), "Somebody changed aEvent to cause a DOM event!"); nsPresShellEventCB eventCB(this); if (aEvent->mClass == eTouchEventClass) { DispatchTouchEventToDOM(aEvent, aStatus, &eventCB, touchIsNew); } else { DispatchEventToDOM(aEvent, aStatus, &eventCB); } } nsContentUtils::SetIsHandlingKeyBoardEvent(wasHandlingKeyBoardEvent); // 3. Give event to event manager for post event state changes and // generation of synthetic events. if (!mIsDestroying && NS_SUCCEEDED(rv)) { rv = manager->PostHandleEvent(mPresContext, aEvent, GetCurrentEventFrame(), aStatus); } } if (!mIsDestroying && aIsHandlingNativeEvent) { // Ensure that notifications to IME should be sent before getting next // native event from the event queue. // XXX Should we check the event message or event class instead of // using aIsHandlingNativeEvent? manager->TryToFlushPendingNotificationsToIME(); } switch (aEvent->mMessage) { case eKeyPress: case eKeyDown: case eKeyUp: { if (aEvent->AsKeyboardEvent()->mKeyCode == NS_VK_ESCAPE) { if (aEvent->mMessage == eKeyUp) { // Reset this flag after key up is handled. mIsLastChromeOnlyEscapeKeyConsumed = false; } else { if (aEvent->mFlags.mOnlyChromeDispatch && aEvent->mFlags.mDefaultPreventedByChrome) { mIsLastChromeOnlyEscapeKeyConsumed = true; } } } break; } case eMouseUp: // reset the capturing content now that the mouse button is up SetCapturingContent(nullptr, 0); break; case eMouseMove: nsIPresShell::AllowMouseCapture(false); break; default: break; } } if (Telemetry::CanRecordBase() && !aEvent->mTimeStamp.IsNull() && aEvent->AsInputEvent()) { double millis = (TimeStamp::Now() - aEvent->mTimeStamp).ToMilliseconds(); Telemetry::Accumulate(Telemetry::INPUT_EVENT_RESPONSE_MS, millis); if (mDocument && mDocument->GetReadyStateEnum() != nsIDocument::READYSTATE_COMPLETE) { Telemetry::Accumulate(Telemetry::LOAD_INPUT_EVENT_RESPONSE_MS, millis); } } return rv; } /* static */ void nsIPresShell::DispatchGotOrLostPointerCaptureEvent( bool aIsGotCapture, const WidgetPointerEvent* aPointerEvent, nsIContent* aCaptureTarget) { nsIDocument* targetDoc = aCaptureTarget->OwnerDoc(); nsCOMPtr<nsIPresShell> shell = targetDoc->GetShell(); NS_ENSURE_TRUE_VOID(shell); if (!aIsGotCapture && !aCaptureTarget->IsInUncomposedDoc()) { // If the capturing element was removed from the DOM tree, fire // ePointerLostCapture at the document. PointerEventInit init; init.mPointerId = aPointerEvent->pointerId; init.mBubbles = true; init.mComposed = true; ConvertPointerTypeToString(aPointerEvent->inputSource, init.mPointerType); init.mIsPrimary = aPointerEvent->mIsPrimary; RefPtr<mozilla::dom::PointerEvent> event; event = PointerEvent::Constructor(aCaptureTarget, NS_LITERAL_STRING("lostpointercapture"), init); bool dummy; targetDoc->DispatchEvent(event->InternalDOMEvent(), &dummy); return; } nsEventStatus status = nsEventStatus_eIgnore; WidgetPointerEvent localEvent(aPointerEvent->IsTrusted(), aIsGotCapture ? ePointerGotCapture : ePointerLostCapture, aPointerEvent->mWidget); localEvent.AssignPointerEventData(*aPointerEvent, true); nsresult rv = shell->HandleEventWithTarget( &localEvent, aCaptureTarget->GetPrimaryFrame(), aCaptureTarget, &status); NS_ENSURE_SUCCESS_VOID(rv); } nsresult PresShell::DispatchEventToDOM(WidgetEvent* aEvent, nsEventStatus* aStatus, nsPresShellEventCB* aEventCB) { nsresult rv = NS_OK; nsCOMPtr<nsINode> eventTarget = mCurrentEventContent.get(); nsPresShellEventCB* eventCBPtr = aEventCB; if (!eventTarget) { nsCOMPtr<nsIContent> targetContent; if (mCurrentEventFrame) { rv = mCurrentEventFrame-> GetContentForEvent(aEvent, getter_AddRefs(targetContent)); } if (NS_SUCCEEDED(rv) && targetContent) { eventTarget = do_QueryInterface(targetContent); } else if (mDocument) { eventTarget = do_QueryInterface(mDocument); // If we don't have any content, the callback wouldn't probably // do nothing. eventCBPtr = nullptr; } } if (eventTarget) { if (aEvent->mClass == eCompositionEventClass) { IMEStateManager::DispatchCompositionEvent(eventTarget, mPresContext, aEvent->AsCompositionEvent(), aStatus, eventCBPtr); } else if (aEvent->mClass == eKeyboardEventClass) { HandleKeyboardEvent(eventTarget, *(aEvent->AsKeyboardEvent()), false, aStatus, eventCBPtr); } else { EventDispatcher::Dispatch(eventTarget, mPresContext, aEvent, nullptr, aStatus, eventCBPtr); } } return rv; } void PresShell::DispatchTouchEventToDOM(WidgetEvent* aEvent, nsEventStatus* aStatus, nsPresShellEventCB* aEventCB, bool aTouchIsNew) { // calling preventDefault on touchstart or the first touchmove for a // point prevents mouse events. calling it on the touchend should // prevent click dispatching. bool canPrevent = (aEvent->mMessage == eTouchStart) || (aEvent->mMessage == eTouchMove && aTouchIsNew) || (aEvent->mMessage == eTouchEnd); bool preventDefault = false; nsEventStatus tmpStatus = nsEventStatus_eIgnore; WidgetTouchEvent* touchEvent = aEvent->AsTouchEvent(); // loop over all touches and dispatch events on any that have changed for (dom::Touch* touch : touchEvent->mTouches) { if (!touch || !touch->mChanged) { continue; } nsCOMPtr<EventTarget> targetPtr = touch->mTarget; nsCOMPtr<nsIContent> content = do_QueryInterface(targetPtr); if (!content) { continue; } nsIDocument* doc = content->OwnerDoc(); nsIContent* capturingContent = GetCapturingContent(); if (capturingContent) { if (capturingContent->OwnerDoc() != doc) { // Wrong document, don't dispatch anything. continue; } content = capturingContent; } // copy the event WidgetTouchEvent newEvent(touchEvent->IsTrusted(), touchEvent->mMessage, touchEvent->mWidget); newEvent.AssignTouchEventData(*touchEvent, false); newEvent.mTarget = targetPtr; RefPtr<PresShell> contentPresShell; if (doc == mDocument) { contentPresShell = static_cast<PresShell*>(doc->GetShell()); if (contentPresShell) { //XXXsmaug huge hack. Pushing possibly capturing content, // even though event target is something else. contentPresShell->PushCurrentEventInfo( content->GetPrimaryFrame(), content); } } nsIPresShell *presShell = doc->GetShell(); if (!presShell) { continue; } nsPresContext *context = presShell->GetPresContext(); tmpStatus = nsEventStatus_eIgnore; EventDispatcher::Dispatch(targetPtr, context, &newEvent, nullptr, &tmpStatus, aEventCB); if (nsEventStatus_eConsumeNoDefault == tmpStatus || newEvent.mFlags.mMultipleActionsPrevented) { preventDefault = true; } if (newEvent.mFlags.mMultipleActionsPrevented) { touchEvent->mFlags.mMultipleActionsPrevented = true; } if (contentPresShell) { contentPresShell->PopCurrentEventInfo(); } } if (preventDefault && canPrevent) { *aStatus = nsEventStatus_eConsumeNoDefault; } else { *aStatus = nsEventStatus_eIgnore; } } // Dispatch event to content only (NOT full processing) // See also HandleEventWithTarget which does full event processing. nsresult PresShell::HandleDOMEventWithTarget(nsIContent* aTargetContent, WidgetEvent* aEvent, nsEventStatus* aStatus) { nsresult rv = NS_OK; PushCurrentEventInfo(nullptr, aTargetContent); // Bug 41013: Check if the event should be dispatched to content. // It's possible that we are in the middle of destroying the window // and the js context is out of date. This check detects the case // that caused a crash in bug 41013, but there may be a better way // to handle this situation! nsCOMPtr<nsISupports> container = mPresContext->GetContainerWeak(); if (container) { // Dispatch event to content rv = EventDispatcher::Dispatch(aTargetContent, mPresContext, aEvent, nullptr, aStatus); } PopCurrentEventInfo(); return rv; } // See the method above. nsresult PresShell::HandleDOMEventWithTarget(nsIContent* aTargetContent, nsIDOMEvent* aEvent, nsEventStatus* aStatus) { nsresult rv = NS_OK; PushCurrentEventInfo(nullptr, aTargetContent); nsCOMPtr<nsISupports> container = mPresContext->GetContainerWeak(); if (container) { rv = EventDispatcher::DispatchDOMEvent(aTargetContent, nullptr, aEvent, mPresContext, aStatus); } PopCurrentEventInfo(); return rv; } bool PresShell::AdjustContextMenuKeyEvent(WidgetMouseEvent* aEvent) { #ifdef MOZ_XUL // if a menu is open, open the context menu relative to the active item on the menu. nsXULPopupManager* pm = nsXULPopupManager::GetInstance(); if (pm) { nsIFrame* popupFrame = pm->GetTopPopup(ePopupTypeMenu); if (popupFrame) { nsIFrame* itemFrame = (static_cast<nsMenuPopupFrame *>(popupFrame))->GetCurrentMenuItem(); if (!itemFrame) itemFrame = popupFrame; nsCOMPtr<nsIWidget> widget = popupFrame->GetNearestWidget(); aEvent->mWidget = widget; LayoutDeviceIntPoint widgetPoint = widget->WidgetToScreenOffset(); aEvent->mRefPoint = LayoutDeviceIntPoint::FromUnknownPoint( itemFrame->GetScreenRect().BottomLeft()) - widgetPoint; mCurrentEventContent = itemFrame->GetContent(); mCurrentEventFrame = itemFrame; return true; } } #endif // If we're here because of the key-equiv for showing context menus, we // have to twiddle with the NS event to make sure the context menu comes // up in the upper left of the relevant content area before we create // the DOM event. Since we never call InitMouseEvent() on the event, // the client X/Y will be 0,0. We can make use of that if the widget is null. // Use the root view manager's widget since it's most likely to have one, // and the coordinates returned by GetCurrentItemAndPositionForElement // are relative to the widget of the root of the root view manager. nsRootPresContext* rootPC = mPresContext->GetRootPresContext(); aEvent->mRefPoint = LayoutDeviceIntPoint(0, 0); if (rootPC) { rootPC->PresShell()->GetViewManager()-> GetRootWidget(getter_AddRefs(aEvent->mWidget)); if (aEvent->mWidget) { // default the refpoint to the topleft of our document nsPoint offset(0, 0); nsIFrame* rootFrame = mFrameConstructor->GetRootFrame(); if (rootFrame) { nsView* view = rootFrame->GetClosestView(&offset); offset += view->GetOffsetToWidget(aEvent->mWidget); aEvent->mRefPoint = LayoutDeviceIntPoint::FromAppUnitsToNearest(offset, mPresContext->AppUnitsPerDevPixel()); } } } else { aEvent->mWidget = nullptr; } // see if we should use the caret position for the popup LayoutDeviceIntPoint caretPoint; // Beware! This may flush notifications via synchronous // ScrollSelectionIntoView. if (PrepareToUseCaretPosition(aEvent->mWidget, caretPoint)) { // caret position is good aEvent->mRefPoint = caretPoint; return true; } // If we're here because of the key-equiv for showing context menus, we // have to reset the event target to the currently focused element. Get it // from the focus controller. nsCOMPtr<nsIDOMElement> currentFocus; nsIFocusManager* fm = nsFocusManager::GetFocusManager(); if (fm) fm->GetFocusedElement(getter_AddRefs(currentFocus)); // Reset event coordinates relative to focused frame in view if (currentFocus) { nsCOMPtr<nsIContent> currentPointElement; GetCurrentItemAndPositionForElement(currentFocus, getter_AddRefs(currentPointElement), aEvent->mRefPoint, aEvent->mWidget); if (currentPointElement) { mCurrentEventContent = currentPointElement; mCurrentEventFrame = nullptr; GetCurrentEventFrame(); } } return true; } // PresShell::PrepareToUseCaretPosition // // This checks to see if we should use the caret position for popup context // menus. Returns true if the caret position should be used, and the // coordinates of that position is returned in aTargetPt. This function // will also scroll the window as needed to make the caret visible. // // The event widget should be the widget that generated the event, and // whose coordinate system the resulting event's mRefPoint should be // relative to. The returned point is in device pixels realtive to the // widget passed in. bool PresShell::PrepareToUseCaretPosition(nsIWidget* aEventWidget, LayoutDeviceIntPoint& aTargetPt) { nsresult rv; // check caret visibility RefPtr<nsCaret> caret = GetCaret(); NS_ENSURE_TRUE(caret, false); bool caretVisible = caret->IsVisible(); if (!caretVisible) return false; // caret selection, this is a temporary weak reference, so no refcounting is // needed nsISelection* domSelection = caret->GetSelection(); NS_ENSURE_TRUE(domSelection, false); // since the match could be an anonymous textnode inside a // <textarea> or text <input>, we need to get the outer frame // note: frames are not refcounted nsIFrame* frame = nullptr; // may be nullptr nsCOMPtr<nsIDOMNode> node; rv = domSelection->GetFocusNode(getter_AddRefs(node)); NS_ENSURE_SUCCESS(rv, false); NS_ENSURE_TRUE(node, false); nsCOMPtr<nsIContent> content(do_QueryInterface(node)); if (content) { nsIContent* nonNative = content->FindFirstNonChromeOnlyAccessContent(); content = nonNative; } if (content) { // It seems like ScrollSelectionIntoView should be enough, but it's // not. The problem is that scrolling the selection into view when it is // below the current viewport will align the top line of the frame exactly // with the bottom of the window. This is fine, BUT, the popup event causes // the control to be re-focused which does this exact call to // ScrollContentIntoView, which has a one-pixel disagreement of whether the // frame is actually in view. The result is that the frame is aligned with // the top of the window, but the menu is still at the bottom. // // Doing this call first forces the frame to be in view, eliminating the // problem. The only difference in the result is that if your cursor is in // an edit box below the current view, you'll get the edit box aligned with // the top of the window. This is arguably better behavior anyway. rv = ScrollContentIntoView(content, nsIPresShell::ScrollAxis( nsIPresShell::SCROLL_MINIMUM, nsIPresShell::SCROLL_IF_NOT_VISIBLE), nsIPresShell::ScrollAxis( nsIPresShell::SCROLL_MINIMUM, nsIPresShell::SCROLL_IF_NOT_VISIBLE), nsIPresShell::SCROLL_OVERFLOW_HIDDEN); NS_ENSURE_SUCCESS(rv, false); frame = content->GetPrimaryFrame(); NS_WARNING_ASSERTION(frame, "No frame for focused content?"); } // Actually scroll the selection (ie caret) into view. Note that this must // be synchronous since we will be checking the caret position on the screen. // // Be easy about errors, and just don't scroll in those cases. Better to have // the correct menu at a weird place than the wrong menu. // After ScrollSelectionIntoView(), the pending notifications might be // flushed and PresShell/PresContext/Frames may be dead. See bug 418470. nsCOMPtr<nsISelectionController> selCon; if (frame) frame->GetSelectionController(GetPresContext(), getter_AddRefs(selCon)); else selCon = static_cast<nsISelectionController *>(this); if (selCon) { rv = selCon->ScrollSelectionIntoView(nsISelectionController::SELECTION_NORMAL, nsISelectionController::SELECTION_FOCUS_REGION, nsISelectionController::SCROLL_SYNCHRONOUS); NS_ENSURE_SUCCESS(rv, false); } nsPresContext* presContext = GetPresContext(); // get caret position relative to the closest view nsRect caretCoords; nsIFrame* caretFrame = caret->GetGeometry(&caretCoords); if (!caretFrame) return false; nsPoint viewOffset; nsView* view = caretFrame->GetClosestView(&viewOffset); if (!view) return false; // and then get the caret coords relative to the event widget if (aEventWidget) { viewOffset += view->GetOffsetToWidget(aEventWidget); } caretCoords.MoveBy(viewOffset); // caret coordinates are in app units, convert to pixels aTargetPt.x = presContext->AppUnitsToDevPixels(caretCoords.x + caretCoords.width); aTargetPt.y = presContext->AppUnitsToDevPixels(caretCoords.y + caretCoords.height); // make sure rounding doesn't return a pixel which is outside the caret // (e.g. one line lower) aTargetPt.y -= 1; return true; } void PresShell::GetCurrentItemAndPositionForElement(nsIDOMElement *aCurrentEl, nsIContent** aTargetToUse, LayoutDeviceIntPoint& aTargetPt, nsIWidget *aRootWidget) { nsCOMPtr<nsIContent> focusedContent(do_QueryInterface(aCurrentEl)); ScrollContentIntoView(focusedContent, ScrollAxis(), ScrollAxis(), nsIPresShell::SCROLL_OVERFLOW_HIDDEN); nsPresContext* presContext = GetPresContext(); bool istree = false, checkLineHeight = true; nscoord extraTreeY = 0; #ifdef MOZ_XUL // Set the position to just underneath the current item for multi-select // lists or just underneath the selected item for single-select lists. If // the element is not a list, or there is no selection, leave the position // as is. nsCOMPtr<nsIDOMXULSelectControlItemElement> item; nsCOMPtr<nsIDOMXULMultiSelectControlElement> multiSelect = do_QueryInterface(aCurrentEl); if (multiSelect) { checkLineHeight = false; int32_t currentIndex; multiSelect->GetCurrentIndex(¤tIndex); if (currentIndex >= 0) { nsCOMPtr<nsIDOMXULElement> xulElement(do_QueryInterface(aCurrentEl)); if (xulElement) { nsCOMPtr<nsIBoxObject> box; xulElement->GetBoxObject(getter_AddRefs(box)); nsCOMPtr<nsITreeBoxObject> treeBox(do_QueryInterface(box)); // Tree view special case (tree items have no frames) // Get the focused row and add its coordinates, which are already in pixels // XXX Boris, should we create a new interface so that this doesn't // need to know about trees? Something like nsINodelessChildCreator which // could provide the current focus coordinates? if (treeBox) { treeBox->EnsureRowIsVisible(currentIndex); int32_t firstVisibleRow, rowHeight; treeBox->GetFirstVisibleRow(&firstVisibleRow); treeBox->GetRowHeight(&rowHeight); extraTreeY += presContext->CSSPixelsToAppUnits( (currentIndex - firstVisibleRow + 1) * rowHeight); istree = true; nsCOMPtr<nsITreeColumns> cols; treeBox->GetColumns(getter_AddRefs(cols)); if (cols) { nsCOMPtr<nsITreeColumn> col; cols->GetFirstColumn(getter_AddRefs(col)); if (col) { nsCOMPtr<nsIDOMElement> colElement; col->GetElement(getter_AddRefs(colElement)); nsCOMPtr<nsIContent> colContent(do_QueryInterface(colElement)); if (colContent) { nsIFrame* frame = colContent->GetPrimaryFrame(); if (frame) { extraTreeY += frame->GetSize().height; } } } } } else { multiSelect->GetCurrentItem(getter_AddRefs(item)); } } } } else { // don't check menulists as the selected item will be inside a popup. nsCOMPtr<nsIDOMXULMenuListElement> menulist = do_QueryInterface(aCurrentEl); if (!menulist) { nsCOMPtr<nsIDOMXULSelectControlElement> select = do_QueryInterface(aCurrentEl); if (select) { checkLineHeight = false; select->GetSelectedItem(getter_AddRefs(item)); } } } if (item) focusedContent = do_QueryInterface(item); #endif nsIFrame *frame = focusedContent->GetPrimaryFrame(); if (frame) { NS_ASSERTION(frame->PresContext() == GetPresContext(), "handling event for focused content that is not in our document?"); nsPoint frameOrigin(0, 0); // Get the frame's origin within its view nsView *view = frame->GetClosestView(&frameOrigin); NS_ASSERTION(view, "No view for frame"); // View's origin relative the widget if (aRootWidget) { frameOrigin += view->GetOffsetToWidget(aRootWidget); } // Start context menu down and to the right from top left of frame // use the lineheight. This is a good distance to move the context // menu away from the top left corner of the frame. If we always // used the frame height, the context menu could end up far away, // for example when we're focused on linked images. // On the other hand, we want to use the frame height if it's less // than the current line height, so that the context menu appears // associated with the correct frame. nscoord extra = 0; if (!istree) { extra = frame->GetSize().height; if (checkLineHeight) { nsIScrollableFrame *scrollFrame = nsLayoutUtils::GetNearestScrollableFrame(frame); if (scrollFrame) { nsSize scrollAmount = scrollFrame->GetLineScrollAmount(); nsIFrame* f = do_QueryFrame(scrollFrame); int32_t APD = presContext->AppUnitsPerDevPixel(); int32_t scrollAPD = f->PresContext()->AppUnitsPerDevPixel(); scrollAmount = scrollAmount.ScaleToOtherAppUnits(scrollAPD, APD); if (extra > scrollAmount.height) { extra = scrollAmount.height; } } } } aTargetPt.x = presContext->AppUnitsToDevPixels(frameOrigin.x); aTargetPt.y = presContext->AppUnitsToDevPixels( frameOrigin.y + extra + extraTreeY); } NS_IF_ADDREF(*aTargetToUse = focusedContent); } bool PresShell::ShouldIgnoreInvalidation() { return mPaintingSuppressed || !mIsActive || mIsNeverPainting; } void PresShell::WillPaint() { // Check the simplest things first. In particular, it's important to // check mIsActive before making any of the more expensive calls such // as GetRootPresContext, for the case of a browser with a large // number of tabs. // Don't bother doing anything if some viewmanager in our tree is painting // while we still have painting suppressed or we are not active. if (!mIsActive || mPaintingSuppressed || !IsVisible()) { return; } nsRootPresContext* rootPresContext = mPresContext->GetRootPresContext(); if (!rootPresContext) { // In some edge cases, such as when we don't have a root frame yet, // we can't find the root prescontext. There's nothing to do in that // case. return; } rootPresContext->FlushWillPaintObservers(); if (mIsDestroying) return; // Process reflows, if we have them, to reduce flicker due to invalidates and // reflow being interspersed. Note that we _do_ allow this to be // interruptible; if we can't do all the reflows it's better to flicker a bit // than to freeze up. FlushPendingNotifications(ChangesToFlush(Flush_InterruptibleLayout, false)); } void PresShell::WillPaintWindow() { nsRootPresContext* rootPresContext = mPresContext->GetRootPresContext(); if (rootPresContext != mPresContext) { // This could be a popup's presshell. We don't allow plugins in popups // so there's nothing to do here. return; } #ifndef XP_MACOSX rootPresContext->ApplyPluginGeometryUpdates(); #endif } void PresShell::DidPaintWindow() { nsRootPresContext* rootPresContext = mPresContext->GetRootPresContext(); if (rootPresContext != mPresContext) { // This could be a popup's presshell. No point in notifying XPConnect // about compositing of popups. return; } if (!mHasReceivedPaintMessage) { mHasReceivedPaintMessage = true; nsCOMPtr<nsIObserverService> obsvc = services::GetObserverService(); if (obsvc && mDocument) { nsPIDOMWindowOuter* window = mDocument->GetWindow(); nsCOMPtr<nsIDOMChromeWindow> chromeWin(do_QueryInterface(window)); if (chromeWin) { obsvc->NotifyObservers(chromeWin, "widget-first-paint", nullptr); } } } } bool PresShell::IsVisible() { if (!mIsActive || !mViewManager) return false; nsView* view = mViewManager->GetRootView(); if (!view) return true; // inner view of subdoc frame view = view->GetParent(); if (!view) return true; // subdoc view view = view->GetParent(); if (!view) return true; nsIFrame* frame = view->GetFrame(); if (!frame) return true; return frame->IsVisibleConsideringAncestors(nsIFrame::VISIBILITY_CROSS_CHROME_CONTENT_BOUNDARY); } nsresult PresShell::GetAgentStyleSheets(nsTArray<RefPtr<StyleSheet>>& aSheets) { aSheets.Clear(); int32_t sheetCount = mStyleSet->SheetCount(SheetType::Agent); if (!aSheets.SetCapacity(sheetCount, fallible)) { return NS_ERROR_OUT_OF_MEMORY; } for (int32_t i = 0; i < sheetCount; ++i) { StyleSheet* sheet = mStyleSet->StyleSheetAt(SheetType::Agent, i); aSheets.AppendElement(sheet); } return NS_OK; } nsresult PresShell::SetAgentStyleSheets(const nsTArray<RefPtr<StyleSheet>>& aSheets) { return mStyleSet->ReplaceSheets(SheetType::Agent, aSheets); } nsresult PresShell::AddOverrideStyleSheet(StyleSheet* aSheet) { return mStyleSet->PrependStyleSheet(SheetType::Override, aSheet); } nsresult PresShell::RemoveOverrideStyleSheet(StyleSheet* aSheet) { return mStyleSet->RemoveStyleSheet(SheetType::Override, aSheet); } static void FreezeElement(nsISupports *aSupports, void * /* unused */) { nsCOMPtr<nsIObjectLoadingContent> olc(do_QueryInterface(aSupports)); if (olc) { olc->StopPluginInstance(); } } static bool FreezeSubDocument(nsIDocument *aDocument, void *aData) { nsIPresShell *shell = aDocument->GetShell(); if (shell) shell->Freeze(); return true; } void PresShell::Freeze() { mUpdateApproximateFrameVisibilityEvent.Revoke(); MaybeReleaseCapturingContent(); mDocument->EnumerateActivityObservers(FreezeElement, nullptr); if (mCaret) { SetCaretEnabled(false); } mPaintingSuppressed = true; if (mDocument) { mDocument->EnumerateSubDocuments(FreezeSubDocument, nullptr); } nsPresContext* presContext = GetPresContext(); if (presContext && presContext->RefreshDriver()->GetPresContext() == presContext) { presContext->RefreshDriver()->Freeze(); } mFrozen = true; if (mDocument) { UpdateImageLockingState(); } } void PresShell::FireOrClearDelayedEvents(bool aFireEvents) { mNoDelayedMouseEvents = false; mNoDelayedKeyEvents = false; if (!aFireEvents) { mDelayedEvents.Clear(); return; } if (mDocument) { nsCOMPtr<nsIDocument> doc = mDocument; while (!mIsDestroying && mDelayedEvents.Length() && !doc->EventHandlingSuppressed()) { nsAutoPtr<DelayedEvent> ev(mDelayedEvents[0].forget()); mDelayedEvents.RemoveElementAt(0); ev->Dispatch(); } if (!doc->EventHandlingSuppressed()) { mDelayedEvents.Clear(); } } } static void ThawElement(nsISupports *aSupports, void *aShell) { nsCOMPtr<nsIObjectLoadingContent> olc(do_QueryInterface(aSupports)); if (olc) { olc->AsyncStartPluginInstance(); } } static bool ThawSubDocument(nsIDocument *aDocument, void *aData) { nsIPresShell *shell = aDocument->GetShell(); if (shell) shell->Thaw(); return true; } void PresShell::Thaw() { nsPresContext* presContext = GetPresContext(); if (presContext && presContext->RefreshDriver()->GetPresContext() == presContext) { presContext->RefreshDriver()->Thaw(); } mDocument->EnumerateActivityObservers(ThawElement, this); if (mDocument) mDocument->EnumerateSubDocuments(ThawSubDocument, nullptr); // Get the activeness of our presshell, as this might have changed // while we were in the bfcache QueryIsActive(); // We're now unfrozen mFrozen = false; UpdateImageLockingState(); UnsuppressPainting(); } //-------------------------------------------------------- // Start of protected and private methods on the PresShell //-------------------------------------------------------- void PresShell::MaybeScheduleReflow() { ASSERT_REFLOW_SCHEDULED_STATE(); if (mReflowScheduled || mIsDestroying || mIsReflowing || mDirtyRoots.Length() == 0) return; if (!mPresContext->HasPendingInterrupt() || !ScheduleReflowOffTimer()) { ScheduleReflow(); } ASSERT_REFLOW_SCHEDULED_STATE(); } void PresShell::ScheduleReflow() { NS_PRECONDITION(!mReflowScheduled, "Why are we trying to schedule a reflow?"); ASSERT_REFLOW_SCHEDULED_STATE(); if (GetPresContext()->RefreshDriver()->AddLayoutFlushObserver(this)) { mReflowScheduled = true; } ASSERT_REFLOW_SCHEDULED_STATE(); } nsresult PresShell::DidCauseReflow() { NS_ASSERTION(mChangeNestCount != 0, "Unexpected call to DidCauseReflow()"); --mChangeNestCount; nsContentUtils::RemoveScriptBlocker(); return NS_OK; } void PresShell::WillDoReflow() { mDocument->FlushUserFontSet(); mPresContext->FlushCounterStyles(); mFrameConstructor->BeginUpdate(); mLastReflowStart = GetPerformanceNow(); } void PresShell::DidDoReflow(bool aInterruptible) { mFrameConstructor->EndUpdate(); HandlePostedReflowCallbacks(aInterruptible); nsCOMPtr<nsIDocShell> docShell = mPresContext->GetDocShell(); if (docShell) { DOMHighResTimeStamp now = GetPerformanceNow(); docShell->NotifyReflowObservers(aInterruptible, mLastReflowStart, now); } if (sSynthMouseMove) { SynthesizeMouseMove(false); } mPresContext->NotifyMissingFonts(); } DOMHighResTimeStamp PresShell::GetPerformanceNow() { DOMHighResTimeStamp now = 0; if (nsPIDOMWindowInner* window = mDocument->GetInnerWindow()) { Performance* perf = window->GetPerformance(); if (perf) { now = perf->Now(); } } return now; } void PresShell::sReflowContinueCallback(nsITimer* aTimer, void* aPresShell) { RefPtr<PresShell> self = static_cast<PresShell*>(aPresShell); NS_PRECONDITION(aTimer == self->mReflowContinueTimer, "Unexpected timer"); self->mReflowContinueTimer = nullptr; self->ScheduleReflow(); } bool PresShell::ScheduleReflowOffTimer() { NS_PRECONDITION(!mReflowScheduled, "Shouldn't get here"); ASSERT_REFLOW_SCHEDULED_STATE(); if (!mReflowContinueTimer) { mReflowContinueTimer = do_CreateInstance("@mozilla.org/timer;1"); if (!mReflowContinueTimer || NS_FAILED(mReflowContinueTimer-> InitWithFuncCallback(sReflowContinueCallback, this, 30, nsITimer::TYPE_ONE_SHOT))) { return false; } } return true; } bool PresShell::DoReflow(nsIFrame* target, bool aInterruptible) { if (mIsZombie) { return true; } gfxTextPerfMetrics* tp = mPresContext->GetTextPerfMetrics(); TimeStamp timeStart; if (tp) { tp->Accumulate(); tp->reflowCount++; timeStart = TimeStamp::Now(); } target->SchedulePaint(); nsIFrame *parent = nsLayoutUtils::GetCrossDocParentFrame(target); while (parent) { nsSVGEffects::InvalidateDirectRenderingObservers(parent); parent = nsLayoutUtils::GetCrossDocParentFrame(parent); } nsIURI *uri = mDocument->GetDocumentURI(); PROFILER_LABEL_PRINTF("PresShell", "DoReflow", js::ProfileEntry::Category::GRAPHICS, "(%s)", uri ? uri->GetSpecOrDefault().get() : "N/A"); nsDocShell* docShell = static_cast<nsDocShell*>(GetPresContext()->GetDocShell()); RefPtr<TimelineConsumers> timelines = TimelineConsumers::Get(); bool isTimelineRecording = timelines && timelines->HasConsumer(docShell); if (isTimelineRecording) { timelines->AddMarkerForDocShell(docShell, "Reflow", MarkerTracingType::START); } if (mReflowContinueTimer) { mReflowContinueTimer->Cancel(); mReflowContinueTimer = nullptr; } nsIFrame* rootFrame = mFrameConstructor->GetRootFrame(); // CreateReferenceRenderingContext can return nullptr nsRenderingContext rcx(CreateReferenceRenderingContext()); #ifdef DEBUG mCurrentReflowRoot = target; #endif // If the target frame is the root of the frame hierarchy, then // use all the available space. If it's simply a `reflow root', // then use the target frame's size as the available space. WritingMode wm = target->GetWritingMode(); LogicalSize size(wm); if (target == rootFrame) { size = LogicalSize(wm, mPresContext->GetVisibleArea().Size()); } else { size = target->GetLogicalSize(); } NS_ASSERTION(!target->GetNextInFlow() && !target->GetPrevInFlow(), "reflow roots should never split"); // Don't pass size directly to the reflow state, since a // constrained height implies page/column breaking. LogicalSize reflowSize(wm, size.ISize(wm), NS_UNCONSTRAINEDSIZE); ReflowInput reflowInput(mPresContext, target, &rcx, reflowSize, ReflowInput::CALLER_WILL_INIT); reflowInput.mOrthogonalLimit = size.BSize(wm); if (rootFrame == target) { reflowInput.Init(mPresContext); // When the root frame is being reflowed with unconstrained block-size // (which happens when we're called from // nsDocumentViewer::SizeToContent), we're effectively doing a // resize in the block direction, since it changes the meaning of // percentage block-sizes even if no block-sizes actually changed. // The same applies when we reflow again after that computation. This is // an unusual case, and isn't caught by ReflowInput::InitResizeFlags. bool hasUnconstrainedBSize = size.BSize(wm) == NS_UNCONSTRAINEDSIZE; if (hasUnconstrainedBSize || mLastRootReflowHadUnconstrainedBSize) { reflowInput.SetBResize(true); } mLastRootReflowHadUnconstrainedBSize = hasUnconstrainedBSize; } else { // Initialize reflow state with current used border and padding, // in case this was set specially by the parent frame when the reflow root // was reflowed by its parent. nsMargin currentBorder = target->GetUsedBorder(); nsMargin currentPadding = target->GetUsedPadding(); reflowInput.Init(mPresContext, nullptr, ¤tBorder, ¤tPadding); } // fix the computed height NS_ASSERTION(reflowInput.ComputedPhysicalMargin() == nsMargin(0, 0, 0, 0), "reflow state should not set margin for reflow roots"); if (size.BSize(wm) != NS_UNCONSTRAINEDSIZE) { nscoord computedBSize = size.BSize(wm) - reflowInput.ComputedLogicalBorderPadding().BStartEnd(wm); computedBSize = std::max(computedBSize, 0); reflowInput.SetComputedBSize(computedBSize); } NS_ASSERTION(reflowInput.ComputedISize() == size.ISize(wm) - reflowInput.ComputedLogicalBorderPadding().IStartEnd(wm), "reflow state computed incorrect inline size"); mPresContext->ReflowStarted(aInterruptible); mIsReflowing = true; nsReflowStatus status; ReflowOutput desiredSize(reflowInput); target->Reflow(mPresContext, desiredSize, reflowInput, status); // If an incremental reflow is initiated at a frame other than the // root frame, then its desired size had better not change! If it's // initiated at the root, then the size better not change unless its // height was unconstrained to start with. nsRect boundsRelativeToTarget = nsRect(0, 0, desiredSize.Width(), desiredSize.Height()); NS_ASSERTION((target == rootFrame && size.BSize(wm) == NS_UNCONSTRAINEDSIZE) || (desiredSize.ISize(wm) == size.ISize(wm) && desiredSize.BSize(wm) == size.BSize(wm)), "non-root frame's desired size changed during an " "incremental reflow"); NS_ASSERTION(target == rootFrame || desiredSize.VisualOverflow().IsEqualInterior(boundsRelativeToTarget), "non-root reflow roots must not have visible overflow"); NS_ASSERTION(target == rootFrame || desiredSize.ScrollableOverflow().IsEqualEdges(boundsRelativeToTarget), "non-root reflow roots must not have scrollable overflow"); NS_ASSERTION(status == NS_FRAME_COMPLETE, "reflow roots should never split"); target->SetSize(boundsRelativeToTarget.Size()); // Always use boundsRelativeToTarget here, not desiredSize.GetVisualOverflowArea(), // because for root frames (where they could be different, since root frames // are allowed to have overflow) the root view bounds need to match the // viewport bounds; the view manager "window dimensions" code depends on it. nsContainerFrame::SyncFrameViewAfterReflow(mPresContext, target, target->GetView(), boundsRelativeToTarget); nsContainerFrame::SyncWindowProperties(mPresContext, target, target->GetView(), &rcx, nsContainerFrame::SET_ASYNC); target->DidReflow(mPresContext, nullptr, nsDidReflowStatus::FINISHED); if (target == rootFrame && size.BSize(wm) == NS_UNCONSTRAINEDSIZE) { mPresContext->SetVisibleArea(boundsRelativeToTarget); } #ifdef DEBUG mCurrentReflowRoot = nullptr; #endif NS_ASSERTION(mPresContext->HasPendingInterrupt() || mFramesToDirty.Count() == 0, "Why do we need to dirty anything if not interrupted?"); mIsReflowing = false; bool interrupted = mPresContext->HasPendingInterrupt(); if (interrupted) { // Make sure target gets reflowed again. for (auto iter = mFramesToDirty.Iter(); !iter.Done(); iter.Next()) { // Mark frames dirty until target frame. nsPtrHashKey<nsIFrame>* p = iter.Get(); for (nsIFrame* f = p->GetKey(); f && !NS_SUBTREE_DIRTY(f); f = f->GetParent()) { f->AddStateBits(NS_FRAME_HAS_DIRTY_CHILDREN); if (f == target) { break; } } } NS_ASSERTION(NS_SUBTREE_DIRTY(target), "Why is the target not dirty?"); mDirtyRoots.AppendElement(target); mDocument->SetNeedLayoutFlush(); // Clear mFramesToDirty after we've done the NS_SUBTREE_DIRTY(target) // assertion so that if it fails it's easier to see what's going on. #ifdef NOISY_INTERRUPTIBLE_REFLOW printf("mFramesToDirty.Count() == %u\n", mFramesToDirty.Count()); #endif /* NOISY_INTERRUPTIBLE_REFLOW */ mFramesToDirty.Clear(); // Any FlushPendingNotifications with interruptible reflows // should be suppressed now. We don't want to do extra reflow work // before our reflow event happens. mSuppressInterruptibleReflows = true; MaybeScheduleReflow(); } // dump text perf metrics for reflows with significant text processing if (tp) { if (tp->current.numChars > 100) { TimeDuration reflowTime = TimeStamp::Now() - timeStart; LogTextPerfStats(tp, this, tp->current, reflowTime.ToMilliseconds(), eLog_reflow, nullptr); } tp->Accumulate(); } if (isTimelineRecording) { timelines->AddMarkerForDocShell(docShell, "Reflow", MarkerTracingType::END); } return !interrupted; } #ifdef DEBUG void PresShell::DoVerifyReflow() { if (GetVerifyReflowEnable()) { // First synchronously render what we have so far so that we can // see it. nsView* rootView = mViewManager->GetRootView(); mViewManager->InvalidateView(rootView); FlushPendingNotifications(Flush_Layout); mInVerifyReflow = true; bool ok = VerifyIncrementalReflow(); mInVerifyReflow = false; if (VERIFY_REFLOW_ALL & gVerifyReflowFlags) { printf("ProcessReflowCommands: finished (%s)\n", ok ? "ok" : "failed"); } if (!mDirtyRoots.IsEmpty()) { printf("XXX yikes! reflow commands queued during verify-reflow\n"); } } } #endif // used with Telemetry metrics #define NS_LONG_REFLOW_TIME_MS 5000 bool PresShell::ProcessReflowCommands(bool aInterruptible) { if (mDirtyRoots.IsEmpty() && !mShouldUnsuppressPainting) { // Nothing to do; bail out return true; } mozilla::TimeStamp timerStart = mozilla::TimeStamp::Now(); bool interrupted = false; if (!mDirtyRoots.IsEmpty()) { #ifdef DEBUG if (VERIFY_REFLOW_DUMP_COMMANDS & gVerifyReflowFlags) { printf("ProcessReflowCommands: begin incremental reflow\n"); } #endif // If reflow is interruptible, then make a note of our deadline. const PRIntervalTime deadline = aInterruptible ? PR_IntervalNow() + PR_MicrosecondsToInterval(gMaxRCProcessingTime) : (PRIntervalTime)0; // Scope for the reflow entry point { nsAutoScriptBlocker scriptBlocker; WillDoReflow(); AUTO_LAYOUT_PHASE_ENTRY_POINT(GetPresContext(), Reflow); nsViewManager::AutoDisableRefresh refreshBlocker(mViewManager); do { // Send an incremental reflow notification to the target frame. int32_t idx = mDirtyRoots.Length() - 1; nsIFrame *target = mDirtyRoots[idx]; mDirtyRoots.RemoveElementAt(idx); if (!NS_SUBTREE_DIRTY(target)) { // It's not dirty anymore, which probably means the notification // was posted in the middle of a reflow (perhaps with a reflow // root in the middle). Don't do anything. continue; } interrupted = !DoReflow(target, aInterruptible); // Keep going until we're out of reflow commands, or we've run // past our deadline, or we're interrupted. } while (!interrupted && !mDirtyRoots.IsEmpty() && (!aInterruptible || PR_IntervalNow() < deadline)); interrupted = !mDirtyRoots.IsEmpty(); } // Exiting the scriptblocker might have killed us if (!mIsDestroying) { DidDoReflow(aInterruptible); } // DidDoReflow might have killed us if (!mIsDestroying) { #ifdef DEBUG if (VERIFY_REFLOW_DUMP_COMMANDS & gVerifyReflowFlags) { printf("\nPresShell::ProcessReflowCommands() finished: this=%p\n", (void*)this); } DoVerifyReflow(); #endif // If any new reflow commands were enqueued during the reflow, schedule // another reflow event to process them. Note that we want to do this // after DidDoReflow(), since that method can change whether there are // dirty roots around by flushing, and there's no point in posting a // reflow event just to have the flush revoke it. if (!mDirtyRoots.IsEmpty()) { MaybeScheduleReflow(); // And tell our document that we might need flushing mDocument->SetNeedLayoutFlush(); } } } if (!mIsDestroying && mShouldUnsuppressPainting && mDirtyRoots.IsEmpty()) { // We only unlock if we're out of reflows. It's pointless // to unlock if reflows are still pending, since reflows // are just going to thrash the frames around some more. By // waiting we avoid an overeager "jitter" effect. mShouldUnsuppressPainting = false; UnsuppressAndInvalidate(); } if (mDocument->GetRootElement()) { TimeDuration elapsed = TimeStamp::Now() - timerStart; int32_t intElapsed = int32_t(elapsed.ToMilliseconds()); if (intElapsed > NS_LONG_REFLOW_TIME_MS) { Telemetry::Accumulate(Telemetry::LONG_REFLOW_INTERRUPTIBLE, aInterruptible ? 1 : 0); } } return !interrupted; } void PresShell::WindowSizeMoveDone() { if (mPresContext) { EventStateManager::ClearGlobalActiveContent(nullptr); ClearMouseCapture(nullptr); } } #ifdef MOZ_XUL /* * It's better to add stuff to the |DidSetStyleContext| method of the * relevant frames than adding it here. These methods should (ideally, * anyway) go away. */ // Return value says whether to walk children. typedef bool (* frameWalkerFn)(nsIFrame *aFrame, void *aClosure); static bool ReResolveMenusAndTrees(nsIFrame *aFrame, void *aClosure) { // Trees have a special style cache that needs to be flushed when // the theme changes. nsTreeBodyFrame *treeBody = do_QueryFrame(aFrame); if (treeBody) treeBody->ClearStyleAndImageCaches(); // We deliberately don't re-resolve style on a menu's popup // sub-content, since doing so slows menus to a crawl. That means we // have to special-case them on a skin switch, and ensure that the // popup frames just get destroyed completely. nsMenuFrame* menu = do_QueryFrame(aFrame); if (menu) menu->CloseMenu(true); return true; } static bool ReframeImageBoxes(nsIFrame *aFrame, void *aClosure) { nsStyleChangeList *list = static_cast<nsStyleChangeList*>(aClosure); if (aFrame->GetType() == nsGkAtoms::imageBoxFrame) { list->AppendChange(aFrame, aFrame->GetContent(), nsChangeHint_ReconstructFrame); return false; // don't walk descendants } return true; // walk descendants } static void WalkFramesThroughPlaceholders(nsPresContext *aPresContext, nsIFrame *aFrame, frameWalkerFn aFunc, void *aClosure) { bool walkChildren = (*aFunc)(aFrame, aClosure); if (!walkChildren) return; nsIFrame::ChildListIterator lists(aFrame); for (; !lists.IsDone(); lists.Next()) { nsFrameList::Enumerator childFrames(lists.CurrentList()); for (; !childFrames.AtEnd(); childFrames.Next()) { nsIFrame* child = childFrames.get(); if (!(child->GetStateBits() & NS_FRAME_OUT_OF_FLOW)) { // only do frames that are in flow, and recur through the // out-of-flows of placeholders. WalkFramesThroughPlaceholders(aPresContext, nsPlaceholderFrame::GetRealFrameFor(child), aFunc, aClosure); } } } } #endif NS_IMETHODIMP PresShell::Observe(nsISupports* aSubject, const char* aTopic, const char16_t* aData) { if (mIsDestroying) { NS_WARNING("our observers should have been unregistered by now"); return NS_OK; } #ifdef MOZ_XUL if (!nsCRT::strcmp(aTopic, "chrome-flush-skin-caches")) { nsIFrame *rootFrame = mFrameConstructor->GetRootFrame(); // Need to null-check because "chrome-flush-skin-caches" can happen // at interesting times during startup. if (rootFrame) { NS_ASSERTION(mViewManager, "View manager must exist"); nsWeakFrame weakRoot(rootFrame); // Have to make sure that the content notifications are flushed before we // start messing with the frame model; otherwise we can get content doubling. mDocument->FlushPendingNotifications(Flush_ContentAndNotify); if (weakRoot.IsAlive()) { WalkFramesThroughPlaceholders(mPresContext, rootFrame, &ReResolveMenusAndTrees, nullptr); // Because "chrome:" URL equality is messy, reframe image box // frames (hack!). nsStyleChangeList changeList; WalkFramesThroughPlaceholders(mPresContext, rootFrame, ReframeImageBoxes, &changeList); // Mark ourselves as not safe to flush while we're doing frame // construction. { nsAutoScriptBlocker scriptBlocker; ++mChangeNestCount; RestyleManagerHandle restyleManager = mPresContext->RestyleManager(); if (restyleManager->IsServo()) { MOZ_CRASH("stylo: PresShell::Observe(\"chrome-flush-skin-caches\") " "not implemented for Servo-backed style system"); } restyleManager->AsGecko()->ProcessRestyledFrames(changeList); restyleManager->AsGecko()->FlushOverflowChangedTracker(); --mChangeNestCount; } } } return NS_OK; } #endif if (!nsCRT::strcmp(aTopic, "agent-sheet-added")) { if (mStyleSet) { AddAgentSheet(aSubject); } return NS_OK; } if (!nsCRT::strcmp(aTopic, "user-sheet-added")) { if (mStyleSet) { AddUserSheet(aSubject); } return NS_OK; } if (!nsCRT::strcmp(aTopic, "author-sheet-added")) { if (mStyleSet) { AddAuthorSheet(aSubject); } return NS_OK; } if (!nsCRT::strcmp(aTopic, "agent-sheet-removed")) { if (mStyleSet) { RemoveSheet(SheetType::Agent, aSubject); } return NS_OK; } if (!nsCRT::strcmp(aTopic, "user-sheet-removed")) { if (mStyleSet) { RemoveSheet(SheetType::User, aSubject); } return NS_OK; } if (!nsCRT::strcmp(aTopic, "author-sheet-removed")) { if (mStyleSet) { RemoveSheet(SheetType::Doc, aSubject); } return NS_OK; } if (!nsCRT::strcmp(aTopic, "memory-pressure")) { if (!AssumeAllFramesVisible() && mPresContext->IsRootContentDocument()) { DoUpdateApproximateFrameVisibility(/* aRemoveOnly = */ true); } return NS_OK; } NS_WARNING("unrecognized topic in PresShell::Observe"); return NS_ERROR_FAILURE; } bool nsIPresShell::AddRefreshObserverInternal(nsARefreshObserver* aObserver, mozFlushType aFlushType) { nsPresContext* presContext = GetPresContext(); return presContext && presContext->RefreshDriver()->AddRefreshObserver(aObserver, aFlushType); } /* virtual */ bool nsIPresShell::AddRefreshObserverExternal(nsARefreshObserver* aObserver, mozFlushType aFlushType) { return AddRefreshObserverInternal(aObserver, aFlushType); } bool nsIPresShell::RemoveRefreshObserverInternal(nsARefreshObserver* aObserver, mozFlushType aFlushType) { nsPresContext* presContext = GetPresContext(); return presContext && presContext->RefreshDriver()->RemoveRefreshObserver(aObserver, aFlushType); } /* virtual */ bool nsIPresShell::RemoveRefreshObserverExternal(nsARefreshObserver* aObserver, mozFlushType aFlushType) { return RemoveRefreshObserverInternal(aObserver, aFlushType); } /* virtual */ bool nsIPresShell::AddPostRefreshObserver(nsAPostRefreshObserver* aObserver) { nsPresContext* presContext = GetPresContext(); if (!presContext) { return false; } presContext->RefreshDriver()->AddPostRefreshObserver(aObserver); return true; } /* virtual */ bool nsIPresShell::RemovePostRefreshObserver(nsAPostRefreshObserver* aObserver) { nsPresContext* presContext = GetPresContext(); if (!presContext) { return false; } presContext->RefreshDriver()->RemovePostRefreshObserver(aObserver); return true; } //------------------------------------------------------ // End of protected and private methods on the PresShell //------------------------------------------------------ //------------------------------------------------------------------ //-- Delayed event Classes Impls //------------------------------------------------------------------ PresShell::DelayedInputEvent::DelayedInputEvent() : DelayedEvent(), mEvent(nullptr) { } PresShell::DelayedInputEvent::~DelayedInputEvent() { delete mEvent; } void PresShell::DelayedInputEvent::Dispatch() { if (!mEvent || !mEvent->mWidget) { return; } nsCOMPtr<nsIWidget> widget = mEvent->mWidget; nsEventStatus status; widget->DispatchEvent(mEvent, status); } PresShell::DelayedMouseEvent::DelayedMouseEvent(WidgetMouseEvent* aEvent) : DelayedInputEvent() { WidgetMouseEvent* mouseEvent = new WidgetMouseEvent(aEvent->IsTrusted(), aEvent->mMessage, aEvent->mWidget, aEvent->mReason, aEvent->mContextMenuTrigger); mouseEvent->AssignMouseEventData(*aEvent, false); mEvent = mouseEvent; } PresShell::DelayedKeyEvent::DelayedKeyEvent(WidgetKeyboardEvent* aEvent) : DelayedInputEvent() { WidgetKeyboardEvent* keyEvent = new WidgetKeyboardEvent(aEvent->IsTrusted(), aEvent->mMessage, aEvent->mWidget); keyEvent->AssignKeyEventData(*aEvent, false); keyEvent->mFlags.mIsSynthesizedForTests = aEvent->mFlags.mIsSynthesizedForTests; mEvent = keyEvent; } // Start of DEBUG only code #ifdef DEBUG static void LogVerifyMessage(nsIFrame* k1, nsIFrame* k2, const char* aMsg) { nsAutoString n1, n2; if (k1) { k1->GetFrameName(n1); } else { n1.AssignLiteral(u"(null)"); } if (k2) { k2->GetFrameName(n2); } else { n2.AssignLiteral(u"(null)"); } printf("verifyreflow: %s %p != %s %p %s\n", NS_LossyConvertUTF16toASCII(n1).get(), (void*)k1, NS_LossyConvertUTF16toASCII(n2).get(), (void*)k2, aMsg); } static void LogVerifyMessage(nsIFrame* k1, nsIFrame* k2, const char* aMsg, const nsRect& r1, const nsRect& r2) { printf("VerifyReflow Error:\n"); nsAutoString name; if (k1) { k1->GetFrameName(name); printf(" %s %p ", NS_LossyConvertUTF16toASCII(name).get(), (void*)k1); } printf("{%d, %d, %d, %d} != \n", r1.x, r1.y, r1.width, r1.height); if (k2) { k2->GetFrameName(name); printf(" %s %p ", NS_LossyConvertUTF16toASCII(name).get(), (void*)k2); } printf("{%d, %d, %d, %d}\n %s\n", r2.x, r2.y, r2.width, r2.height, aMsg); } static void LogVerifyMessage(nsIFrame* k1, nsIFrame* k2, const char* aMsg, const nsIntRect& r1, const nsIntRect& r2) { printf("VerifyReflow Error:\n"); nsAutoString name; if (k1) { k1->GetFrameName(name); printf(" %s %p ", NS_LossyConvertUTF16toASCII(name).get(), (void*)k1); } printf("{%d, %d, %d, %d} != \n", r1.x, r1.y, r1.width, r1.height); if (k2) { k2->GetFrameName(name); printf(" %s %p ", NS_LossyConvertUTF16toASCII(name).get(), (void*)k2); } printf("{%d, %d, %d, %d}\n %s\n", r2.x, r2.y, r2.width, r2.height, aMsg); } static bool CompareTrees(nsPresContext* aFirstPresContext, nsIFrame* aFirstFrame, nsPresContext* aSecondPresContext, nsIFrame* aSecondFrame) { if (!aFirstPresContext || !aFirstFrame || !aSecondPresContext || !aSecondFrame) return true; // XXX Evil hack to reduce false positives; I can't seem to figure // out how to flush scrollbar changes correctly //if (aFirstFrame->GetType() == nsGkAtoms::scrollbarFrame) // return true; bool ok = true; nsIFrame::ChildListIterator lists1(aFirstFrame); nsIFrame::ChildListIterator lists2(aSecondFrame); do { const nsFrameList& kids1 = !lists1.IsDone() ? lists1.CurrentList() : nsFrameList(); const nsFrameList& kids2 = !lists2.IsDone() ? lists2.CurrentList() : nsFrameList(); int32_t l1 = kids1.GetLength(); int32_t l2 = kids2.GetLength(); if (l1 != l2) { ok = false; LogVerifyMessage(kids1.FirstChild(), kids2.FirstChild(), "child counts don't match: "); printf("%d != %d\n", l1, l2); if (0 == (VERIFY_REFLOW_ALL & gVerifyReflowFlags)) { break; } } LayoutDeviceIntRect r1, r2; nsView* v1; nsView* v2; for (nsFrameList::Enumerator e1(kids1), e2(kids2); ; e1.Next(), e2.Next()) { nsIFrame* k1 = e1.get(); nsIFrame* k2 = e2.get(); if (((nullptr == k1) && (nullptr != k2)) || ((nullptr != k1) && (nullptr == k2))) { ok = false; LogVerifyMessage(k1, k2, "child lists are different\n"); break; } else if (nullptr != k1) { // Verify that the frames are the same size if (!k1->GetRect().IsEqualInterior(k2->GetRect())) { ok = false; LogVerifyMessage(k1, k2, "(frame rects)", k1->GetRect(), k2->GetRect()); } // Make sure either both have views or neither have views; if they // do have views, make sure the views are the same size. If the // views have widgets, make sure they both do or neither does. If // they do, make sure the widgets are the same size. v1 = k1->GetView(); v2 = k2->GetView(); if (((nullptr == v1) && (nullptr != v2)) || ((nullptr != v1) && (nullptr == v2))) { ok = false; LogVerifyMessage(k1, k2, "child views are not matched\n"); } else if (nullptr != v1) { if (!v1->GetBounds().IsEqualInterior(v2->GetBounds())) { LogVerifyMessage(k1, k2, "(view rects)", v1->GetBounds(), v2->GetBounds()); } nsIWidget* w1 = v1->GetWidget(); nsIWidget* w2 = v2->GetWidget(); if (((nullptr == w1) && (nullptr != w2)) || ((nullptr != w1) && (nullptr == w2))) { ok = false; LogVerifyMessage(k1, k2, "child widgets are not matched\n"); } else if (nullptr != w1) { r1 = w1->GetBounds(); r2 = w2->GetBounds(); if (!r1.IsEqualEdges(r2)) { LogVerifyMessage(k1, k2, "(widget rects)", r1.ToUnknownRect(), r2.ToUnknownRect()); } } } if (!ok && (0 == (VERIFY_REFLOW_ALL & gVerifyReflowFlags))) { break; } // XXX Should perhaps compare their float managers. // Compare the sub-trees too if (!CompareTrees(aFirstPresContext, k1, aSecondPresContext, k2)) { ok = false; if (0 == (VERIFY_REFLOW_ALL & gVerifyReflowFlags)) { break; } } } else { break; } } if (!ok && (0 == (VERIFY_REFLOW_ALL & gVerifyReflowFlags))) { break; } lists1.Next(); lists2.Next(); if (lists1.IsDone() != lists2.IsDone() || (!lists1.IsDone() && lists1.CurrentID() != lists2.CurrentID())) { if (0 == (VERIFY_REFLOW_ALL & gVerifyReflowFlags)) { ok = false; } LogVerifyMessage(kids1.FirstChild(), kids2.FirstChild(), "child list names are not matched: "); fprintf(stdout, "%s != %s\n", !lists1.IsDone() ? mozilla::layout::ChildListName(lists1.CurrentID()) : "(null)", !lists2.IsDone() ? mozilla::layout::ChildListName(lists2.CurrentID()) : "(null)"); break; } } while (ok && !lists1.IsDone()); return ok; } #endif #if 0 static nsIFrame* FindTopFrame(nsIFrame* aRoot) { if (aRoot) { nsIContent* content = aRoot->GetContent(); if (content) { nsIAtom* tag; content->GetTag(tag); if (nullptr != tag) { NS_RELEASE(tag); return aRoot; } } // Try one of the children for (nsIFrame* kid : aRoot->PrincipalChildList()) { nsIFrame* result = FindTopFrame(kid); if (nullptr != result) { return result; } } } return nullptr; } #endif #ifdef DEBUG nsStyleSet* PresShell::CloneStyleSet(nsStyleSet* aSet) { nsStyleSet* clone = new nsStyleSet(); int32_t i, n = aSet->SheetCount(SheetType::Override); for (i = 0; i < n; i++) { CSSStyleSheet* ss = aSet->StyleSheetAt(SheetType::Override, i); if (ss) clone->AppendStyleSheet(SheetType::Override, ss); } // The document expects to insert document stylesheets itself #if 0 n = aSet->SheetCount(SheetType::Doc); for (i = 0; i < n; i++) { CSSStyleSheet* ss = aSet->StyleSheetAt(SheetType::Doc, i); if (ss) clone->AddDocStyleSheet(ss, mDocument); } #endif n = aSet->SheetCount(SheetType::User); for (i = 0; i < n; i++) { CSSStyleSheet* ss = aSet->StyleSheetAt(SheetType::User, i); if (ss) clone->AppendStyleSheet(SheetType::User, ss); } n = aSet->SheetCount(SheetType::Agent); for (i = 0; i < n; i++) { CSSStyleSheet* ss = aSet->StyleSheetAt(SheetType::Agent, i); if (ss) clone->AppendStyleSheet(SheetType::Agent, ss); } return clone; } // After an incremental reflow, we verify the correctness by doing a // full reflow into a fresh frame tree. bool PresShell::VerifyIncrementalReflow() { if (VERIFY_REFLOW_NOISY & gVerifyReflowFlags) { printf("Building Verification Tree...\n"); } // Create a presentation context to view the new frame tree RefPtr<nsPresContext> cx = new nsRootPresContext(mDocument, mPresContext->IsPaginated() ? nsPresContext::eContext_PrintPreview : nsPresContext::eContext_Galley); NS_ENSURE_TRUE(cx, false); nsDeviceContext *dc = mPresContext->DeviceContext(); nsresult rv = cx->Init(dc); NS_ENSURE_SUCCESS(rv, false); // Get our scrolling preference nsView* rootView = mViewManager->GetRootView(); NS_ENSURE_TRUE(rootView->HasWidget(), false); nsIWidget* parentWidget = rootView->GetWidget(); // Create a new view manager. RefPtr<nsViewManager> vm = new nsViewManager(); NS_ENSURE_TRUE(vm, false); rv = vm->Init(dc); NS_ENSURE_SUCCESS(rv, false); // Create a child window of the parent that is our "root view/window" // Create a view nsRect tbounds = mPresContext->GetVisibleArea(); nsView* view = vm->CreateView(tbounds, nullptr); NS_ENSURE_TRUE(view, false); //now create the widget for the view rv = view->CreateWidgetForParent(parentWidget, nullptr, true); NS_ENSURE_SUCCESS(rv, false); // Setup hierarchical relationship in view manager vm->SetRootView(view); // Make the new presentation context the same size as our // presentation context. nsRect r = mPresContext->GetVisibleArea(); cx->SetVisibleArea(r); // Create a new presentation shell to view the document. Use the // exact same style information that this document has. if (mStyleSet->IsServo()) { NS_WARNING("VerifyIncrementalReflow cannot handle ServoStyleSets"); return true; } nsAutoPtr<nsStyleSet> newSet(CloneStyleSet(mStyleSet->AsGecko())); nsCOMPtr<nsIPresShell> sh = mDocument->CreateShell(cx, vm, newSet.get()); NS_ENSURE_TRUE(sh, false); newSet.forget(); // Note that after we create the shell, we must make sure to destroy it sh->SetVerifyReflowEnable(false); // turn off verify reflow while we're reflowing the test frame tree vm->SetPresShell(sh); { nsAutoCauseReflowNotifier crNotifier(this); sh->Initialize(r.width, r.height); } mDocument->BindingManager()->ProcessAttachedQueue(); sh->FlushPendingNotifications(Flush_Layout); sh->SetVerifyReflowEnable(true); // turn on verify reflow again now that we're done reflowing the test frame tree // Force the non-primary presshell to unsuppress; it doesn't want to normally // because it thinks it's hidden ((PresShell*)sh.get())->mPaintingSuppressed = false; if (VERIFY_REFLOW_NOISY & gVerifyReflowFlags) { printf("Verification Tree built, comparing...\n"); } // Now that the document has been reflowed, use its frame tree to // compare against our frame tree. nsIFrame* root1 = mFrameConstructor->GetRootFrame(); nsIFrame* root2 = sh->GetRootFrame(); bool ok = CompareTrees(mPresContext, root1, cx, root2); if (!ok && (VERIFY_REFLOW_NOISY & gVerifyReflowFlags)) { printf("Verify reflow failed, primary tree:\n"); root1->List(stdout, 0); printf("Verification tree:\n"); root2->List(stdout, 0); } #if 0 // Sample code for dumping page to png // XXX Needs to be made more flexible if (!ok) { nsString stra; static int num = 0; stra.AppendLiteral("C:\\mozilla\\mozilla\\debug\\filea"); stra.AppendInt(num); stra.AppendLiteral(".png"); gfxUtils::WriteAsPNG(sh, stra); nsString strb; strb.AppendLiteral("C:\\mozilla\\mozilla\\debug\\fileb"); strb.AppendInt(num); strb.AppendLiteral(".png"); gfxUtils::WriteAsPNG(sh, strb); ++num; } #endif sh->EndObservingDocument(); sh->Destroy(); if (VERIFY_REFLOW_NOISY & gVerifyReflowFlags) { printf("Finished Verifying Reflow...\n"); } return ok; } // Layout debugging hooks void PresShell::ListStyleContexts(nsIFrame *aRootFrame, FILE *out, int32_t aIndent) { nsStyleContext *sc = aRootFrame->StyleContext(); if (sc) sc->List(out, aIndent); } void PresShell::ListStyleSheets(FILE *out, int32_t aIndent) { int32_t sheetCount = mStyleSet->SheetCount(SheetType::Doc); for (int32_t i = 0; i < sheetCount; ++i) { mStyleSet->StyleSheetAt(SheetType::Doc, i)->List(out, aIndent); fputs("\n", out); } } void PresShell::VerifyStyleTree() { VERIFY_STYLE_TREE; } #endif //============================================================= //============================================================= //-- Debug Reflow Counts //============================================================= //============================================================= #ifdef MOZ_REFLOW_PERF //------------------------------------------------------------- void PresShell::DumpReflows() { if (mReflowCountMgr) { nsAutoCString uriStr; if (mDocument) { nsIURI *uri = mDocument->GetDocumentURI(); if (uri) { uri->GetPath(uriStr); } } mReflowCountMgr->DisplayTotals(uriStr.get()); mReflowCountMgr->DisplayHTMLTotals(uriStr.get()); mReflowCountMgr->DisplayDiffsInTotals(); } } //------------------------------------------------------------- void PresShell::CountReflows(const char * aName, nsIFrame * aFrame) { if (mReflowCountMgr) { mReflowCountMgr->Add(aName, aFrame); } } //------------------------------------------------------------- void PresShell::PaintCount(const char * aName, nsRenderingContext* aRenderingContext, nsPresContext* aPresContext, nsIFrame * aFrame, const nsPoint& aOffset, uint32_t aColor) { if (mReflowCountMgr) { mReflowCountMgr->PaintCount(aName, aRenderingContext, aPresContext, aFrame, aOffset, aColor); } } //------------------------------------------------------------- void PresShell::SetPaintFrameCount(bool aPaintFrameCounts) { if (mReflowCountMgr) { mReflowCountMgr->SetPaintFrameCounts(aPaintFrameCounts); } } bool PresShell::IsPaintingFrameCounts() { if (mReflowCountMgr) return mReflowCountMgr->IsPaintingFrameCounts(); return false; } //------------------------------------------------------------------ //-- Reflow Counter Classes Impls //------------------------------------------------------------------ //------------------------------------------------------------------ ReflowCounter::ReflowCounter(ReflowCountMgr * aMgr) : mMgr(aMgr) { ClearTotals(); SetTotalsCache(); } //------------------------------------------------------------------ ReflowCounter::~ReflowCounter() { } //------------------------------------------------------------------ void ReflowCounter::ClearTotals() { mTotal = 0; } //------------------------------------------------------------------ void ReflowCounter::SetTotalsCache() { mCacheTotal = mTotal; } //------------------------------------------------------------------ void ReflowCounter::CalcDiffInTotals() { mCacheTotal = mTotal - mCacheTotal; } //------------------------------------------------------------------ void ReflowCounter::DisplayTotals(const char * aStr) { DisplayTotals(mTotal, aStr?aStr:"Totals"); } //------------------------------------------------------------------ void ReflowCounter::DisplayDiffTotals(const char * aStr) { DisplayTotals(mCacheTotal, aStr?aStr:"Diff Totals"); } //------------------------------------------------------------------ void ReflowCounter::DisplayHTMLTotals(const char * aStr) { DisplayHTMLTotals(mTotal, aStr?aStr:"Totals"); } //------------------------------------------------------------------ void ReflowCounter::DisplayTotals(uint32_t aTotal, const char * aTitle) { // figure total if (aTotal == 0) { return; } ReflowCounter * gTots = (ReflowCounter *)mMgr->LookUp(kGrandTotalsStr); printf("%25s\t", aTitle); printf("%d\t", aTotal); if (gTots != this && aTotal > 0) { gTots->Add(aTotal); } } //------------------------------------------------------------------ void ReflowCounter::DisplayHTMLTotals(uint32_t aTotal, const char * aTitle) { if (aTotal == 0) { return; } ReflowCounter * gTots = (ReflowCounter *)mMgr->LookUp(kGrandTotalsStr); FILE * fd = mMgr->GetOutFile(); if (!fd) { return; } fprintf(fd, "<tr><td><center>%s</center></td>", aTitle); fprintf(fd, "<td><center>%d</center></td></tr>\n", aTotal); if (gTots != this && aTotal > 0) { gTots->Add(aTotal); } } //------------------------------------------------------------------ //-- ReflowCountMgr //------------------------------------------------------------------ #define KEY_BUF_SIZE_FOR_PTR 24 // adequate char[] buffer to sprintf a pointer ReflowCountMgr::ReflowCountMgr() { mCounts = PL_NewHashTable(10, PL_HashString, PL_CompareStrings, PL_CompareValues, nullptr, nullptr); mIndiFrameCounts = PL_NewHashTable(10, PL_HashString, PL_CompareStrings, PL_CompareValues, nullptr, nullptr); mCycledOnce = false; mDumpFrameCounts = false; mDumpFrameByFrameCounts = false; mPaintFrameByFrameCounts = false; } //------------------------------------------------------------------ ReflowCountMgr::~ReflowCountMgr() { CleanUp(); } //------------------------------------------------------------------ ReflowCounter * ReflowCountMgr::LookUp(const char * aName) { if (nullptr != mCounts) { ReflowCounter * counter = (ReflowCounter *)PL_HashTableLookup(mCounts, aName); return counter; } return nullptr; } //------------------------------------------------------------------ void ReflowCountMgr::Add(const char * aName, nsIFrame * aFrame) { NS_ASSERTION(aName != nullptr, "Name shouldn't be null!"); if (mDumpFrameCounts && nullptr != mCounts) { ReflowCounter * counter = (ReflowCounter *)PL_HashTableLookup(mCounts, aName); if (counter == nullptr) { counter = new ReflowCounter(this); char * name = NS_strdup(aName); NS_ASSERTION(name != nullptr, "null ptr"); PL_HashTableAdd(mCounts, name, counter); } counter->Add(); } if ((mDumpFrameByFrameCounts || mPaintFrameByFrameCounts) && nullptr != mIndiFrameCounts && aFrame != nullptr) { char key[KEY_BUF_SIZE_FOR_PTR]; SprintfLiteral(key, "%p", (void*)aFrame); IndiReflowCounter * counter = (IndiReflowCounter *)PL_HashTableLookup(mIndiFrameCounts, key); if (counter == nullptr) { counter = new IndiReflowCounter(this); counter->mFrame = aFrame; counter->mName.AssignASCII(aName); PL_HashTableAdd(mIndiFrameCounts, NS_strdup(key), counter); } // this eliminates extra counts from super classes if (counter != nullptr && counter->mName.EqualsASCII(aName)) { counter->mCount++; counter->mCounter.Add(1); } } } //------------------------------------------------------------------ void ReflowCountMgr::PaintCount(const char* aName, nsRenderingContext* aRenderingContext, nsPresContext* aPresContext, nsIFrame* aFrame, const nsPoint& aOffset, uint32_t aColor) { if (mPaintFrameByFrameCounts && nullptr != mIndiFrameCounts && aFrame != nullptr) { char key[KEY_BUF_SIZE_FOR_PTR]; SprintfLiteral(key, "%p", (void*)aFrame); IndiReflowCounter * counter = (IndiReflowCounter *)PL_HashTableLookup(mIndiFrameCounts, key); if (counter != nullptr && counter->mName.EqualsASCII(aName)) { DrawTarget* drawTarget = aRenderingContext->GetDrawTarget(); int32_t appUnitsPerDevPixel = aPresContext->AppUnitsPerDevPixel(); aRenderingContext->ThebesContext()->Save(); gfxPoint devPixelOffset = nsLayoutUtils::PointToGfxPoint(aOffset, appUnitsPerDevPixel); aRenderingContext->ThebesContext()->SetMatrix( aRenderingContext->ThebesContext()->CurrentMatrix().Translate(devPixelOffset)); // We don't care about the document language or user fonts here; // just get a default Latin font. nsFont font(eFamily_serif, nsPresContext::CSSPixelsToAppUnits(11)); nsFontMetrics::Params params; params.language = nsGkAtoms::x_western; params.textPerf = aPresContext->GetTextPerfMetrics(); RefPtr<nsFontMetrics> fm = aPresContext->DeviceContext()->GetMetricsFor(font, params); char buf[16]; int len = SprintfLiteral(buf, "%d", counter->mCount); nscoord x = 0, y = fm->MaxAscent(); nscoord width, height = fm->MaxHeight(); fm->SetTextRunRTL(false); width = fm->GetWidth(buf, len, drawTarget); Color color; Color color2; if (aColor != 0) { color = Color::FromABGR(aColor); color2 = Color(0.f, 0.f, 0.f); } else { gfx::Float rc = 0.f, gc = 0.f, bc = 0.f; if (counter->mCount < 5) { rc = 1.f; gc = 1.f; } else if (counter->mCount < 11) { gc = 1.f; } else { rc = 1.f; } color = Color(rc, gc, bc); color2 = Color(rc/2, gc/2, bc/2); } nsRect rect(0,0, width+15, height+15); Rect devPxRect = NSRectToSnappedRect(rect, appUnitsPerDevPixel, *drawTarget); ColorPattern black(ToDeviceColor(Color(0.f, 0.f, 0.f, 1.f))); drawTarget->FillRect(devPxRect, black); aRenderingContext->ThebesContext()->SetColor(color2); fm->DrawString(buf, len, x+15, y+15, aRenderingContext); aRenderingContext->ThebesContext()->SetColor(color); fm->DrawString(buf, len, x, y, aRenderingContext); aRenderingContext->ThebesContext()->Restore(); } } } //------------------------------------------------------------------ int ReflowCountMgr::RemoveItems(PLHashEntry *he, int i, void *arg) { char *str = (char *)he->key; ReflowCounter * counter = (ReflowCounter *)he->value; delete counter; free(str); return HT_ENUMERATE_REMOVE; } //------------------------------------------------------------------ int ReflowCountMgr::RemoveIndiItems(PLHashEntry *he, int i, void *arg) { char *str = (char *)he->key; IndiReflowCounter * counter = (IndiReflowCounter *)he->value; delete counter; free(str); return HT_ENUMERATE_REMOVE; } //------------------------------------------------------------------ void ReflowCountMgr::CleanUp() { if (nullptr != mCounts) { PL_HashTableEnumerateEntries(mCounts, RemoveItems, nullptr); PL_HashTableDestroy(mCounts); mCounts = nullptr; } if (nullptr != mIndiFrameCounts) { PL_HashTableEnumerateEntries(mIndiFrameCounts, RemoveIndiItems, nullptr); PL_HashTableDestroy(mIndiFrameCounts); mIndiFrameCounts = nullptr; } } //------------------------------------------------------------------ int ReflowCountMgr::DoSingleTotal(PLHashEntry *he, int i, void *arg) { char *str = (char *)he->key; ReflowCounter * counter = (ReflowCounter *)he->value; counter->DisplayTotals(str); return HT_ENUMERATE_NEXT; } //------------------------------------------------------------------ void ReflowCountMgr::DoGrandTotals() { if (nullptr != mCounts) { ReflowCounter * gTots = (ReflowCounter *)PL_HashTableLookup(mCounts, kGrandTotalsStr); if (gTots == nullptr) { gTots = new ReflowCounter(this); PL_HashTableAdd(mCounts, NS_strdup(kGrandTotalsStr), gTots); } else { gTots->ClearTotals(); } printf("\t\t\t\tTotal\n"); for (uint32_t i=0;i<78;i++) { printf("-"); } printf("\n"); PL_HashTableEnumerateEntries(mCounts, DoSingleTotal, this); } } static void RecurseIndiTotals(nsPresContext* aPresContext, PLHashTable * aHT, nsIFrame * aParentFrame, int32_t aLevel) { if (aParentFrame == nullptr) { return; } char key[KEY_BUF_SIZE_FOR_PTR]; SprintfLiteral(key, "%p", (void*)aParentFrame); IndiReflowCounter * counter = (IndiReflowCounter *)PL_HashTableLookup(aHT, key); if (counter) { counter->mHasBeenOutput = true; char * name = ToNewCString(counter->mName); for (int32_t i=0;i<aLevel;i++) printf(" "); printf("%s - %p [%d][", name, (void*)aParentFrame, counter->mCount); printf("%d", counter->mCounter.GetTotal()); printf("]\n"); free(name); } for (nsIFrame* child : aParentFrame->PrincipalChildList()) { RecurseIndiTotals(aPresContext, aHT, child, aLevel+1); } } //------------------------------------------------------------------ int ReflowCountMgr::DoSingleIndi(PLHashEntry *he, int i, void *arg) { IndiReflowCounter * counter = (IndiReflowCounter *)he->value; if (counter && !counter->mHasBeenOutput) { char * name = ToNewCString(counter->mName); printf("%s - %p [%d][", name, (void*)counter->mFrame, counter->mCount); printf("%d", counter->mCounter.GetTotal()); printf("]\n"); free(name); } return HT_ENUMERATE_NEXT; } //------------------------------------------------------------------ void ReflowCountMgr::DoIndiTotalsTree() { if (nullptr != mCounts) { printf("\n------------------------------------------------\n"); printf("-- Individual Frame Counts\n"); printf("------------------------------------------------\n"); if (mPresShell) { nsIFrame * rootFrame = mPresShell->FrameManager()->GetRootFrame(); RecurseIndiTotals(mPresContext, mIndiFrameCounts, rootFrame, 0); printf("------------------------------------------------\n"); printf("-- Individual Counts of Frames not in Root Tree\n"); printf("------------------------------------------------\n"); PL_HashTableEnumerateEntries(mIndiFrameCounts, DoSingleIndi, this); } } } //------------------------------------------------------------------ int ReflowCountMgr::DoSingleHTMLTotal(PLHashEntry *he, int i, void *arg) { char *str = (char *)he->key; ReflowCounter * counter = (ReflowCounter *)he->value; counter->DisplayHTMLTotals(str); return HT_ENUMERATE_NEXT; } //------------------------------------------------------------------ void ReflowCountMgr::DoGrandHTMLTotals() { if (nullptr != mCounts) { ReflowCounter * gTots = (ReflowCounter *)PL_HashTableLookup(mCounts, kGrandTotalsStr); if (gTots == nullptr) { gTots = new ReflowCounter(this); PL_HashTableAdd(mCounts, NS_strdup(kGrandTotalsStr), gTots); } else { gTots->ClearTotals(); } static const char * title[] = {"Class", "Reflows"}; fprintf(mFD, "<tr>"); for (uint32_t i=0; i < ArrayLength(title); i++) { fprintf(mFD, "<td><center><b>%s<b></center></td>", title[i]); } fprintf(mFD, "</tr>\n"); PL_HashTableEnumerateEntries(mCounts, DoSingleHTMLTotal, this); } } //------------------------------------ void ReflowCountMgr::DisplayTotals(const char * aStr) { #ifdef DEBUG_rods printf("%s\n", aStr?aStr:"No name"); #endif if (mDumpFrameCounts) { DoGrandTotals(); } if (mDumpFrameByFrameCounts) { DoIndiTotalsTree(); } } //------------------------------------ void ReflowCountMgr::DisplayHTMLTotals(const char * aStr) { #ifdef WIN32x // XXX NOT XP! char name[1024]; char * sptr = strrchr(aStr, '/'); if (sptr) { sptr++; strcpy(name, sptr); char * eptr = strrchr(name, '.'); if (eptr) { *eptr = 0; } strcat(name, "_stats.html"); } mFD = fopen(name, "w"); if (mFD) { fprintf(mFD, "<html><head><title>Reflow Stats</title></head><body>\n"); const char * title = aStr?aStr:"No name"; fprintf(mFD, "<center><b>%s</b><br><table border=1 style=\"background-color:#e0e0e0\">", title); DoGrandHTMLTotals(); fprintf(mFD, "</center></table>\n"); fprintf(mFD, "</body></html>\n"); fclose(mFD); mFD = nullptr; } #endif // not XP! } //------------------------------------------------------------------ int ReflowCountMgr::DoClearTotals(PLHashEntry *he, int i, void *arg) { ReflowCounter * counter = (ReflowCounter *)he->value; counter->ClearTotals(); return HT_ENUMERATE_NEXT; } //------------------------------------------------------------------ void ReflowCountMgr::ClearTotals() { PL_HashTableEnumerateEntries(mCounts, DoClearTotals, this); } //------------------------------------------------------------------ void ReflowCountMgr::ClearGrandTotals() { if (nullptr != mCounts) { ReflowCounter * gTots = (ReflowCounter *)PL_HashTableLookup(mCounts, kGrandTotalsStr); if (gTots == nullptr) { gTots = new ReflowCounter(this); PL_HashTableAdd(mCounts, NS_strdup(kGrandTotalsStr), gTots); } else { gTots->ClearTotals(); gTots->SetTotalsCache(); } } } //------------------------------------------------------------------ int ReflowCountMgr::DoDisplayDiffTotals(PLHashEntry *he, int i, void *arg) { bool cycledOnce = (arg != 0); char *str = (char *)he->key; ReflowCounter * counter = (ReflowCounter *)he->value; if (cycledOnce) { counter->CalcDiffInTotals(); counter->DisplayDiffTotals(str); } counter->SetTotalsCache(); return HT_ENUMERATE_NEXT; } //------------------------------------------------------------------ void ReflowCountMgr::DisplayDiffsInTotals() { if (mCycledOnce) { printf("Differences\n"); for (int32_t i=0;i<78;i++) { printf("-"); } printf("\n"); ClearGrandTotals(); } PL_HashTableEnumerateEntries(mCounts, DoDisplayDiffTotals, (void *)mCycledOnce); mCycledOnce = true; } #endif // MOZ_REFLOW_PERF nsIFrame* nsIPresShell::GetAbsoluteContainingBlock(nsIFrame *aFrame) { return FrameConstructor()->GetAbsoluteContainingBlock(aFrame, nsCSSFrameConstructor::ABS_POS); } #ifdef ACCESSIBILITY bool nsIPresShell::IsAccessibilityActive() { return GetAccService() != nullptr; } nsAccessibilityService* nsIPresShell::AccService() { return GetAccService(); } #endif void nsIPresShell::InitializeStatics() { MOZ_ASSERT(!sPointerCaptureList, "InitializeStatics called multiple times!"); sPointerCaptureList = new nsClassHashtable<nsUint32HashKey, PointerCaptureInfo>; sActivePointersIds = new nsClassHashtable<nsUint32HashKey, PointerInfo>; } void nsIPresShell::ReleaseStatics() { MOZ_ASSERT(sPointerCaptureList, "ReleaseStatics called without Initialize!"); delete sPointerCaptureList; sPointerCaptureList = nullptr; delete sActivePointersIds; sActivePointersIds = nullptr; } // Asks our docshell whether we're active. void PresShell::QueryIsActive() { nsCOMPtr<nsISupports> container = mPresContext->GetContainerWeak(); if (mDocument) { nsIDocument* displayDoc = mDocument->GetDisplayDocument(); if (displayDoc) { // Ok, we're an external resource document -- we need to use our display // document's docshell to determine "IsActive" status, since we lack // a container. MOZ_ASSERT(!container, "external resource doc shouldn't have its own container"); nsIPresShell* displayPresShell = displayDoc->GetShell(); if (displayPresShell) { container = displayPresShell->GetPresContext()->GetContainerWeak(); } } } nsCOMPtr<nsIDocShell> docshell(do_QueryInterface(container)); if (docshell) { bool isActive; nsresult rv = docshell->GetIsActive(&isActive); // Even though in theory the docshell here could be "Inactive and // Foreground", thus implying aIsHidden=false for SetIsActive(), // this is a newly created PresShell so we'd like to invalidate anyway // upon being made active to ensure that the contents get painted. if (NS_SUCCEEDED(rv)) SetIsActive(isActive); } } // Helper for propagating mIsActive changes to external resources static bool SetExternalResourceIsActive(nsIDocument* aDocument, void* aClosure) { nsIPresShell* shell = aDocument->GetShell(); if (shell) { shell->SetIsActive(*static_cast<bool*>(aClosure)); } return true; } static void SetPluginIsActive(nsISupports* aSupports, void* aClosure) { nsCOMPtr<nsIContent> content(do_QueryInterface(aSupports)); if (!content) { return; } nsIFrame *frame = content->GetPrimaryFrame(); nsIObjectFrame *objectFrame = do_QueryFrame(frame); if (objectFrame) { objectFrame->SetIsDocumentActive(*static_cast<bool*>(aClosure)); } } nsresult PresShell::SetIsActive(bool aIsActive) { NS_PRECONDITION(mDocument, "should only be called with a document"); mIsActive = aIsActive; nsPresContext* presContext = GetPresContext(); if (presContext && presContext->RefreshDriver()->GetPresContext() == presContext) { presContext->RefreshDriver()->SetThrottled(!mIsActive); } // Propagate state-change to my resource documents' PresShells mDocument->EnumerateExternalResources(SetExternalResourceIsActive, &aIsActive); mDocument->EnumerateActivityObservers(SetPluginIsActive, &aIsActive); nsresult rv = UpdateImageLockingState(); #ifdef ACCESSIBILITY if (aIsActive) { nsAccessibilityService* accService = AccService(); if (accService) { accService->PresShellActivated(this); } } #endif return rv; } /* * Determines the current image locking state. Called when one of the * dependent factors changes. */ nsresult PresShell::UpdateImageLockingState() { // We're locked if we're both thawed and active. bool locked = !mFrozen && mIsActive; nsresult rv = mDocument->ImageTracker()->SetLockingState(locked); if (locked) { // Request decodes for visible image frames; we want to start decoding as // quickly as possible when we get foregrounded to minimize flashing. for (auto iter = mApproximatelyVisibleFrames.Iter(); !iter.Done(); iter.Next()) { nsImageFrame* imageFrame = do_QueryFrame(iter.Get()->GetKey()); if (imageFrame) { imageFrame->MaybeDecodeForPredictedSize(); } } } return rv; } PresShell* PresShell::GetRootPresShell() { if (mPresContext) { nsPresContext* rootPresContext = mPresContext->GetRootPresContext(); if (rootPresContext) { return static_cast<PresShell*>(rootPresContext->PresShell()); } } return nullptr; } void PresShell::AddSizeOfIncludingThis(MallocSizeOf aMallocSizeOf, nsArenaMemoryStats *aArenaObjectsSize, size_t *aPresShellSize, size_t *aStyleSetsSize, size_t *aTextRunsSize, size_t *aPresContextSize) { mFrameArena.AddSizeOfExcludingThis(aMallocSizeOf, aArenaObjectsSize); *aPresShellSize += aMallocSizeOf(this); if (mCaret) { *aPresShellSize += mCaret->SizeOfIncludingThis(aMallocSizeOf); } *aPresShellSize += mApproximatelyVisibleFrames.ShallowSizeOfExcludingThis(aMallocSizeOf); *aPresShellSize += mFramesToDirty.ShallowSizeOfExcludingThis(aMallocSizeOf); *aPresShellSize += aArenaObjectsSize->mOther; if (nsStyleSet* styleSet = StyleSet()->GetAsGecko()) { *aStyleSetsSize += styleSet->SizeOfIncludingThis(aMallocSizeOf); } else { NS_WARNING("ServoStyleSets do not support memory measurements yet"); } *aTextRunsSize += SizeOfTextRuns(aMallocSizeOf); *aPresContextSize += mPresContext->SizeOfIncludingThis(aMallocSizeOf); } size_t PresShell::SizeOfTextRuns(MallocSizeOf aMallocSizeOf) const { nsIFrame* rootFrame = mFrameConstructor->GetRootFrame(); if (!rootFrame) { return 0; } // clear the TEXT_RUN_MEMORY_ACCOUNTED flags nsLayoutUtils::SizeOfTextRunsForFrames(rootFrame, nullptr, /* clear = */true); // collect the total memory in use for textruns return nsLayoutUtils::SizeOfTextRunsForFrames(rootFrame, aMallocSizeOf, /* clear = */false); } void nsIPresShell::MarkFixedFramesForReflow(IntrinsicDirty aIntrinsicDirty) { nsIFrame* rootFrame = mFrameConstructor->GetRootFrame(); if (rootFrame) { const nsFrameList& childList = rootFrame->GetChildList(nsIFrame::kFixedList); for (nsIFrame* childFrame : childList) { FrameNeedsReflow(childFrame, aIntrinsicDirty, NS_FRAME_IS_DIRTY); } } } void nsIPresShell::SetScrollPositionClampingScrollPortSize(nscoord aWidth, nscoord aHeight) { if (!mScrollPositionClampingScrollPortSizeSet || mScrollPositionClampingScrollPortSize.width != aWidth || mScrollPositionClampingScrollPortSize.height != aHeight) { mScrollPositionClampingScrollPortSizeSet = true; mScrollPositionClampingScrollPortSize.width = aWidth; mScrollPositionClampingScrollPortSize.height = aHeight; if (nsIScrollableFrame* rootScrollFrame = GetRootScrollFrameAsScrollable()) { rootScrollFrame->MarkScrollbarsDirtyForReflow(); } MarkFixedFramesForReflow(nsIPresShell::eResize); } } void PresShell::SetupFontInflation() { mFontSizeInflationEmPerLine = nsLayoutUtils::FontSizeInflationEmPerLine(); mFontSizeInflationMinTwips = nsLayoutUtils::FontSizeInflationMinTwips(); mFontSizeInflationLineThreshold = nsLayoutUtils::FontSizeInflationLineThreshold(); mFontSizeInflationForceEnabled = nsLayoutUtils::FontSizeInflationForceEnabled(); mFontSizeInflationDisabledInMasterProcess = nsLayoutUtils::FontSizeInflationDisabledInMasterProcess(); NotifyFontSizeInflationEnabledIsDirty(); } void nsIPresShell::RecomputeFontSizeInflationEnabled() { mFontSizeInflationEnabledIsDirty = false; MOZ_ASSERT(mPresContext, "our pres context should not be null"); if ((FontSizeInflationEmPerLine() == 0 && FontSizeInflationMinTwips() == 0) || mPresContext->IsChrome()) { mFontSizeInflationEnabled = false; return; } // Force-enabling font inflation always trumps the heuristics here. if (!FontSizeInflationForceEnabled()) { if (TabChild* tab = TabChild::GetFrom(this)) { // We're in a child process. Cancel inflation if we're not // async-pan zoomed. if (!tab->AsyncPanZoomEnabled()) { mFontSizeInflationEnabled = false; return; } } else if (XRE_IsParentProcess()) { // We're in the master process. Cancel inflation if it's been // explicitly disabled. if (FontSizeInflationDisabledInMasterProcess()) { mFontSizeInflationEnabled = false; return; } } } // XXXjwir3: // See bug 706918, comment 23 for more information on this particular section // of the code. We're using "screen size" in place of the size of the content // area, because on mobile, these are close or equal. This will work for our // purposes (bug 706198), but it will need to be changed in the future to be // more correct when we bring the rest of the viewport code into platform. // We actually want the size of the content area, in the event that we don't // have any metadata about the width and/or height. On mobile, the screen size // and the size of the content area are very close, or the same value. // In XUL fennec, the content area is the size of the <browser> widget, but // in native fennec, the content area is the size of the Gecko LayerView // object. // TODO: // Once bug 716575 has been resolved, this code should be changed so that it // does the right thing on all platforms. nsresult rv; nsCOMPtr<nsIScreenManager> screenMgr = do_GetService("@mozilla.org/gfx/screenmanager;1", &rv); if (!NS_SUCCEEDED(rv)) { mFontSizeInflationEnabled = false; return; } nsCOMPtr<nsIScreen> screen; screenMgr->GetPrimaryScreen(getter_AddRefs(screen)); if (screen) { int32_t screenLeft, screenTop, screenWidth, screenHeight; screen->GetRect(&screenLeft, &screenTop, &screenWidth, &screenHeight); nsViewportInfo vInf = GetDocument()->GetViewportInfo(ScreenIntSize(screenWidth, screenHeight)); if (vInf.GetDefaultZoom() >= CSSToScreenScale(1.0f) || vInf.IsAutoSizeEnabled()) { mFontSizeInflationEnabled = false; return; } } mFontSizeInflationEnabled = true; } bool nsIPresShell::FontSizeInflationEnabled() { if (mFontSizeInflationEnabledIsDirty) { RecomputeFontSizeInflationEnabled(); } return mFontSizeInflationEnabled; } void PresShell::PausePainting() { if (GetPresContext()->RefreshDriver()->GetPresContext() != GetPresContext()) return; mPaintingIsFrozen = true; GetPresContext()->RefreshDriver()->Freeze(); } void PresShell::ResumePainting() { if (GetPresContext()->RefreshDriver()->GetPresContext() != GetPresContext()) return; mPaintingIsFrozen = false; GetPresContext()->RefreshDriver()->Thaw(); } void nsIPresShell::SyncWindowProperties(nsView* aView) { nsIFrame* frame = aView->GetFrame(); if (frame && mPresContext) { // CreateReferenceRenderingContext can return nullptr nsRenderingContext rcx(CreateReferenceRenderingContext()); nsContainerFrame::SyncWindowProperties(mPresContext, frame, aView, &rcx, 0); } } nsresult nsIPresShell::HasRuleProcessorUsedByMultipleStyleSets(uint32_t aSheetType, bool* aRetVal) { SheetType type; switch (aSheetType) { case nsIStyleSheetService::AGENT_SHEET: type = SheetType::Agent; break; case nsIStyleSheetService::USER_SHEET: type = SheetType::User; break; case nsIStyleSheetService::AUTHOR_SHEET: type = SheetType::Doc; break; default: MOZ_ASSERT(false, "unexpected aSheetType value"); return NS_ERROR_ILLEGAL_VALUE; } *aRetVal = false; if (nsStyleSet* styleSet = mStyleSet->GetAsGecko()) { // ServoStyleSets do not have rule processors. *aRetVal = styleSet->HasRuleProcessorUsedByMultipleStyleSets(type); } return NS_OK; }