forked from mirrors/gecko-dev
		
	 d2f3dc13ec
			
		
	
	
		d2f3dc13ec
		
	
	
	
	
		
			
			It's not only inefficient, but also prone to buggyness. Since styles may not be up-to-date when it happens. Post a reconstruct instead, which ensures a style flush happens before running frame construction. MozReview-Commit-ID: DrakHsJv5fY Signed-off-by: Emilio Cobos Álvarez <emilio@crisal.io> --HG-- extra : rebase_source : 11900af908654336cc2391ab9480542c5474e38f
		
			
				
	
	
		
			11225 lines
		
	
	
	
		
			365 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			11225 lines
		
	
	
	
		
			365 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- 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/PresShell.h"
 | |
| 
 | |
| #include "mozilla/ArrayUtils.h"
 | |
| #include "mozilla/Attributes.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/Logging.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 "nsPresContext.h"
 | |
| #include "nsIContent.h"
 | |
| #include "nsIContentIterator.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 "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/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"
 | |
| #include "nsXULElement.h"
 | |
| #include "mozilla/dom/BoxObject.h"
 | |
| #endif // MOZ_XUL
 | |
| 
 | |
| #include "mozilla/layers/CompositorBridgeChild.h"
 | |
| #include "ClientLayerManager.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/dom/ScriptSettings.h"
 | |
| #include "mozilla/ErrorResult.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/RestyleManager.h"
 | |
| #include "mozilla/RestyleManagerInlines.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"
 | |
| #include "nsIDocShellTreeOwner.h"
 | |
| 
 | |
| #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;
 | |
| typedef FrameMetrics::ViewID ViewID;
 | |
| 
 | |
| 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(ObservingLayoutFlushes() ==                                    \
 | |
|                  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(FlushType::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
 | |
| nsIPresShell::SetVerifyReflowEnable(bool aEnabled)
 | |
| {
 | |
|   gVerifyReflowEnabled = aEnabled;
 | |
| }
 | |
| 
 | |
| /* virtual */ void
 | |
| nsIPresShell::AddAutoWeakFrameExternal(AutoWeakFrame* aWeakFrame)
 | |
| {
 | |
|   AddAutoWeakFrameInternal(aWeakFrame);
 | |
| }
 | |
| 
 | |
| void
 | |
| nsIPresShell::AddAutoWeakFrameInternal(AutoWeakFrame* aWeakFrame)
 | |
| {
 | |
|   if (aWeakFrame->GetFrame()) {
 | |
|     aWeakFrame->GetFrame()->AddStateBits(NS_FRAME_EXTERNAL_REFERENCE);
 | |
|   }
 | |
|   aWeakFrame->SetPreviousWeakFrame(mAutoWeakFrames);
 | |
|   mAutoWeakFrames = aWeakFrame;
 | |
| }
 | |
| 
 | |
| /* virtual */ void
 | |
| nsIPresShell::AddWeakFrameExternal(WeakFrame* aWeakFrame)
 | |
| {
 | |
|   AddWeakFrameInternal(aWeakFrame);
 | |
| }
 | |
| 
 | |
| void
 | |
| nsIPresShell::AddWeakFrameInternal(WeakFrame* aWeakFrame)
 | |
| {
 | |
|   if (aWeakFrame->GetFrame()) {
 | |
|     aWeakFrame->GetFrame()->AddStateBits(NS_FRAME_EXTERNAL_REFERENCE);
 | |
|   }
 | |
|   MOZ_ASSERT(!mWeakFrames.GetEntry(aWeakFrame));
 | |
|   mWeakFrames.PutEntry(aWeakFrame);
 | |
| }
 | |
| 
 | |
| /* virtual */ void
 | |
| nsIPresShell::RemoveAutoWeakFrameExternal(AutoWeakFrame* aWeakFrame)
 | |
| {
 | |
|   RemoveAutoWeakFrameInternal(aWeakFrame);
 | |
| }
 | |
| 
 | |
| void
 | |
| nsIPresShell::RemoveAutoWeakFrameInternal(AutoWeakFrame* aWeakFrame)
 | |
| {
 | |
|   if (mAutoWeakFrames == aWeakFrame) {
 | |
|     mAutoWeakFrames = aWeakFrame->GetPreviousWeakFrame();
 | |
|     return;
 | |
|   }
 | |
|   AutoWeakFrame* nextWeak = mAutoWeakFrames;
 | |
|   while (nextWeak && nextWeak->GetPreviousWeakFrame() != aWeakFrame) {
 | |
|     nextWeak = nextWeak->GetPreviousWeakFrame();
 | |
|   }
 | |
|   if (nextWeak) {
 | |
|     nextWeak->SetPreviousWeakFrame(aWeakFrame->GetPreviousWeakFrame());
 | |
|   }
 | |
| }
 | |
| 
 | |
| /* virtual */ void
 | |
| nsIPresShell::RemoveWeakFrameExternal(WeakFrame* aWeakFrame)
 | |
| {
 | |
|   RemoveWeakFrameInternal(aWeakFrame);
 | |
| }
 | |
| 
 | |
| void
 | |
| nsIPresShell::RemoveWeakFrameInternal(WeakFrame* aWeakFrame)
 | |
| {
 | |
|   MOZ_ASSERT(mWeakFrames.GetEntry(aWeakFrame));
 | |
|   mWeakFrames.RemoveEntry(aWeakFrame);
 | |
| }
 | |
| 
 | |
| 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
 | |
| 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;
 | |
| }
 | |
| 
 | |
| nsIPresShell::nsIPresShell()
 | |
|     : mFrameConstructor(nullptr)
 | |
|     , mViewManager(nullptr)
 | |
|     , mFrameManager(nullptr)
 | |
| #ifdef ACCESSIBILITY
 | |
|     , mDocAccessible(nullptr)
 | |
| #endif
 | |
| #ifdef DEBUG
 | |
|     , mDrawEventTargetFrame(nullptr)
 | |
| #endif
 | |
|     , mPaintCount(0)
 | |
|     , mAutoWeakFrames(nullptr)
 | |
|     , mCanvasBackgroundColor(NS_RGBA(0,0,0,0))
 | |
|     , mSelectionFlags(0)
 | |
|     , mRenderFlags(0)
 | |
|     , mStylesHaveChanged(false)
 | |
|     , mDidInitialize(false)
 | |
|     , mIsDestroying(false)
 | |
|     , mIsReflowing(false)
 | |
|     , mPaintingSuppressed(false)
 | |
|     , mIsThemeSupportDisabled(false)
 | |
|     , mIsActive(false)
 | |
|     , mFrozen(false)
 | |
|     , mIsFirstPaint(false)
 | |
|     , mObservesMutationsForPrint(false)
 | |
|     , mSuppressInterruptibleReflows(false)
 | |
|     , mScrollPositionClampingScrollPortSizeSet(false)
 | |
|     , mNeedLayoutFlush(true)
 | |
|     , mNeedStyleFlush(true)
 | |
|     , mObservingStyleFlushes(false)
 | |
|     , mObservingLayoutFlushes(false)
 | |
|     , mNeedThrottledAnimationFlush(true)
 | |
|     , mPresShellId(0)
 | |
|     , mFontSizeInflationEmPerLine(0)
 | |
|     , mFontSizeInflationMinTwips(0)
 | |
|     , mFontSizeInflationLineThreshold(0)
 | |
|     , mFontSizeInflationForceEnabled(false)
 | |
|     , mFontSizeInflationDisabledInMasterProcess(false)
 | |
|     , mFontSizeInflationEnabled(false)
 | |
|     , mFontSizeInflationEnabledIsDirty(false)
 | |
|     , mPaintingIsFrozen(false)
 | |
|     , mIsNeverPainting(false)
 | |
|     , mInFlush(false)
 | |
|   {}
 | |
| 
 | |
| PresShell::PresShell()
 | |
|   : mCaretEnabled(false)
 | |
| #ifdef DEBUG
 | |
|   , mInVerifyReflow(false)
 | |
|   , mCurrentReflowRoot(nullptr)
 | |
|   , mUpdateCount(0)
 | |
| #endif
 | |
| #ifdef MOZ_REFLOW_PERF
 | |
|   , mReflowCountMgr(nullptr)
 | |
| #endif
 | |
|   , mMouseLocation(NS_UNCONSTRAINEDSIZE, NS_UNCONSTRAINEDSIZE)
 | |
|   , mCurrentEventFrame(nullptr)
 | |
|   , mFirstCallbackEventRequest(nullptr)
 | |
|   , mLastCallbackEventRequest(nullptr)
 | |
|   , mLastReflowStart(0.0)
 | |
|   , mLastAnchorScrollPositionY(0)
 | |
|   , mChangeNestCount(0)
 | |
|   , mDocumentLoading(false)
 | |
|   , mIgnoreFrameDestruction(false)
 | |
|   , mHaveShutDown(false)
 | |
|   , mLastRootReflowHadUnconstrainedBSize(false)
 | |
|   , mNoDelayedMouseEvents(false)
 | |
|   , mNoDelayedKeyEvents(false)
 | |
|   , mIsDocumentGone(false)
 | |
|   , mShouldUnsuppressPainting(false)
 | |
|   , mAsyncResizeTimerIsActive(false)
 | |
|   , mInResize(false)
 | |
|   , mApproximateFrameVisibilityVisited(false)
 | |
|   , mNextPaintCompressed(false)
 | |
|   , mHasCSSBackgroundColor(false)
 | |
|   , mScaleToResolution(false)
 | |
|   , mIsLastChromeOnlyEscapeKeyConsumed(false)
 | |
|   , mHasReceivedPaintMessage(false)
 | |
| {
 | |
| #ifdef MOZ_REFLOW_PERF
 | |
|   mReflowCountMgr = new ReflowCountMgr();
 | |
|   mReflowCountMgr->SetPresContext(mPresContext);
 | |
|   mReflowCountMgr->SetPresShell(this);
 | |
| #endif
 | |
|   mLastOSWake = 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;
 | |
| 
 | |
|   // mDocument is now set.  It might have a display document whose "need layout/
 | |
|   // style" flush flags are not set, but ours will be set.  To keep these
 | |
|   // consistent, call the flag setting functions to propagate those flags up
 | |
|   // to the display document.
 | |
|   SetNeedLayoutFlush();
 | |
|   SetNeedStyleFlush();
 | |
| 
 | |
|   // 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;
 | |
|   mPresContext->AttachShell(this, aStyleSet->BackendType());
 | |
| 
 | |
|   // 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);
 | |
|   }
 | |
| 
 | |
|   if (nsStyleSheetService* ss = nsStyleSheetService::GetInstance()) {
 | |
|     ss->RegisterPresShell(this);
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     nsCOMPtr<nsIObserverService> os = mozilla::services::GetObserverService();
 | |
|     if (os) {
 | |
| #ifdef MOZ_XUL
 | |
|       os->AddObserver(this, "chrome-flush-skin-caches", false);
 | |
| #endif
 | |
|       os->AddObserver(this, "memory-pressure", false);
 | |
|       os->AddObserver(this, NS_WIDGET_WAKE_OBSERVER_TOPIC, 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);
 | |
|   }
 | |
| 
 | |
|   if (nsStyleSheetService* ss = nsStyleSheetService::GetInstance()) {
 | |
|     ss->UnregisterPresShell(this);
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     nsCOMPtr<nsIObserverService> os = mozilla::services::GetObserverService();
 | |
|     if (os) {
 | |
| #ifdef MOZ_XUL
 | |
|       os->RemoveObserver(this, "chrome-flush-skin-caches");
 | |
| #endif
 | |
|       os->RemoveObserver(this, "memory-pressure");
 | |
|       os->RemoveObserver(this, NS_WIDGET_WAKE_OBSERVER_TOPIC);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // 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 (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(!mAutoWeakFrames && mWeakFrames.IsEmpty(),
 | |
|                        "Weak frames alive after destroying FrameManager");
 | |
|   while (mAutoWeakFrames) {
 | |
|     mAutoWeakFrames->Clear(this);
 | |
|   }
 | |
|   nsTArray<WeakFrame*> toRemove(mWeakFrames.Count());
 | |
|   for (auto iter = mWeakFrames.Iter(); !iter.Done(); iter.Next()) {
 | |
|     toRemove.AppendElement(iter.Get()->GetKey());
 | |
|   }
 | |
|   for (WeakFrame* weakFrame : toRemove) {
 | |
|     weakFrame->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();
 | |
| }
 | |
| 
 | |
| 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(StyleSheet* aSheet)
 | |
| {
 | |
|   // 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(mStyleSet->BackendType());
 | |
|   // 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(StyleSheet* aSheet)
 | |
| {
 | |
|   // Make sure this does what nsDocumentViewer::CreateStyleSet does
 | |
|   // wrt ordering.
 | |
|   mStyleSet->AppendStyleSheet(SheetType::Agent, aSheet);
 | |
|   RestyleForCSSRuleChanges();
 | |
| }
 | |
| 
 | |
| void
 | |
| PresShell::AddAuthorSheet(StyleSheet* aSheet)
 | |
| {
 | |
|   // 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, aSheet, firstAuthorSheet);
 | |
|   } else {
 | |
|     mStyleSet->AppendStyleSheet(SheetType::Doc, aSheet);
 | |
|   }
 | |
| 
 | |
|   RestyleForCSSRuleChanges();
 | |
| }
 | |
| 
 | |
| void
 | |
| PresShell::RemoveSheet(SheetType aType, StyleSheet* aSheet)
 | |
| {
 | |
|   mStyleSet->RemoveStyleSheet(aType, aSheet);
 | |
|   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
 | |
| 
 | |
| 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));
 | |
| 
 | |
|   // 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();
 | |
|     }
 | |
| 
 | |
|     // nsAutoScriptBlocker 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
 | |
|     mDocument->BindingManager()->ProcessAttachedQueue();
 | |
| 
 | |
|     // Constructors may have killed us too
 | |
|     NS_ENSURE_STATE(!mHaveShutDown);
 | |
| 
 | |
|     // Now flush out pending restyles before we actually reflow, in
 | |
|     // case XBL constructors changed styles somewhere.
 | |
|     {
 | |
|       nsAutoScriptBlocker scriptBlocker;
 | |
|       mPresContext->RestyleManager()->ProcessPendingRestyles();
 | |
|     }
 | |
| 
 | |
|     // And that might have run _more_ XBL constructors
 | |
|     NS_ENSURE_STATE(!mHaveShutDown);
 | |
|   }
 | |
| 
 | |
|   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(mObservingLayoutFlushes, "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->SetTarget(
 | |
|           mDocument->EventTargetFor(TaskCategory::Other));
 | |
|       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;
 | |
|   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(FlushType::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->SetTarget(
 | |
|             mDocument->EventTargetFor(TaskCategory::Other));
 | |
|         mAsyncResizeEventTimer->InitWithNamedFuncCallback(AsyncResizeEventCallback,
 | |
|                                                           this, 15,
 | |
|                                                           nsITimer::TYPE_ONE_SHOT,
 | |
|                                                           "AsyncResizeEventCallback");
 | |
|       }
 | |
|     } else {
 | |
|       RefPtr<nsRunnableMethod<PresShell>> event =
 | |
|         NewRunnableMethod(this, &PresShell::FireResizeEvent);
 | |
|       nsresult rv = mDocument->Dispatch("PresShell::FireResizeEvent",
 | |
|                                         TaskCategory::Other,
 | |
|                                         do_AddRef(event));
 | |
|       if (NS_SUCCEEDED(rv)) {
 | |
|         mResizeEvent = event;
 | |
|         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);
 | |
|           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 (mObservingLayoutFlushes) {
 | |
|     GetPresContext()->RefreshDriver()->RemoveLayoutFlushObserver(this);
 | |
|     mObservingLayoutFlushes = 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(FlushType::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;
 | |
| }
 | |
| 
 | |
| 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);
 | |
| 
 | |
|   AutoWeakFrame* weakFrame = mAutoWeakFrames;
 | |
|   while (weakFrame) {
 | |
|     AutoWeakFrame* prev = weakFrame->GetPreviousWeakFrame();
 | |
|     if (weakFrame->GetFrame() == aFrame) {
 | |
|       // This removes weakFrame from mAutoWeakFrames.
 | |
|       weakFrame->Clear(this);
 | |
|     }
 | |
|     weakFrame = prev;
 | |
|   }
 | |
| 
 | |
|   AutoTArray<WeakFrame*, 4> toRemove;
 | |
|   for (auto iter = mWeakFrames.Iter(); !iter.Done(); iter.Next()) {
 | |
|     WeakFrame* weakFrame = iter.Get()->GetKey();
 | |
|     if (weakFrame->GetFrame() == aFrame) {
 | |
|       toRemove.AppendElement(weakFrame);
 | |
|     }
 | |
|   }
 | |
|   for (WeakFrame* weakFrame : toRemove) {
 | |
|     weakFrame->Clear(this);
 | |
|   }
 | |
| }
 | |
| 
 | |
| 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.
 | |
|   if (nsIPresShell* shell = composedDoc->GetShell()) {
 | |
|     shell->SetNeedLayoutFlush();
 | |
|   }
 | |
|   composedDoc->FlushPendingNotifications(FlushType::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;
 | |
| }
 | |
| 
 | |
| void
 | |
| PresShell::ScheduleViewManagerFlush(PaintType aType)
 | |
| {
 | |
|   if (aType == PAINT_DELAYED_COMPRESS) {
 | |
|     // Delay paint for 1 second.
 | |
|     static const uint32_t kPaintDelayPeriod = 1000;
 | |
|     if (!mDelayedPaintTimer) {
 | |
|       nsTimerCallbackFunc
 | |
|         PaintTimerCallBack = [](nsITimer* aTimer, void* aClosure) {
 | |
|           // The passed-in PresShell is always alive here. Because if PresShell
 | |
|           // died, mDelayedPaintTimer->Cancel() would be called during the
 | |
|           // destruction and this callback would never be invoked.
 | |
|           auto self = static_cast<PresShell*>(aClosure);
 | |
|           self->SetNextPaintCompressed();
 | |
|           self->ScheduleViewManagerFlush();
 | |
|       };
 | |
| 
 | |
|       mDelayedPaintTimer = do_CreateInstance(NS_TIMER_CONTRACTID);
 | |
|       mDelayedPaintTimer->SetTarget(
 | |
|           mDocument->EventTargetFor(TaskCategory::Other));
 | |
|       mDelayedPaintTimer->InitWithNamedFuncCallback(PaintTimerCallBack,
 | |
|                                                     this,
 | |
|                                                     kPaintDelayPeriod,
 | |
|                                                     nsITimer::TYPE_ONE_SHOT,
 | |
|                                                     "PaintTimerCallBack");
 | |
|     }
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   nsPresContext* presContext = GetPresContext();
 | |
|   if (presContext) {
 | |
|     presContext->RefreshDriver()->ScheduleViewManagerFlush();
 | |
|   }
 | |
|   SetNeedLayoutFlush();
 | |
| }
 | |
| 
 | |
| bool
 | |
| FlushLayoutRecursive(nsIDocument* aDocument,
 | |
|                      void* aData = nullptr)
 | |
| {
 | |
|   MOZ_ASSERT(!aData);
 | |
|   nsCOMPtr<nsIDocument> kungFuDeathGrip(aDocument);
 | |
|   aDocument->EnumerateSubDocuments(FlushLayoutRecursive, nullptr);
 | |
|   aDocument->FlushPendingNotifications(FlushType::Layout);
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void
 | |
| PresShell::DispatchSynthMouseMove(WidgetGUIEvent* aEvent,
 | |
|                                   bool aFlushOnHoverChange)
 | |
| {
 | |
|   GeckoProfilerTracingRAII tracer("Paint", "DispatchSynthMouseMove");
 | |
|   RestyleManager* 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;
 | |
|        }
 | |
|      }
 | |
|    }
 | |
| 
 | |
|    FlushType flushType =
 | |
|      aInterruptible ? FlushType::InterruptibleLayout : FlushType::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::DoFlushPendingNotifications(FlushType aType)
 | |
| {
 | |
|   // by default, flush animations if aType >= FlushType::Style
 | |
|   mozilla::ChangesToFlush flush(aType, aType >= FlushType::Style);
 | |
|   FlushPendingNotifications(flush);
 | |
| }
 | |
| 
 | |
| void
 | |
| PresShell::DoFlushPendingNotifications(mozilla::ChangesToFlush aFlush)
 | |
| {
 | |
|   // Per our API contract, hold a strong ref to ourselves until we return.
 | |
|   nsCOMPtr<nsIPresShell> kungFuDeathGrip = this;
 | |
| 
 | |
|   /**
 | |
|    * 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 shell.
 | |
|    */
 | |
|   FlushType flushType = aFlush.mFlushType;
 | |
| 
 | |
|   MOZ_ASSERT(NeedFlush(flushType), "Why did we get called?");
 | |
| 
 | |
| #ifdef MOZ_GECKO_PROFILER
 | |
|   static const EnumeratedArray<FlushType,
 | |
|                                FlushType::Count,
 | |
|                                const char*> flushTypeNames = {
 | |
|     "",
 | |
|     "Content",
 | |
|     "ContentAndNotify",
 | |
|     "Style",
 | |
|     "InterruptibleLayout",
 | |
|     "Layout",
 | |
|     "Display"
 | |
|   };
 | |
| 
 | |
|   PROFILER_LABEL_DYNAMIC("PresShell", "Flush",
 | |
|     js::ProfileEntry::Category::GRAPHICS,
 | |
|     flushTypeNames[flushType]);
 | |
| #endif
 | |
| 
 | |
| #ifdef ACCESSIBILITY
 | |
| #ifdef DEBUG
 | |
|   nsAccessibilityService* accService = GetAccService();
 | |
|   if (accService) {
 | |
|     NS_ASSERTION(!accService->IsProcessingRefreshDriverNotification(),
 | |
|                  "Flush during accessible tree update!");
 | |
|   }
 | |
| #endif
 | |
| #endif
 | |
| 
 | |
|   NS_ASSERTION(flushType >= FlushType::Frames, "Why did we get called?");
 | |
| 
 | |
|   mNeedStyleFlush = false;
 | |
|   mNeedThrottledAnimationFlush =
 | |
|     mNeedThrottledAnimationFlush && !aFlush.mFlushAnimations;
 | |
|   mNeedLayoutFlush =
 | |
|     mNeedLayoutFlush && (flushType < FlushType::InterruptibleLayout);
 | |
| 
 | |
|   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;
 | |
|   if (isSafeToFlush && viewManager) {
 | |
|     // Record that we are in a flush, so that our optimization in
 | |
|     // nsDocument::FlushPendingNotifications doesn't skip any re-entrant
 | |
|     // calls to us.  Otherwise, we might miss some needed flushes, since
 | |
|     // we clear mNeedStyleFlush / mNeedLayoutFlush here at the top of
 | |
|     // the function but we might not have done the work yet.
 | |
|     AutoRestore<bool> guard(mInFlush);
 | |
|     mInFlush = true;
 | |
| 
 | |
|     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 >= FlushType::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(FlushType::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
 | |
|                         ? FlushType::Layout
 | |
|                         : FlushType::InterruptibleLayout) &&
 | |
|         !mIsDestroying) {
 | |
|       didLayoutFlush = true;
 | |
|       mFrameConstructor->RecalcQuotesAndCounters();
 | |
|       viewManager->FlushDelayedResize(true);
 | |
|       if (ProcessReflowCommands(flushType < FlushType::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 >= FlushType::Layout) {
 | |
|       if (!mIsDestroying) {
 | |
|         viewManager->UpdateWidgetGeometry();
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!didStyleFlush && flushType >= FlushType::Style && !mIsDestroying) {
 | |
|     SetNeedStyleFlush();
 | |
|     if (aFlush.mFlushAnimations) {
 | |
|       SetNeedThrottledAnimationFlush();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!didLayoutFlush && flushType >= FlushType::InterruptibleLayout &&
 | |
|       !mIsDestroying) {
 | |
|     // We suppressed this flush either due to it not being safe to flush,
 | |
|     // or due to mSuppressInterruptibleReflows.  Either way, the
 | |
|     // mNeedLayoutFlush flag needs to be re-set.
 | |
|     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* geckoSet = mStyleSet->GetAsGecko();
 | |
|   if (!geckoSet) {
 | |
|     // 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.");
 | |
| 
 | |
|   } else if (mDidInitialize &&
 | |
|              geckoSet->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();
 | |
| }
 | |
| 
 | |
| void
 | |
| PresShell::ReconstructFrames()
 | |
| {
 | |
|   NS_PRECONDITION(!mFrameConstructor->GetRootFrame() || mDidInitialize,
 | |
|                   "Must not have root frame before initial reflow");
 | |
|   if (!mDidInitialize || mIsDestroying) {
 | |
|     // Nothing to do here
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   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(FlushType::ContentAndNotify);
 | |
| 
 | |
|   if (mIsDestroying) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   nsAutoCauseReflowNotifier crNotifier(this);
 | |
|   mFrameConstructor->BeginUpdate();
 | |
|   mFrameConstructor->ReconstructDocElementHierarchy();
 | |
|   VERIFY_STYLE_TREE;
 | |
|   mFrameConstructor->EndUpdate();
 | |
| }
 | |
| 
 | |
| 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();
 | |
|   }
 | |
| 
 | |
|   // Tell Servo that the contents of style sheets have changed.
 | |
|   //
 | |
|   // NB: It's important to do so before bailing out.
 | |
|   //
 | |
|   // Even if we have no frames, we can end up styling those when creating
 | |
|   // them, and it's important for Servo to know that it needs to use the
 | |
|   // correct styles.
 | |
|   // We don't do this notification if author styles are disabled, because
 | |
|   // the ServoStyleSet has already taken care of it.
 | |
|   if (mStyleSet->IsServo() && !mStyleSet->AsServo()->GetAuthorStyleDisabled()) {
 | |
|     mStyleSet->AsServo()->NoteStyleSheetsChanged();
 | |
|   }
 | |
| 
 | |
|   Element* root = mDocument->GetRootElement();
 | |
|   if (!mDidInitialize) {
 | |
|     // Nothing to do here, since we have no frames yet
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (!root) {
 | |
|     // No content to restyle
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   RestyleManager* 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>");
 | |
|   }
 | |
| 
 | |
|   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 or WebRenderLayerManagers in content processes
 | |
|       // don't support taking snapshots.
 | |
|       if (layerManager &&
 | |
|           (!layerManager->AsKnowsCompositor() ||
 | |
|            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);
 | |
|           DisplayItemClipChain newClipChain = { newClip, i->GetActiveScrolledRoot(), nullptr };
 | |
|           i->IntersectClip(aBuilder, &newClipChain);
 | |
|           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, FlushType::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,
 | |
|                                       const 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(const 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) {
 | |
|     bool ignoreDisplayPort = false;
 | |
|     if (nsLayoutUtils::IsMissingDisplayPortBaseRect(aFrame->GetContent())) {
 | |
|       // We can properly set the base rect for root scroll frames on top level
 | |
|       // and root content documents. Otherwise the base rect we compute might
 | |
|       // be way too big without the limiting that
 | |
|       // ScrollFrameHelper::DecideScrollableLayer does, so we just ignore the
 | |
|       // displayport in that case.
 | |
|       nsPresContext* pc = aFrame->PresContext();
 | |
|       if (scrollFrame->IsRootScrollFrameOfDocument() &&
 | |
|           (pc->IsRootContentDocument() || !pc->GetParentPresContext())) {
 | |
|         nsRect baseRect =
 | |
|           nsRect(nsPoint(0, 0), nsLayoutUtils::CalculateCompositionSizeForFrame(aFrame));
 | |
|         nsLayoutUtils::SetDisplayPortBase(aFrame->GetContent(), baseRect);
 | |
|       } else {
 | |
|         ignoreDisplayPort = true;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     scrollFrame->NotifyApproximateFrameVisibilityUpdate(ignoreDisplayPort);
 | |
| 
 | |
|     nsRect displayPort;
 | |
|     bool usingDisplayport = !ignoreDisplayPort &&
 | |
|       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>> event =
 | |
|     NewRunnableMethod(this, &PresShell::UpdateApproximateFrameVisibility);
 | |
|   nsresult rv =
 | |
|     mDocument->Dispatch("PresShell::UpdateApproximateFrameVisibility",
 | |
|                         TaskCategory::Other,
 | |
|                         do_AddRef(event));
 | |
| 
 | |
|   if (NS_SUCCEEDED(rv)) {
 | |
|     mUpdateApproximateFrameVisibilityEvent = event;
 | |
|   }
 | |
| }
 | |
| 
 | |
| 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()
 | |
|   {
 | |
|     if (mFlags & nsIPresShell::PAINT_COMPOSITE) {
 | |
|       mShell->GetPresContext()->NotifyDidPaintForSubtree();
 | |
|     }
 | |
|   }
 | |
| 
 | |
| 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)
 | |
| {
 | |
| #ifdef MOZ_GECKO_PROFILER
 | |
|   nsIURI* uri = mDocument->GetDocumentURI();
 | |
|   nsIDocument* contentRoot = GetPrimaryContentDocument();
 | |
|   if (contentRoot) {
 | |
|     uri = contentRoot->GetDocumentURI();
 | |
|   }
 | |
|   nsCString uriString = uri ? uri->GetSpecOrDefault() : NS_LITERAL_CSTRING("N/A");
 | |
|   PROFILER_LABEL_DYNAMIC("PresShell", "Paint",
 | |
|     js::ProfileEntry::Category::GRAPHICS, uriString.get());
 | |
| #endif
 | |
| 
 | |
|   Maybe<js::AutoAssertNoContentJS> nojs;
 | |
| 
 | |
|   // On Android, Flash can call into content JS during painting, so we can't
 | |
|   // assert there. However, we don't rely on this assertion on Android because
 | |
|   // we don't paint while JS is running.
 | |
| #if !defined(MOZ_WIDGET_ANDROID)
 | |
|   if (!(aFlags & nsIPresShell::PAINT_COMPOSITE)) {
 | |
|     // We need to allow content JS when the flag is set since we may trigger
 | |
|     // MozAfterPaint events in content in those cases.
 | |
|     nojs.emplace(dom::danger::GetJSContext());
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   NS_ASSERTION(!mIsDestroying, "painting a destroyed PresShell");
 | |
|   NS_ASSERTION(aViewToPaint, "null view");
 | |
| 
 | |
|   MOZ_ASSERT(!mApproximateFrameVisibilityVisited, "Should have been cleared");
 | |
| 
 | |
|   if (!mIsActive) {
 | |
|     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 relied 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(layerManager->GetLastTransactionId(), bounds);
 | |
|           }
 | |
|         } 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;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * This function handles the preventDefault behavior of pointerdown. When user
 | |
|  * preventDefault on pointerdown, We have to mark the active pointer to prevent
 | |
|  * sebsequent mouse events (except mouse transition events) and default
 | |
|  * behaviors.
 | |
|  *
 | |
|  * We add mPreventMouseEventByContent flag in PointerInfo to represent the
 | |
|  * active pointer won't firing compatible mouse events. It's set to true when
 | |
|  * content preventDefault on pointerdown
 | |
|  */
 | |
| static void
 | |
| PostHandlePointerEventsPreventDefault(WidgetPointerEvent* aPointerEvent,
 | |
|                                       WidgetGUIEvent* aMouseOrTouchEvent)
 | |
| {
 | |
|   if (!aPointerEvent->mIsPrimary || aPointerEvent->mMessage != ePointerDown ||
 | |
|       !aPointerEvent->DefaultPreventedByContent()) {
 | |
|     return;
 | |
|   }
 | |
|   nsIPresShell::PointerInfo* pointerInfo = nullptr;
 | |
|   if (!sActivePointersIds->Get(aPointerEvent->pointerId, &pointerInfo) ||
 | |
|       !pointerInfo) {
 | |
|     // We already added the PointerInfo for active pointer when
 | |
|     // PresShell::HandleEvent handling pointerdown event.
 | |
| #ifdef DEBUG
 | |
|     MOZ_CRASH("Got ePointerDown w/o active pointer info!!");
 | |
| #endif // #ifdef DEBUG
 | |
|     return;
 | |
|   }
 | |
|   // PreventDefault only applied for active pointers.
 | |
|   if (!pointerInfo->mActiveState) {
 | |
|     return;
 | |
|   }
 | |
|   aMouseOrTouchEvent->PreventDefault(false);
 | |
|   pointerInfo->mPreventMouseEventByContent = true;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * This function handles the case when content had called preventDefault on the
 | |
|  * active pointer. In that case we have to prevent firing subsequent mouse
 | |
|  * to content. We check the flag PointerInfo::mPreventMouseEventByContent and
 | |
|  * call PreventDefault(false) to stop default behaviors and stop firing mouse
 | |
|  * events to content and chrome.
 | |
|  *
 | |
|  * note: mouse transition events are excluded
 | |
|  * note: we have to clean mPreventMouseEventByContent on pointerup for those
 | |
|  *       devices support hover
 | |
|  * note: we don't suppress firing mouse events to chrome and system group
 | |
|  *       handlers because they may implement default behaviors
 | |
|  */
 | |
| static void
 | |
| PreHandlePointerEventsPreventDefault(WidgetPointerEvent* aPointerEvent,
 | |
|                                      WidgetGUIEvent* aMouseOrTouchEvent)
 | |
| {
 | |
|   if (!aPointerEvent->mIsPrimary || aPointerEvent->mMessage == ePointerDown) {
 | |
|     return;
 | |
|   }
 | |
|   nsIPresShell::PointerInfo* pointerInfo = nullptr;
 | |
|   if (!sActivePointersIds->Get(aPointerEvent->pointerId, &pointerInfo) ||
 | |
|       !pointerInfo) {
 | |
|     // The PointerInfo for active pointer should be added for normal cases. But
 | |
|     // in some cases, we may receive mouse events before adding PointerInfo in
 | |
|     // sActivePointersIds. (e.g. receive mousemove before eMouseEnterIntoWidget
 | |
|     // or change preference 'dom.w3c_pointer_events.enabled' from off to on).
 | |
|     // In these cases, we could ignore them because they are not the events
 | |
|     // between a DefaultPrevented pointerdown and the corresponding pointerup.
 | |
|     return;
 | |
|   }
 | |
|   if (!pointerInfo->mPreventMouseEventByContent) {
 | |
|     return;
 | |
|   }
 | |
|   aMouseOrTouchEvent->PreventDefault(false);
 | |
|   if (aPointerEvent->mMessage == ePointerUp) {
 | |
|     pointerInfo->mPreventMouseEventByContent = false;
 | |
|   }
 | |
| }
 | |
| 
 | |
| 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 = WidgetMouseEvent::eNoButton;
 | |
|       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;
 | |
|     PreHandlePointerEventsPreventDefault(&event, aEvent);
 | |
|     aShell->HandleEvent(aFrame, &event, aDontRetargetEvents, aStatus,
 | |
|                         aTargetContent);
 | |
|     PostHandlePointerEventsPreventDefault(&event, aEvent);
 | |
|   } else if (aEvent->mClass == eTouchEventClass) {
 | |
|     WidgetTouchEvent* touchEvent = aEvent->AsTouchEvent();
 | |
|     int16_t button = WidgetMouseEvent::eLeftButton;
 | |
|     int16_t buttons = WidgetMouseEvent::eLeftButtonFlag;
 | |
|     // loop over all touches and dispatch pointer events on each touch
 | |
|     // copy the event
 | |
|     switch (touchEvent->mMessage) {
 | |
|     case eTouchMove:
 | |
|       pointerMessage = ePointerMove;
 | |
|       button = WidgetMouseEvent::eNoButton;
 | |
|       break;
 | |
|     case eTouchEnd:
 | |
|       pointerMessage = ePointerUp;
 | |
|       buttons = WidgetMouseEvent::eNoButtonFlag;
 | |
|       break;
 | |
|     case eTouchStart:
 | |
|       pointerMessage = ePointerDown;
 | |
|       break;
 | |
|     case eTouchCancel:
 | |
|     case eTouchPointerCancel:
 | |
|       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 (!TouchManager::ShouldConvertTouchToPointer(touch, touchEvent)) {
 | |
|         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 = button;
 | |
|       event.buttons = buttons;
 | |
|       event.inputSource = nsIDOMMouseEvent::MOZ_SOURCE_TOUCH;
 | |
|       event.convertToPointer = touch->convertToPointer = false;
 | |
|       PreHandlePointerEventsPreventDefault(&event, aEvent);
 | |
|       aShell->HandleEvent(aFrame, &event, aDontRetargetEvents, aStatus,
 | |
|                           aTargetContent);
 | |
|       PostHandlePointerEventsPreventDefault(&event, aEvent);
 | |
|     }
 | |
|   }
 | |
|   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;
 | |
| };
 | |
| 
 | |
| bool
 | |
| PresShell::CanDispatchEvent(const WidgetGUIEvent* aEvent) const
 | |
| {
 | |
|   bool rv =
 | |
|     mPresContext && !mHaveShutDown && nsContentUtils::IsSafeToRunScript();
 | |
|   if (aEvent) {
 | |
|     rv &= (aEvent && aEvent->mWidget && !aEvent->mWidget->Destroyed());
 | |
|   }
 | |
|   return rv;
 | |
| }
 | |
| 
 | |
| nsresult
 | |
| PresShell::HandleEvent(nsIFrame* aFrame,
 | |
|                        WidgetGUIEvent* aEvent,
 | |
|                        bool aDontRetargetEvents,
 | |
|                        nsEventStatus* aEventStatus,
 | |
|                        nsIContent** aTargetContent)
 | |
| {
 | |
| #ifdef MOZ_TASK_TRACER
 | |
|   Maybe<AutoSourceEvent> taskTracerEvent;
 | |
|   if (MOZ_UNLIKELY(IsStartLogging())) {
 | |
|     // 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;
 | |
|     }
 | |
|     taskTracerEvent.emplace(type);
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   NS_ASSERTION(aFrame, "aFrame should be not null");
 | |
| 
 | |
|   if (sPointerEventEnabled) {
 | |
|     AutoWeakFrame 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 != nsEventStatus_eConsumeNoDefault) {
 | |
|       // Don't dispatch event to AccessibleCaretEventHub when the event status
 | |
|       // is nsEventStatus_eConsumeNoDefault. This might be happened when content
 | |
|       // preventDefault on the pointer events. In such case, we also call
 | |
|       // preventDefault on mouse events to stop default behaviors.
 | |
|       *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 = GetPrimaryContentDocument();
 | |
| #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();
 | |
|       }
 | |
| 
 | |
|       AutoWeakFrame 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?");
 | |
|     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())) {
 | |
| 
 | |
|       // 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 only want to use the popup list if the capture is
 | |
|       // inside the popup.
 | |
|       if (framePresContext == rootPresContext &&
 | |
|           frame == mFrameConstructor->GetRootFrame()) {
 | |
|         frame = popupFrame;
 | |
|       } else if (capturingContent &&
 | |
|                  nsContentUtils::ContentIsDescendantOf(
 | |
|                    capturingContent, 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.
 | |
|         AutoWeakFrame 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
 | |
|     AutoWeakFrame 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;
 | |
| }
 | |
| 
 | |
| nsIDocument*
 | |
| PresShell::GetPrimaryContentDocument()
 | |
| {
 | |
|   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();
 | |
| }
 | |
| 
 | |
| #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;
 | |
|           mPresContext->RecordInteractionTime(
 | |
|             nsPresContext::InteractionType::eKeyInteraction,
 | |
|             aEvent->mTimeStamp);
 | |
|         }
 | |
| 
 | |
|         Telemetry::AccumulateTimeDelta(Telemetry::INPUT_EVENT_QUEUED_KEYBOARD_MS, aEvent->mTimeStamp);
 | |
|         break;
 | |
|       }
 | |
|       case eMouseDown:
 | |
|       case eMouseUp:
 | |
|         Telemetry::AccumulateTimeDelta(Telemetry::INPUT_EVENT_QUEUED_CLICK_MS, aEvent->mTimeStamp);
 | |
|         MOZ_FALLTHROUGH;
 | |
|       case ePointerDown:
 | |
|       case ePointerUp:
 | |
|         isHandlingUserInput = true;
 | |
|         mPresContext->RecordInteractionTime(
 | |
|           nsPresContext::InteractionType::eClickInteraction,
 | |
|           aEvent->mTimeStamp);
 | |
|         break;
 | |
| 
 | |
|       case eMouseMove:
 | |
|         if (aEvent->mFlags.mHandledByAPZ) {
 | |
|           Telemetry::AccumulateTimeDelta(Telemetry::INPUT_EVENT_QUEUED_APZ_MOUSE_MOVE_MS, aEvent->mTimeStamp);
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|       case eDrop: {
 | |
|         nsCOMPtr<nsIDragSession> session = nsContentUtils::GetDragSession();
 | |
|         if (session) {
 | |
|           bool onlyChromeDrop = false;
 | |
|           session->GetOnlyChromeDrop(&onlyChromeDrop);
 | |
|           if (onlyChromeDrop) {
 | |
|             aEvent->mFlags.mOnlyChromeDispatch = true;
 | |
|           }
 | |
|         }
 | |
|         break;
 | |
|       }
 | |
| 
 | |
|       case eWheel:
 | |
|         if (aEvent->mFlags.mHandledByAPZ) {
 | |
|           Telemetry::AccumulateTimeDelta(Telemetry::INPUT_EVENT_QUEUED_APZ_WHEEL_MS, aEvent->mTimeStamp);
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|       case eTouchMove:
 | |
|         if (aEvent->mFlags.mHandledByAPZ) {
 | |
|           Telemetry::AccumulateTimeDelta(Telemetry::INPUT_EVENT_QUEUED_APZ_TOUCH_MOVE_MS, aEvent->mTimeStamp);
 | |
|         }
 | |
|         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);
 | |
| 
 | |
|       mPresContext->RecordInteractionTime(
 | |
|         nsPresContext::InteractionType::eMouseMoveInteraction,
 | |
|         aEvent->mTimeStamp);
 | |
|     }
 | |
| 
 | |
|     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;
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       if (aEvent->mMessage == eKeyDown) {
 | |
|         mIsLastKeyDownCanceled = aEvent->mFlags.mDefaultPrevented;
 | |
|       }
 | |
|       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->mTimeStamp > mLastOSWake &&
 | |
|       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 {
 | |
|       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) {
 | |
|       RefPtr<nsXULElement> xulElement =
 | |
|         nsXULElement::FromContent(focusedContent);
 | |
|       if (xulElement) {
 | |
|         IgnoredErrorResult ignored;
 | |
|         nsCOMPtr<nsIBoxObject> box = xulElement->GetBoxObject(ignored);
 | |
|         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(FlushType::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->DisableInteractionTimeRecording();
 | |
|     if (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);
 | |
|       if (ev->IsKeyPressEvent() && mIsLastKeyDownCanceled) {
 | |
|         continue;
 | |
|       }
 | |
|       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 (mObservingLayoutFlushes || mIsDestroying || mIsReflowing ||
 | |
|       mDirtyRoots.IsEmpty())
 | |
|     return;
 | |
| 
 | |
|   if (!mPresContext->HasPendingInterrupt() || !ScheduleReflowOffTimer()) {
 | |
|     ScheduleReflow();
 | |
|   }
 | |
| 
 | |
|   ASSERT_REFLOW_SCHEDULED_STATE();
 | |
| }
 | |
| 
 | |
| void
 | |
| PresShell::ScheduleReflow()
 | |
| {
 | |
|   ASSERT_REFLOW_SCHEDULED_STATE();
 | |
|   DoObserveLayoutFlushes();
 | |
|   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(!mObservingLayoutFlushes, "Shouldn't get here");
 | |
|   ASSERT_REFLOW_SCHEDULED_STATE();
 | |
| 
 | |
|   if (!mReflowContinueTimer) {
 | |
|     mReflowContinueTimer = do_CreateInstance("@mozilla.org/timer;1");
 | |
|     mReflowContinueTimer->SetTarget(
 | |
|         mDocument->EventTargetFor(TaskCategory::Other));
 | |
|     if (!mReflowContinueTimer ||
 | |
|         NS_FAILED(mReflowContinueTimer->
 | |
|                     InitWithNamedFuncCallback(sReflowContinueCallback, this, 30,
 | |
|                                               nsITimer::TYPE_ONE_SHOT,
 | |
|                                               "sReflowContinueCallback"))) {
 | |
|       return false;
 | |
|     }
 | |
|   }
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| bool
 | |
| PresShell::DoReflow(nsIFrame* target, bool aInterruptible)
 | |
| {
 | |
|   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);
 | |
|   }
 | |
| 
 | |
| #ifdef MOZ_GECKO_PROFILER
 | |
|   nsIURI* uri = mDocument->GetDocumentURI();
 | |
|   nsCString uriString = uri ? uri->GetSpecOrDefault() : NS_LITERAL_CSTRING("N/A");
 | |
|   PROFILER_LABEL_DYNAMIC("PresShell", "DoReflow",
 | |
|     js::ProfileEntry::Category::GRAPHICS, uriString.get());
 | |
| #endif
 | |
| 
 | |
|   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.IsEmpty(),
 | |
|                "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);
 | |
|     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(FlushType::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 record that we might need flushing
 | |
|         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");
 | |
| 
 | |
|       AutoWeakFrame 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(FlushType::ContentAndNotify);
 | |
| 
 | |
|       if (weakRoot.IsAlive()) {
 | |
|         WalkFramesThroughPlaceholders(mPresContext, rootFrame,
 | |
|                                       &ReResolveMenusAndTrees, nullptr);
 | |
| 
 | |
|         // Because "chrome:" URL equality is messy, reframe image box
 | |
|         // frames (hack!).
 | |
|         nsStyleChangeList changeList(mPresContext->StyleSet()->BackendType());
 | |
|         WalkFramesThroughPlaceholders(mPresContext, rootFrame,
 | |
|                                       ReframeImageBoxes, &changeList);
 | |
|         // Mark ourselves as not safe to flush while we're doing frame
 | |
|         // construction.
 | |
|         {
 | |
|           nsAutoScriptBlocker scriptBlocker;
 | |
|           ++mChangeNestCount;
 | |
|           RestyleManager* 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, "memory-pressure")) {
 | |
|     if (!AssumeAllFramesVisible() && mPresContext->IsRootContentDocument()) {
 | |
|       DoUpdateApproximateFrameVisibility(/* aRemoveOnly = */ true);
 | |
|     }
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   if (!nsCRT::strcmp(aTopic, NS_WIDGET_WAKE_OBSERVER_TOPIC)) {
 | |
|     mLastOSWake = TimeStamp::Now();
 | |
|   }
 | |
| 
 | |
|   NS_WARNING("unrecognized topic in PresShell::Observe");
 | |
|   return NS_ERROR_FAILURE;
 | |
| }
 | |
| 
 | |
| bool
 | |
| nsIPresShell::AddRefreshObserverInternal(nsARefreshObserver* aObserver,
 | |
|                                          FlushType aFlushType)
 | |
| {
 | |
|   nsPresContext* presContext = GetPresContext();
 | |
|   return presContext &&
 | |
|       presContext->RefreshDriver()->AddRefreshObserver(aObserver, aFlushType);
 | |
| }
 | |
| 
 | |
| /* virtual */ bool
 | |
| nsIPresShell::AddRefreshObserverExternal(nsARefreshObserver* aObserver,
 | |
|                                          FlushType aFlushType)
 | |
| {
 | |
|   return AddRefreshObserverInternal(aObserver, aFlushType);
 | |
| }
 | |
| 
 | |
| bool
 | |
| nsIPresShell::RemoveRefreshObserverInternal(nsARefreshObserver* aObserver,
 | |
|                                             FlushType aFlushType)
 | |
| {
 | |
|   nsPresContext* presContext = GetPresContext();
 | |
|   return presContext &&
 | |
|       presContext->RefreshDriver()->RemoveRefreshObserver(aObserver, aFlushType);
 | |
| }
 | |
| 
 | |
| /* virtual */ bool
 | |
| nsIPresShell::RemoveRefreshObserverExternal(nsARefreshObserver* aObserver,
 | |
|                                             FlushType 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;
 | |
| }
 | |
| 
 | |
| void
 | |
| nsIPresShell::DoObserveStyleFlushes()
 | |
| {
 | |
|   MOZ_ASSERT(!ObservingStyleFlushes());
 | |
|   mObservingStyleFlushes =
 | |
|     mPresContext->RefreshDriver()->AddStyleFlushObserver(this);
 | |
| }
 | |
| 
 | |
| void
 | |
| nsIPresShell::DoObserveLayoutFlushes()
 | |
| {
 | |
|   MOZ_ASSERT(!ObservingLayoutFlushes());
 | |
|   mObservingLayoutFlushes =
 | |
|     mPresContext->RefreshDriver()->AddLayoutFlushObserver(this);
 | |
| }
 | |
| 
 | |
| //------------------------------------------------------
 | |
| // 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;
 | |
|   keyEvent->mFlags.mIsSuppressedOrDelayed = true;
 | |
|   mEvent = keyEvent;
 | |
| }
 | |
| 
 | |
| bool
 | |
| PresShell::DelayedKeyEvent::IsKeyPressEvent()
 | |
| {
 | |
|   return mEvent->mMessage == eKeyPress;
 | |
| }
 | |
| 
 | |
| // 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(FlushType::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(FILE *out, int32_t aIndent)
 | |
| {
 | |
|   nsIFrame* rootFrame = GetRootFrame();
 | |
|   if (rootFrame) {
 | |
|     rootFrame->StyleContext()->List(out, aIndent);
 | |
|   }
 | |
| 
 | |
|   // The root element's frame's style context is the root of a separate tree.
 | |
|   Element* rootElement = mDocument->GetRootElement();
 | |
|   if (rootElement) {
 | |
|     nsIFrame* rootElementFrame = rootElement->GetPrimaryFrame();
 | |
|     if (rootElementFrame) {
 | |
|       rootElementFrame->StyleContext()->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 if (ServoStyleSet* styleSet = StyleSet()->GetAsServo()) {
 | |
|     *aStyleSetsSize += styleSet->SizeOfIncludingThis(aMallocSizeOf);
 | |
|   } else {
 | |
|     MOZ_CRASH();
 | |
|   }
 | |
| 
 | |
|   *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;
 | |
|   mFontSizeInflationEnabled = DetermineFontSizeInflationState();
 | |
| 
 | |
|   HandleSystemFontScale();
 | |
| }
 | |
| 
 | |
| bool
 | |
| nsIPresShell::DetermineFontSizeInflationState()
 | |
| {
 | |
|   MOZ_ASSERT(mPresContext, "our pres context should not be null");
 | |
|   if ((FontSizeInflationEmPerLine() == 0 &&
 | |
|       FontSizeInflationMinTwips() == 0) || mPresContext->IsChrome()) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // 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()) {
 | |
|         return false;
 | |
|       }
 | |
|     } else if (XRE_IsParentProcess()) {
 | |
|       // We're in the master process.  Cancel inflation if it's been
 | |
|       // explicitly disabled.
 | |
|       if (FontSizeInflationDisabledInMasterProcess()) {
 | |
|         return false;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // 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)) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   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()) {
 | |
|       return false;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| bool
 | |
| nsIPresShell::FontSizeInflationEnabled()
 | |
| {
 | |
|   if (mFontSizeInflationEnabledIsDirty) {
 | |
|     RecomputeFontSizeInflationEnabled();
 | |
|   }
 | |
| 
 | |
|   return mFontSizeInflationEnabled;
 | |
| }
 | |
| 
 | |
| void
 | |
| nsIPresShell::HandleSystemFontScale()
 | |
| {
 | |
|   float fontScale = nsLayoutUtils::SystemFontScale();
 | |
|   if (fontScale == 0.0f) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(mDocument && mPresContext, "our document and pres context should not be null");
 | |
| 
 | |
|   if (!mFontSizeInflationEnabled && !mDocument->IsSyntheticDocument()) {
 | |
|     mPresContext->SetSystemFontScale(fontScale);
 | |
|   } else {
 | |
|     mPresContext->SetSystemFontScale(1.0f);
 | |
|   }
 | |
| }
 | |
| 
 | |
| 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);
 | |
|   }
 | |
| }
 | |
| 
 | |
| static SheetType
 | |
| ToSheetType(uint32_t aServiceSheetType)
 | |
| {
 | |
|   switch (aServiceSheetType) {
 | |
|     case nsIStyleSheetService::AGENT_SHEET:
 | |
|       return SheetType::Agent;
 | |
|       break;
 | |
|     case nsIStyleSheetService::USER_SHEET:
 | |
|       return SheetType::User;
 | |
|       break;
 | |
|     default:
 | |
|       MOZ_FALLTHROUGH_ASSERT("unexpected aSheetType value");
 | |
|     case nsIStyleSheetService::AUTHOR_SHEET:
 | |
|       return SheetType::Doc;
 | |
|   }
 | |
| }
 | |
| 
 | |
| nsresult
 | |
| nsIPresShell::HasRuleProcessorUsedByMultipleStyleSets(uint32_t aSheetType,
 | |
|                                                       bool* aRetVal)
 | |
| {
 | |
|   *aRetVal = false;
 | |
|   if (nsStyleSet* styleSet = mStyleSet->GetAsGecko()) {
 | |
|     // ServoStyleSets do not have rule processors.
 | |
|     SheetType type = ToSheetType(aSheetType);
 | |
|     *aRetVal = styleSet->HasRuleProcessorUsedByMultipleStyleSets(type);
 | |
|   }
 | |
|   return NS_OK;
 | |
| }
 | |
| 
 | |
| void
 | |
| PresShell::NotifyStyleSheetServiceSheetAdded(StyleSheet* aSheet,
 | |
|                                              uint32_t aSheetType)
 | |
| {
 | |
|   if (!mStyleSet) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   switch (aSheetType) {
 | |
|     case nsIStyleSheetService::AGENT_SHEET:
 | |
|       AddAgentSheet(aSheet);
 | |
|       break;
 | |
|     case nsIStyleSheetService::USER_SHEET:
 | |
|       AddUserSheet(aSheet);
 | |
|       break;
 | |
|     case nsIStyleSheetService::AUTHOR_SHEET:
 | |
|       AddAuthorSheet(aSheet);
 | |
|       break;
 | |
|     default:
 | |
|       MOZ_ASSERT_UNREACHABLE("unexpected aSheetType value");
 | |
|       break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void
 | |
| PresShell::NotifyStyleSheetServiceSheetRemoved(StyleSheet* aSheet,
 | |
|                                                uint32_t aSheetType)
 | |
| {
 | |
|   if (!mStyleSet) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   RemoveSheet(ToSheetType(aSheetType), aSheet);
 | |
| }
 |