forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			3142 lines
		
	
	
	
		
			103 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			3142 lines
		
	
	
	
		
			103 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
 | 
						|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
 | 
						|
/* This Source Code Form is subject to the terms of the Mozilla Public
 | 
						|
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 | 
						|
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | 
						|
 | 
						|
/* a presentation of a document, part 1 */
 | 
						|
 | 
						|
#include "nsPresContext.h"
 | 
						|
#include "nsPresContextInlines.h"
 | 
						|
 | 
						|
#include "mozilla/ArrayUtils.h"
 | 
						|
#if defined(MOZ_WIDGET_ANDROID)
 | 
						|
#  include "mozilla/AsyncEventDispatcher.h"
 | 
						|
#endif
 | 
						|
#include "mozilla/CycleCollectedJSContext.h"
 | 
						|
#include "mozilla/DebugOnly.h"
 | 
						|
#include "mozilla/Encoding.h"
 | 
						|
#include "mozilla/EventDispatcher.h"
 | 
						|
#include "mozilla/EventStateManager.h"
 | 
						|
#include "mozilla/PresShell.h"
 | 
						|
#include "mozilla/PresShellInlines.h"
 | 
						|
 | 
						|
#include "base/basictypes.h"
 | 
						|
#include "nsCRT.h"
 | 
						|
#include "nsCOMPtr.h"
 | 
						|
#include "nsCSSFrameConstructor.h"
 | 
						|
#include "nsDocShell.h"
 | 
						|
#include "nsIConsoleService.h"
 | 
						|
#include "nsIDocumentViewer.h"
 | 
						|
#include "nsPIDOMWindow.h"
 | 
						|
#include "mozilla/ServoStyleSet.h"
 | 
						|
#include "mozilla/MediaFeatureChange.h"
 | 
						|
#include "nsIContent.h"
 | 
						|
#include "nsIFrame.h"
 | 
						|
#include "mozilla/dom/BrowsingContext.h"
 | 
						|
#include "mozilla/dom/Document.h"
 | 
						|
#include "mozilla/dom/DocumentInlines.h"
 | 
						|
#include "nsIPrintSettings.h"
 | 
						|
#include "nsLanguageAtomService.h"
 | 
						|
#include "mozilla/LookAndFeel.h"
 | 
						|
#include "nsIInterfaceRequestorUtils.h"
 | 
						|
#include "nsHTMLDocument.h"
 | 
						|
#include "nsIWeakReferenceUtils.h"
 | 
						|
#include "nsThreadUtils.h"
 | 
						|
#include "nsLayoutUtils.h"
 | 
						|
#include "nsViewManager.h"
 | 
						|
#include "mozilla/RestyleManager.h"
 | 
						|
#include "gfxPlatform.h"
 | 
						|
#include "nsFontFaceLoader.h"
 | 
						|
#include "mozilla/AnimationEventDispatcher.h"
 | 
						|
#include "mozilla/EffectCompositor.h"
 | 
						|
#include "mozilla/EventListenerManager.h"
 | 
						|
#include "prenv.h"
 | 
						|
#include "nsTransitionManager.h"
 | 
						|
#include "nsAnimationManager.h"
 | 
						|
#include "CounterStyleManager.h"
 | 
						|
#include "mozilla/MemoryReporting.h"
 | 
						|
#include "mozilla/dom/Element.h"
 | 
						|
#include "nsIMessageManager.h"
 | 
						|
#include "mozilla/dom/HTMLBodyElement.h"
 | 
						|
#include "mozilla/dom/MediaQueryList.h"
 | 
						|
#include "mozilla/SMILAnimationController.h"
 | 
						|
#include "mozilla/css/ImageLoader.h"
 | 
						|
#include "mozilla/dom/PBrowserParent.h"
 | 
						|
#include "mozilla/dom/BrowserChild.h"
 | 
						|
#include "mozilla/dom/BrowserParent.h"
 | 
						|
#include "mozilla/dom/FontFaceSet.h"
 | 
						|
#include "mozilla/StaticPresData.h"
 | 
						|
#include "nsRefreshDriver.h"
 | 
						|
#include "LayerUserData.h"
 | 
						|
#include "mozilla/dom/NotifyPaintEvent.h"
 | 
						|
#include "nsFontCache.h"
 | 
						|
#include "nsFrameLoader.h"
 | 
						|
#include "nsContentUtils.h"
 | 
						|
#include "nsPIWindowRoot.h"
 | 
						|
#include "mozilla/Preferences.h"
 | 
						|
#include "gfxTextRun.h"
 | 
						|
#include "nsFontFaceUtils.h"
 | 
						|
#include "COLRFonts.h"
 | 
						|
#include "mozilla/ContentBlockingAllowList.h"
 | 
						|
#include "mozilla/GlobalStyleSheetCache.h"
 | 
						|
#include "mozilla/ServoBindings.h"
 | 
						|
#include "mozilla/StaticPrefs_bidi.h"
 | 
						|
#include "mozilla/StaticPrefs_layout.h"
 | 
						|
#include "mozilla/StaticPrefs_widget.h"
 | 
						|
#include "mozilla/StaticPrefs_zoom.h"
 | 
						|
#include "mozilla/StyleSheet.h"
 | 
						|
#include "mozilla/StyleSheetInlines.h"
 | 
						|
#include "mozilla/Telemetry.h"
 | 
						|
#include "mozilla/TimelineManager.h"
 | 
						|
#include "mozilla/dom/Performance.h"
 | 
						|
#include "mozilla/dom/PerformanceMainThread.h"
 | 
						|
#include "mozilla/dom/PerformanceTiming.h"
 | 
						|
#include "mozilla/dom/PerformancePaintTiming.h"
 | 
						|
#include "mozilla/layers/APZThreadUtils.h"
 | 
						|
#include "MobileViewportManager.h"
 | 
						|
#include "mozilla/dom/ImageTracker.h"
 | 
						|
#ifdef ACCESSIBILITY
 | 
						|
#  include "mozilla/a11y/DocAccessible.h"
 | 
						|
#endif
 | 
						|
 | 
						|
// Needed for Start/Stop of Image Animation
 | 
						|
#include "imgIContainer.h"
 | 
						|
#include "nsIImageLoadingContent.h"
 | 
						|
 | 
						|
#include "nsBidiUtils.h"
 | 
						|
#include "nsServiceManagerUtils.h"
 | 
						|
 | 
						|
#include "mozilla/dom/URL.h"
 | 
						|
#include "mozilla/ServoCSSParser.h"
 | 
						|
 | 
						|
using namespace mozilla;
 | 
						|
using namespace mozilla::dom;
 | 
						|
using namespace mozilla::gfx;
 | 
						|
using namespace mozilla::layers;
 | 
						|
 | 
						|
/**
 | 
						|
 * Layer UserData for ContainerLayers that want to be notified
 | 
						|
 * of local invalidations of them and their descendant layers.
 | 
						|
 * Pass a callback to ComputeDifferences to have these called.
 | 
						|
 */
 | 
						|
class ContainerLayerPresContext : public LayerUserData {
 | 
						|
 public:
 | 
						|
  nsPresContext* mPresContext;
 | 
						|
};
 | 
						|
 | 
						|
bool nsPresContext::IsDOMPaintEventPending() {
 | 
						|
  if (!mTransactions.IsEmpty()) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  nsRootPresContext* drpc = GetRootPresContext();
 | 
						|
  if (drpc && drpc->mRefreshDriver->ViewManagerFlushIsPending()) {
 | 
						|
    // Since we're promising that there will be a MozAfterPaint event
 | 
						|
    // fired, we record an empty invalidation in case display list
 | 
						|
    // invalidation doesn't invalidate anything further.
 | 
						|
    NotifyInvalidation(drpc->mRefreshDriver->LastTransactionId().Next(),
 | 
						|
                       nsRect(0, 0, 0, 0));
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
struct WeakRunnableMethod : Runnable {
 | 
						|
  using Method = void (nsPresContext::*)();
 | 
						|
 | 
						|
  WeakRunnableMethod(const char* aName, nsPresContext* aPc, Method aMethod)
 | 
						|
      : Runnable(aName), mPresContext(aPc), mMethod(aMethod) {}
 | 
						|
 | 
						|
  NS_IMETHOD Run() override {
 | 
						|
    if (nsPresContext* pc = mPresContext.get()) {
 | 
						|
      (pc->*mMethod)();
 | 
						|
    }
 | 
						|
    return NS_OK;
 | 
						|
  }
 | 
						|
 | 
						|
 private:
 | 
						|
  WeakPtr<nsPresContext> mPresContext;
 | 
						|
  Method mMethod;
 | 
						|
};
 | 
						|
 | 
						|
// When forcing a font-info-update reflow from style, we don't need to reframe,
 | 
						|
// but we'll need to restyle to pick up updated font metrics. In order to avoid
 | 
						|
// synchronously having to deal with multiple restyles, we use an early refresh
 | 
						|
// driver runner, which should prevent flashing for users.
 | 
						|
//
 | 
						|
// We might do a bit of extra work if the page flushes layout between the
 | 
						|
// restyle and when this happens, which is a bit unfortunate, but not worse than
 | 
						|
// what we used to do...
 | 
						|
//
 | 
						|
// A better solution would be to be able to synchronously initialize font
 | 
						|
// information from style worker threads, perhaps...
 | 
						|
void nsPresContext::ForceReflowForFontInfoUpdateFromStyle() {
 | 
						|
  if (mPendingFontInfoUpdateReflowFromStyle) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  mPendingFontInfoUpdateReflowFromStyle = true;
 | 
						|
  nsCOMPtr<nsIRunnable> ev = new WeakRunnableMethod(
 | 
						|
      "nsPresContext::DoForceReflowForFontInfoUpdateFromStyle", this,
 | 
						|
      &nsPresContext::DoForceReflowForFontInfoUpdateFromStyle);
 | 
						|
  RefreshDriver()->AddEarlyRunner(ev);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::DoForceReflowForFontInfoUpdateFromStyle() {
 | 
						|
  mPendingFontInfoUpdateReflowFromStyle = false;
 | 
						|
  ForceReflowForFontInfoUpdate(false);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::ForceReflowForFontInfoUpdate(bool aNeedsReframe) {
 | 
						|
  // In the case of a static-clone document used for printing or print-preview,
 | 
						|
  // this is undesirable because the nsPrintJob is holding weak refs to frames
 | 
						|
  // that will get blown away unexpectedly by this reconstruction. So the
 | 
						|
  // prescontext for a print/preview doc ignores the font-list update.
 | 
						|
  //
 | 
						|
  // This means the print document may still be using cached fonts that are no
 | 
						|
  // longer present in the font list, but that should be safe given that all the
 | 
						|
  // required font instances have already been created, so it won't be depending
 | 
						|
  // on access to the font-list entries.
 | 
						|
  //
 | 
						|
  // XXX Actually, I think it's probably a bad idea to do *any* restyling of
 | 
						|
  // print documents in response to pref changes. We could be in the middle
 | 
						|
  // of printing the document, and reflowing all the frames might cause some
 | 
						|
  // kind of unwanted mid-document discontinuity.
 | 
						|
  if (IsPrintingOrPrintPreview()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // If there's a user font set, discard any src:local() faces it may have
 | 
						|
  // loaded because their font entries may no longer be valid.
 | 
						|
  if (auto* fonts = Document()->GetFonts()) {
 | 
						|
    fonts->GetImpl()->ForgetLocalFaces();
 | 
						|
  }
 | 
						|
 | 
						|
  FlushFontCache();
 | 
						|
 | 
						|
  if (!mPresShell) {
 | 
						|
    // RebuildAllStyleData won't do anything without mPresShell.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  nsChangeHint changeHint =
 | 
						|
      aNeedsReframe ? nsChangeHint_ReconstructFrame : NS_STYLE_HINT_REFLOW;
 | 
						|
 | 
						|
  // We also need to trigger restyling for ex/ch units changes to take effect,
 | 
						|
  // if needed.
 | 
						|
  auto restyleHint = StyleSet()->UsesFontMetrics()
 | 
						|
                         ? RestyleHint::RecascadeSubtree()
 | 
						|
                         : RestyleHint{0};
 | 
						|
 | 
						|
  RebuildAllStyleData(changeHint, restyleHint);
 | 
						|
}
 | 
						|
 | 
						|
static bool IsVisualCharset(NotNull<const Encoding*> aCharset) {
 | 
						|
  return aCharset == ISO_8859_8_ENCODING;
 | 
						|
}
 | 
						|
 | 
						|
nsPresContext::nsPresContext(dom::Document* aDocument, nsPresContextType aType)
 | 
						|
    : mPresShell(nullptr),
 | 
						|
      mDocument(aDocument),
 | 
						|
      mMedium(aType == eContext_Galley ? nsGkAtoms::screen : nsGkAtoms::print),
 | 
						|
      mTextZoom(1.0),
 | 
						|
      mFullZoom(1.0),
 | 
						|
      mLastFontInflationScreenSize(gfxSize(-1.0, -1.0)),
 | 
						|
      mCurAppUnitsPerDevPixel(0),
 | 
						|
      mDynamicToolbarMaxHeight(0),
 | 
						|
      mDynamicToolbarHeight(0),
 | 
						|
      mPageSize(-1, -1),
 | 
						|
      mPageScale(0.0),
 | 
						|
      mPPScale(1.0f),
 | 
						|
      mViewportScrollOverrideElement(nullptr),
 | 
						|
      mElementsRestyled(0),
 | 
						|
      mFramesConstructed(0),
 | 
						|
      mFramesReflowed(0),
 | 
						|
      mAnimationTriggeredRestyles(0),
 | 
						|
      mInterruptChecksToSkip(0),
 | 
						|
      mNextFrameRateMultiplier(0),
 | 
						|
      mMeasuredTicksSinceLoading(0),
 | 
						|
      mViewportScrollStyles(StyleOverflow::Auto, StyleOverflow::Auto),
 | 
						|
      // mImageAnimationMode is initialised below, in constructor body
 | 
						|
      mImageAnimationModePref(imgIContainer::kNormalAnimMode),
 | 
						|
      mType(aType),
 | 
						|
      mInflationDisabledForShrinkWrap(false),
 | 
						|
      mInteractionTimeEnabled(true),
 | 
						|
      mHasPendingInterrupt(false),
 | 
						|
      mHasEverBuiltInvisibleText(false),
 | 
						|
      mPendingInterruptFromTest(false),
 | 
						|
      mInterruptsEnabled(false),
 | 
						|
      mSendAfterPaintToContent(false),
 | 
						|
      mDrawImageBackground(true),  // always draw the background
 | 
						|
      mDrawColorBackground(true),
 | 
						|
      // mNeverAnimate is initialised below, in constructor body
 | 
						|
      mPaginated(aType != eContext_Galley),
 | 
						|
      mCanPaginatedScroll(false),
 | 
						|
      mDoScaledTwips(true),
 | 
						|
      mIsRootPaginatedDocument(false),
 | 
						|
      mPrefBidiDirection(false),
 | 
						|
      mPrefScrollbarSide(0),
 | 
						|
      mPendingThemeChanged(false),
 | 
						|
      mPendingThemeChangeKind(0),
 | 
						|
      mPendingUIResolutionChanged(false),
 | 
						|
      mPendingFontInfoUpdateReflowFromStyle(false),
 | 
						|
      mIsGlyph(false),
 | 
						|
      mCounterStylesDirty(true),
 | 
						|
      mFontFeatureValuesDirty(true),
 | 
						|
      mFontPaletteValuesDirty(true),
 | 
						|
      mIsVisual(false),
 | 
						|
      mInRDMPane(false),
 | 
						|
      mHasWarnedAboutTooLargeDashedOrDottedRadius(false),
 | 
						|
      mQuirkSheetAdded(false),
 | 
						|
      mHadNonBlankPaint(false),
 | 
						|
      mHadFirstContentfulPaint(false),
 | 
						|
      mHadNonTickContentfulPaint(false),
 | 
						|
      mHadContentfulPaintComposite(false),
 | 
						|
      mUserInputEventsAllowed(false),
 | 
						|
#ifdef DEBUG
 | 
						|
      mInitialized(false),
 | 
						|
#endif
 | 
						|
      mOverriddenOrEmbedderColorScheme(dom::PrefersColorSchemeOverride::None) {
 | 
						|
#ifdef DEBUG
 | 
						|
  PodZero(&mLayoutPhaseCount);
 | 
						|
#endif
 | 
						|
 | 
						|
  if (!IsDynamic()) {
 | 
						|
    mImageAnimationMode = imgIContainer::kDontAnimMode;
 | 
						|
    mNeverAnimate = true;
 | 
						|
  } else {
 | 
						|
    mImageAnimationMode = imgIContainer::kNormalAnimMode;
 | 
						|
    mNeverAnimate = false;
 | 
						|
  }
 | 
						|
  NS_ASSERTION(mDocument, "Null document");
 | 
						|
 | 
						|
  // if text perf logging enabled, init stats struct
 | 
						|
  if (MOZ_LOG_TEST(gfxPlatform::GetLog(eGfxLog_textperf), LogLevel::Warning)) {
 | 
						|
    mTextPerf = MakeUnique<gfxTextPerfMetrics>();
 | 
						|
  }
 | 
						|
 | 
						|
  if (StaticPrefs::gfx_missing_fonts_notify()) {
 | 
						|
    mMissingFonts = MakeUnique<gfxMissingFontRecorder>();
 | 
						|
  }
 | 
						|
 | 
						|
  if (StaticPrefs::layout_dynamic_toolbar_max_height() > 0) {
 | 
						|
    // The pref for dynamic toolbar max height is only used in reftests so it's
 | 
						|
    // fine to set here.
 | 
						|
    mDynamicToolbarMaxHeight = StaticPrefs::layout_dynamic_toolbar_max_height();
 | 
						|
  }
 | 
						|
 | 
						|
  UpdateFontVisibility();
 | 
						|
}
 | 
						|
 | 
						|
static const char* gExactCallbackPrefs[] = {
 | 
						|
    "browser.active_color",
 | 
						|
    "browser.anchor_color",
 | 
						|
    "browser.visited_color",
 | 
						|
    "dom.meta-viewport.enabled",
 | 
						|
    "dom.send_after_paint_to_content",
 | 
						|
    "image.animation_mode",
 | 
						|
    "intl.accept_languages",
 | 
						|
    "layout.css.devPixelsPerPx",
 | 
						|
    "layout.css.dpi",
 | 
						|
    "layout.css.text-transform.uppercase-eszett.enabled",
 | 
						|
    "privacy.trackingprotection.enabled",
 | 
						|
    "ui.use_standins_for_native_colors",
 | 
						|
    nullptr,
 | 
						|
};
 | 
						|
 | 
						|
static const char* gPrefixCallbackPrefs[] = {
 | 
						|
    "bidi.", "browser.display.",    "browser.viewport.",
 | 
						|
    "font.", "gfx.font_rendering.", "layout.css.font-visibility.",
 | 
						|
    nullptr,
 | 
						|
};
 | 
						|
 | 
						|
void nsPresContext::Destroy() {
 | 
						|
  if (mEventManager) {
 | 
						|
    // unclear if these are needed, but can't hurt
 | 
						|
    mEventManager->NotifyDestroyPresContext(this);
 | 
						|
    mEventManager->SetPresContext(nullptr);
 | 
						|
    mEventManager = nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mFontCache) {
 | 
						|
    mFontCache->Destroy();
 | 
						|
    mFontCache = nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  // Unregister preference callbacks
 | 
						|
  Preferences::UnregisterPrefixCallbacks(nsPresContext::PreferenceChanged,
 | 
						|
                                         gPrefixCallbackPrefs, this);
 | 
						|
  Preferences::UnregisterCallbacks(nsPresContext::PreferenceChanged,
 | 
						|
                                   gExactCallbackPrefs, this);
 | 
						|
 | 
						|
  mRefreshDriver = nullptr;
 | 
						|
  MOZ_ASSERT(mManagedPostRefreshObservers.IsEmpty());
 | 
						|
}
 | 
						|
 | 
						|
nsPresContext::~nsPresContext() {
 | 
						|
  MOZ_ASSERT(!mPresShell, "Presshell forgot to clear our mPresShell pointer");
 | 
						|
  DetachPresShell();
 | 
						|
 | 
						|
  Destroy();
 | 
						|
}
 | 
						|
 | 
						|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(nsPresContext)
 | 
						|
  NS_INTERFACE_MAP_ENTRY(nsISupports)
 | 
						|
NS_INTERFACE_MAP_END
 | 
						|
 | 
						|
NS_IMPL_CYCLE_COLLECTING_ADDREF(nsPresContext)
 | 
						|
NS_IMPL_CYCLE_COLLECTING_RELEASE_WITH_LAST_RELEASE(nsPresContext, LastRelease())
 | 
						|
 | 
						|
void nsPresContext::LastRelease() {
 | 
						|
  if (mMissingFonts) {
 | 
						|
    mMissingFonts->Clear();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
NS_IMPL_CYCLE_COLLECTION_CLASS(nsPresContext)
 | 
						|
 | 
						|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(nsPresContext)
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mAnimationEventDispatcher);
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mDocument);
 | 
						|
  // NS_IMPL_CYCLE_COLLECTION_TRAVERSE_RAWPTR(mDeviceContext); // not xpcom
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mEffectCompositor);
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mEventManager);
 | 
						|
  // NS_IMPL_CYCLE_COLLECTION_TRAVERSE_RAWPTR(mLanguage); // an atom
 | 
						|
 | 
						|
  // NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTheme); // a service
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mPrintSettings);
 | 
						|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
 | 
						|
 | 
						|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(nsPresContext)
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_UNLINK(mAnimationEventDispatcher);
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_UNLINK(mDocument);
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_UNLINK(mDeviceContext);  // worth bothering?
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_UNLINK(mEffectCompositor);
 | 
						|
  // NS_RELEASE(tmp->mLanguage); // an atom
 | 
						|
  // NS_IMPL_CYCLE_COLLECTION_UNLINK(mTheme); // a service
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_UNLINK(mPrintSettings);
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_UNLINK_WEAK_PTR
 | 
						|
 | 
						|
  tmp->Destroy();
 | 
						|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
 | 
						|
 | 
						|
bool nsPresContext::IsChrome() const {
 | 
						|
  return Document()->IsInChromeDocShell();
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::GetUserPreferences() {
 | 
						|
  if (!GetPresShell()) {
 | 
						|
    // No presshell means nothing to do here.  We'll do this when we
 | 
						|
    // get a presshell.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  PreferenceSheet::EnsureInitialized();
 | 
						|
 | 
						|
  mSendAfterPaintToContent = Preferences::GetBool(
 | 
						|
      "dom.send_after_paint_to_content", mSendAfterPaintToContent);
 | 
						|
 | 
						|
  mPrefScrollbarSide = Preferences::GetInt("layout.scrollbar.side");
 | 
						|
 | 
						|
  Document()->SetMayNeedFontPrefsUpdate();
 | 
						|
 | 
						|
  // * image animation
 | 
						|
  nsAutoCString animatePref;
 | 
						|
  Preferences::GetCString("image.animation_mode", animatePref);
 | 
						|
  if (animatePref.EqualsLiteral("normal"))
 | 
						|
    mImageAnimationModePref = imgIContainer::kNormalAnimMode;
 | 
						|
  else if (animatePref.EqualsLiteral("none"))
 | 
						|
    mImageAnimationModePref = imgIContainer::kDontAnimMode;
 | 
						|
  else if (animatePref.EqualsLiteral("once"))
 | 
						|
    mImageAnimationModePref = imgIContainer::kLoopOnceAnimMode;
 | 
						|
  else  // dynamic change to invalid value should act like it does initially
 | 
						|
    mImageAnimationModePref = imgIContainer::kNormalAnimMode;
 | 
						|
 | 
						|
  uint32_t bidiOptions = GetBidi();
 | 
						|
 | 
						|
  mPrefBidiDirection = StaticPrefs::bidi_direction();
 | 
						|
  SET_BIDI_OPTION_DIRECTION(bidiOptions, mPrefBidiDirection);
 | 
						|
  SET_BIDI_OPTION_TEXTTYPE(bidiOptions, StaticPrefs::bidi_texttype());
 | 
						|
  SET_BIDI_OPTION_NUMERAL(bidiOptions, StaticPrefs::bidi_numeral());
 | 
						|
 | 
						|
  // We don't need to force reflow: either we are initializing a new
 | 
						|
  // prescontext or we are being called from PreferenceChanged()
 | 
						|
  // which triggers a reflow anyway.
 | 
						|
  SetBidi(bidiOptions);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::InvalidatePaintedLayers() {
 | 
						|
  if (!mPresShell) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  if (nsIFrame* rootFrame = mPresShell->GetRootFrame()) {
 | 
						|
    // FrameLayerBuilder caches invalidation-related values that depend on the
 | 
						|
    // appunits-per-dev-pixel ratio, so ensure that all PaintedLayer drawing
 | 
						|
    // is completely flushed.
 | 
						|
    rootFrame->InvalidateFrameSubtree();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::AppUnitsPerDevPixelChanged() {
 | 
						|
  int32_t oldAppUnitsPerDevPixel = mCurAppUnitsPerDevPixel;
 | 
						|
 | 
						|
  InvalidatePaintedLayers();
 | 
						|
 | 
						|
  FlushFontCache();
 | 
						|
 | 
						|
  MediaFeatureValuesChanged(
 | 
						|
      {RestyleHint::RecascadeSubtree(), NS_STYLE_HINT_REFLOW,
 | 
						|
       MediaFeatureChangeReason::ResolutionChange},
 | 
						|
      MediaFeatureChangePropagation::JustThisDocument);
 | 
						|
 | 
						|
  mCurAppUnitsPerDevPixel = mDeviceContext->AppUnitsPerDevPixel();
 | 
						|
 | 
						|
#ifdef ACCESSIBILITY
 | 
						|
  if (mCurAppUnitsPerDevPixel != oldAppUnitsPerDevPixel) {
 | 
						|
    if (nsAccessibilityService* accService = GetAccService()) {
 | 
						|
      accService->NotifyOfDevPixelRatioChange(mPresShell,
 | 
						|
                                              mCurAppUnitsPerDevPixel);
 | 
						|
    }
 | 
						|
  }
 | 
						|
#endif
 | 
						|
 | 
						|
  // Recompute the size for vh units since it's changed by the dynamic toolbar
 | 
						|
  // max height which is stored in screen coord.
 | 
						|
  if (IsRootContentDocumentCrossProcess()) {
 | 
						|
    AdjustSizeForViewportUnits();
 | 
						|
  }
 | 
						|
 | 
						|
  // nsSubDocumentFrame uses a AppUnitsPerDevPixel difference between parent and
 | 
						|
  // child document to determine if it needs to build a nsDisplayZoom item. So
 | 
						|
  // if we that changes then we need to invalidate the subdoc frame so that
 | 
						|
  // item gets created/removed.
 | 
						|
  if (mPresShell) {
 | 
						|
    if (nsIFrame* frame = mPresShell->GetRootFrame()) {
 | 
						|
      frame = nsLayoutUtils::GetCrossDocParentFrameInProcess(frame);
 | 
						|
      if (frame) {
 | 
						|
        int32_t parentAPD = frame->PresContext()->AppUnitsPerDevPixel();
 | 
						|
        if ((parentAPD == oldAppUnitsPerDevPixel) !=
 | 
						|
            (parentAPD == mCurAppUnitsPerDevPixel)) {
 | 
						|
          frame->InvalidateFrame();
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // We would also have to look at all of our child subdocuments but the
 | 
						|
  // InvalidatePaintedLayers call above calls InvalidateFrameSubtree which
 | 
						|
  // would invalidate all subdocument frames already.
 | 
						|
}
 | 
						|
 | 
						|
// static
 | 
						|
void nsPresContext::PreferenceChanged(const char* aPrefName, void* aSelf) {
 | 
						|
  static_cast<nsPresContext*>(aSelf)->PreferenceChanged(aPrefName);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::PreferenceChanged(const char* aPrefName) {
 | 
						|
  if (!mPresShell) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  nsDependentCString prefName(aPrefName);
 | 
						|
  if (prefName.EqualsLiteral("layout.css.dpi") ||
 | 
						|
      prefName.EqualsLiteral("layout.css.devPixelsPerPx")) {
 | 
						|
    int32_t oldAppUnitsPerDevPixel = mDeviceContext->AppUnitsPerDevPixel();
 | 
						|
    // We need to assume the DPI changes, since `mDeviceContext` is shared with
 | 
						|
    // other documents, and we'd need to save the return value of the first call
 | 
						|
    // for all of them.
 | 
						|
    Unused << mDeviceContext->CheckDPIChange();
 | 
						|
    OwningNonNull<mozilla::PresShell> presShell(*mPresShell);
 | 
						|
    // Re-fetch the view manager's window dimensions in case there's a
 | 
						|
    // deferred resize which hasn't affected our mVisibleArea yet
 | 
						|
    nscoord oldWidthAppUnits, oldHeightAppUnits;
 | 
						|
    RefPtr<nsViewManager> vm = presShell->GetViewManager();
 | 
						|
    if (!vm) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    vm->GetWindowDimensions(&oldWidthAppUnits, &oldHeightAppUnits);
 | 
						|
    float oldWidthDevPixels = oldWidthAppUnits / oldAppUnitsPerDevPixel;
 | 
						|
    float oldHeightDevPixels = oldHeightAppUnits / oldAppUnitsPerDevPixel;
 | 
						|
 | 
						|
    UIResolutionChangedInternal();
 | 
						|
 | 
						|
    nscoord width = NSToCoordRound(oldWidthDevPixels * AppUnitsPerDevPixel());
 | 
						|
    nscoord height = NSToCoordRound(oldHeightDevPixels * AppUnitsPerDevPixel());
 | 
						|
    vm->SetWindowDimensions(width, height);
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (StringBeginsWith(prefName, "browser.viewport."_ns) ||
 | 
						|
      StringBeginsWith(prefName, "font.size.inflation."_ns) ||
 | 
						|
      prefName.EqualsLiteral("dom.meta-viewport.enabled")) {
 | 
						|
    mPresShell->MaybeReflowForInflationScreenSizeChange();
 | 
						|
  }
 | 
						|
 | 
						|
  auto changeHint = nsChangeHint{0};
 | 
						|
  auto restyleHint = RestyleHint{0};
 | 
						|
  // Changing any of these potentially changes the value of @media
 | 
						|
  // (prefers-contrast).
 | 
						|
  // The layout.css.prefers-contrast.enabled pref itself is not handled here,
 | 
						|
  // because that pref doesn't just affect the "live" value of the media query;
 | 
						|
  // it affects whether it is parsed at all.
 | 
						|
  if (prefName.EqualsLiteral("browser.display.document_color_use") ||
 | 
						|
      prefName.EqualsLiteral("browser.display.foreground_color") ||
 | 
						|
      prefName.EqualsLiteral("browser.display.background_color")) {
 | 
						|
    MediaFeatureValuesChanged({MediaFeatureChangeReason::PreferenceChange},
 | 
						|
                              MediaFeatureChangePropagation::JustThisDocument);
 | 
						|
  }
 | 
						|
  if (prefName.EqualsLiteral(GFX_MISSING_FONTS_NOTIFY_PREF)) {
 | 
						|
    if (StaticPrefs::gfx_missing_fonts_notify()) {
 | 
						|
      if (!mMissingFonts) {
 | 
						|
        mMissingFonts = MakeUnique<gfxMissingFontRecorder>();
 | 
						|
        // trigger reflow to detect missing fonts on the current page
 | 
						|
        changeHint |= NS_STYLE_HINT_REFLOW;
 | 
						|
      }
 | 
						|
    } else {
 | 
						|
      if (mMissingFonts) {
 | 
						|
        mMissingFonts->Clear();
 | 
						|
      }
 | 
						|
      mMissingFonts = nullptr;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (StringBeginsWith(prefName, "font."_ns) ||
 | 
						|
      // Changes to font family preferences don't change anything in the
 | 
						|
      // computed style data, so the style system won't generate a reflow hint
 | 
						|
      // for us.  We need to do that manually.
 | 
						|
      prefName.EqualsLiteral("intl.accept_languages") ||
 | 
						|
      // Changes to bidi prefs need to trigger a reflow (see bug 443629)
 | 
						|
      StringBeginsWith(prefName, "bidi."_ns) ||
 | 
						|
      // Changes to font_rendering prefs need to trigger a reflow
 | 
						|
      StringBeginsWith(prefName, "gfx.font_rendering."_ns)) {
 | 
						|
    changeHint |= NS_STYLE_HINT_REFLOW;
 | 
						|
    if (StyleSet()->UsesFontMetrics()) {
 | 
						|
      restyleHint |= RestyleHint::RecascadeSubtree();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (prefName.EqualsLiteral(
 | 
						|
          "layout.css.text-transform.uppercase-eszett.enabled")) {
 | 
						|
    changeHint |= NS_STYLE_HINT_REFLOW;
 | 
						|
  }
 | 
						|
 | 
						|
  if (PreferenceSheet::AffectedByPref(prefName)) {
 | 
						|
    restyleHint |= RestyleHint::RestyleSubtree();
 | 
						|
    PreferenceSheet::Refresh();
 | 
						|
  }
 | 
						|
 | 
						|
  // Same, this just frees a bunch of memory.
 | 
						|
  StaticPresData::Get()->InvalidateFontPrefs();
 | 
						|
  Document()->SetMayNeedFontPrefsUpdate();
 | 
						|
 | 
						|
  // Initialize our state from the user preferences.
 | 
						|
  GetUserPreferences();
 | 
						|
 | 
						|
  FlushFontCache();
 | 
						|
  if (UpdateFontVisibility()) {
 | 
						|
    changeHint |= NS_STYLE_HINT_REFLOW;
 | 
						|
  }
 | 
						|
 | 
						|
  // Preferences require rerunning selector matching because we rebuild
 | 
						|
  // the pref style sheet for some preference changes.
 | 
						|
  if (changeHint || restyleHint) {
 | 
						|
    RebuildAllStyleData(changeHint, restyleHint);
 | 
						|
  }
 | 
						|
 | 
						|
  InvalidatePaintedLayers();
 | 
						|
}
 | 
						|
 | 
						|
nsresult nsPresContext::Init(nsDeviceContext* aDeviceContext) {
 | 
						|
  NS_ASSERTION(!mInitialized, "attempt to reinit pres context");
 | 
						|
  NS_ENSURE_ARG(aDeviceContext);
 | 
						|
 | 
						|
  mDeviceContext = aDeviceContext;
 | 
						|
 | 
						|
  // In certain rare cases (such as changing page mode), we tear down layout
 | 
						|
  // state and re-initialize a new prescontext for a document. Given that we
 | 
						|
  // hang style state off the DOM, we detect that re-initialization case and
 | 
						|
  // lazily drop the servo data. We don't do this eagerly during layout teardown
 | 
						|
  // because that would incur an extra whole-tree traversal that's unnecessary
 | 
						|
  // most of the time.
 | 
						|
  //
 | 
						|
  // FIXME(emilio): I'm pretty sure this doesn't happen after bug 1414999.
 | 
						|
  Element* root = mDocument->GetRootElement();
 | 
						|
  if (root && root->HasServoData()) {
 | 
						|
    RestyleManager::ClearServoDataFromSubtree(root);
 | 
						|
  }
 | 
						|
 | 
						|
  if (mDeviceContext->SetFullZoom(mFullZoom)) {
 | 
						|
    FlushFontCache();
 | 
						|
  }
 | 
						|
  mCurAppUnitsPerDevPixel = mDeviceContext->AppUnitsPerDevPixel();
 | 
						|
 | 
						|
  mEventManager = new mozilla::EventStateManager();
 | 
						|
 | 
						|
  mAnimationEventDispatcher = new mozilla::AnimationEventDispatcher(this);
 | 
						|
  mEffectCompositor = new mozilla::EffectCompositor(this);
 | 
						|
  mTransitionManager = MakeUnique<nsTransitionManager>(this);
 | 
						|
  mAnimationManager = MakeUnique<nsAnimationManager>(this);
 | 
						|
  mTimelineManager = MakeUnique<mozilla::TimelineManager>(this);
 | 
						|
 | 
						|
  if (mDocument->GetDisplayDocument()) {
 | 
						|
    NS_ASSERTION(mDocument->GetDisplayDocument()->GetPresContext(),
 | 
						|
                 "Why are we being initialized?");
 | 
						|
    mRefreshDriver =
 | 
						|
        mDocument->GetDisplayDocument()->GetPresContext()->RefreshDriver();
 | 
						|
  } else {
 | 
						|
    dom::Document* parent = mDocument->GetInProcessParentDocument();
 | 
						|
    // Unfortunately, sometimes |parent| here has no presshell because
 | 
						|
    // printing screws up things.  Assert that in other cases it does,
 | 
						|
    // but whenever the shell is null just fall back on using our own
 | 
						|
    // refresh driver.
 | 
						|
    NS_ASSERTION(
 | 
						|
        !parent || mDocument->IsStaticDocument() || parent->GetPresShell(),
 | 
						|
        "How did we end up with a presshell if our parent doesn't "
 | 
						|
        "have one?");
 | 
						|
    if (parent && parent->GetPresContext()) {
 | 
						|
      // XXX the document can change in AttachPresShell, does this work?
 | 
						|
      dom::BrowsingContext* browsingContext = mDocument->GetBrowsingContext();
 | 
						|
      if (browsingContext && !browsingContext->IsTop()) {
 | 
						|
        Element* containingElement = mDocument->GetEmbedderElement();
 | 
						|
        if (!containingElement->IsXULElement() ||
 | 
						|
            !containingElement->HasAttr(nsGkAtoms::forceOwnRefreshDriver)) {
 | 
						|
          mRefreshDriver = parent->GetPresContext()->RefreshDriver();
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    if (!mRefreshDriver) {
 | 
						|
      mRefreshDriver = new nsRefreshDriver(this);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Register callbacks so we're notified when the preferences change
 | 
						|
  Preferences::RegisterPrefixCallbacks(nsPresContext::PreferenceChanged,
 | 
						|
                                       gPrefixCallbackPrefs, this);
 | 
						|
  Preferences::RegisterCallbacks(nsPresContext::PreferenceChanged,
 | 
						|
                                 gExactCallbackPrefs, this);
 | 
						|
 | 
						|
  nsresult rv = mEventManager->Init();
 | 
						|
  NS_ENSURE_SUCCESS(rv, rv);
 | 
						|
 | 
						|
  mEventManager->SetPresContext(this);
 | 
						|
 | 
						|
#if defined(MOZ_WIDGET_ANDROID)
 | 
						|
  if (IsRootContentDocumentCrossProcess() &&
 | 
						|
      MOZ_LIKELY(
 | 
						|
          !Preferences::HasUserValue("layout.dynamic-toolbar-max-height"))) {
 | 
						|
    if (BrowserChild* browserChild =
 | 
						|
            BrowserChild::GetFrom(mDocument->GetDocShell())) {
 | 
						|
      mDynamicToolbarMaxHeight = browserChild->GetDynamicToolbarMaxHeight();
 | 
						|
      mDynamicToolbarHeight = mDynamicToolbarMaxHeight;
 | 
						|
    }
 | 
						|
  }
 | 
						|
#endif
 | 
						|
 | 
						|
#ifdef DEBUG
 | 
						|
  mInitialized = true;
 | 
						|
#endif
 | 
						|
 | 
						|
  return NS_OK;
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::UpdateFontVisibility() {
 | 
						|
  FontVisibility oldValue = mFontVisibility;
 | 
						|
 | 
						|
  /*
 | 
						|
   * Expected behavior in order of precedence:
 | 
						|
   *  1  Chrome Rules give User Level (3)
 | 
						|
   *  2  RFP gives Highest Level (1 aka Base)
 | 
						|
   *  3  An RFPTarget of Base gives Base Level (1)
 | 
						|
   *  4  An RFPTarget of LangPack gives LangPack Level (2)
 | 
						|
   *  5  The value of the Standard Font Visibility Pref
 | 
						|
   *
 | 
						|
   * If the ETP toggle is disabled (aka
 | 
						|
   * ContentBlockingAllowList::Check is true), it will only override 3-5,
 | 
						|
   * not rules 1 or 2.
 | 
						|
   */
 | 
						|
 | 
						|
  // Rule 1: Allow all font access for privileged contexts, including
 | 
						|
  // chrome and devtools contexts.
 | 
						|
  if (Document()->ChromeRulesEnabled()) {
 | 
						|
    mFontVisibility = FontVisibility::User;
 | 
						|
    return mFontVisibility != oldValue;
 | 
						|
  }
 | 
						|
 | 
						|
  // Is this a private browsing context?
 | 
						|
  bool isPrivate = false;
 | 
						|
  if (nsCOMPtr<nsILoadContext> loadContext = mDocument->GetLoadContext()) {
 | 
						|
    isPrivate = loadContext->UsePrivateBrowsing();
 | 
						|
  }
 | 
						|
 | 
						|
  int32_t level;
 | 
						|
  // Rule 3
 | 
						|
  if (mDocument->ShouldResistFingerprinting(
 | 
						|
          RFPTarget::FontVisibilityBaseSystem)) {
 | 
						|
    // Rule 2: Check RFP pref
 | 
						|
    // This is inside Rule 3 in case this document is exempted from RFP.
 | 
						|
    // But if it is not exempted, and RFP is enabled, we return immediately
 | 
						|
    // to prevent the override below from occurring.
 | 
						|
    if (nsRFPService::IsRFPPrefEnabled(isPrivate)) {
 | 
						|
      mFontVisibility = FontVisibility::Base;
 | 
						|
      return mFontVisibility != oldValue;
 | 
						|
    }
 | 
						|
 | 
						|
    level = int32_t(FontVisibility::Base);
 | 
						|
  }
 | 
						|
  // Rule 4
 | 
						|
  else if (mDocument->ShouldResistFingerprinting(
 | 
						|
               RFPTarget::FontVisibilityLangPack)) {
 | 
						|
    level = int32_t(FontVisibility::LangPack);
 | 
						|
  }
 | 
						|
  // Rule 5
 | 
						|
  else {
 | 
						|
    level = StaticPrefs::layout_css_font_visibility();
 | 
						|
  }
 | 
						|
 | 
						|
  // Override Rules 3-5 Only: Determine if the user has exempted the
 | 
						|
  // domain from tracking protections, if so, use the default value.
 | 
						|
  if (level != StaticPrefs::layout_css_font_visibility() &&
 | 
						|
      ContentBlockingAllowList::Check(mDocument->CookieJarSettings())) {
 | 
						|
    level = StaticPrefs::layout_css_font_visibility();
 | 
						|
  }
 | 
						|
 | 
						|
  // Clamp result to the valid range of levels.
 | 
						|
  level = std::max(std::min(level, int32_t(FontVisibility::User)),
 | 
						|
                   int32_t(FontVisibility::Base));
 | 
						|
 | 
						|
  mFontVisibility = FontVisibility(level);
 | 
						|
  return mFontVisibility != oldValue;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::ReportBlockedFontFamilyName(const nsCString& aFamily,
 | 
						|
                                                FontVisibility aVisibility) {
 | 
						|
  if (!mBlockedFonts.EnsureInserted(aFamily)) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  nsAutoString msg;
 | 
						|
  msg.AppendPrintf(
 | 
						|
      "Request for font \"%s\" blocked at visibility level %d (requires %d)\n",
 | 
						|
      aFamily.get(), int(GetFontVisibility()), int(aVisibility));
 | 
						|
  nsContentUtils::ReportToConsoleNonLocalized(msg, nsIScriptError::warningFlag,
 | 
						|
                                              "Security"_ns, mDocument);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::ReportBlockedFontFamily(const fontlist::Family& aFamily) {
 | 
						|
  auto* fontList = gfxPlatformFontList::PlatformFontList()->SharedFontList();
 | 
						|
  const nsCString& name = aFamily.DisplayName().AsString(fontList);
 | 
						|
  ReportBlockedFontFamilyName(name, aFamily.Visibility());
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::ReportBlockedFontFamily(const gfxFontFamily& aFamily) {
 | 
						|
  ReportBlockedFontFamilyName(aFamily.Name(), aFamily.Visibility());
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::InitFontCache() {
 | 
						|
  if (!mFontCache) {
 | 
						|
    mFontCache = new nsFontCache();
 | 
						|
    mFontCache->Init(this);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::UpdateFontCacheUserFonts(gfxUserFontSet* aUserFontSet) {
 | 
						|
  if (mFontCache) {
 | 
						|
    mFontCache->UpdateUserFonts(aUserFontSet);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
already_AddRefed<nsFontMetrics> nsPresContext::GetMetricsFor(
 | 
						|
    const nsFont& aFont, const nsFontMetrics::Params& aParams) {
 | 
						|
  InitFontCache();
 | 
						|
  return mFontCache->GetMetricsFor(aFont, aParams);
 | 
						|
}
 | 
						|
 | 
						|
nsresult nsPresContext::FlushFontCache() {
 | 
						|
  if (mFontCache) {
 | 
						|
    mFontCache->Flush();
 | 
						|
  }
 | 
						|
  return NS_OK;
 | 
						|
}
 | 
						|
 | 
						|
nsresult nsPresContext::FontMetricsDeleted(const nsFontMetrics* aFontMetrics) {
 | 
						|
  if (mFontCache) {
 | 
						|
    mFontCache->FontMetricsDeleted(aFontMetrics);
 | 
						|
  }
 | 
						|
  return NS_OK;
 | 
						|
}
 | 
						|
 | 
						|
// Note: We don't hold a reference on the shell; it has a reference to
 | 
						|
// us
 | 
						|
void nsPresContext::AttachPresShell(mozilla::PresShell* aPresShell) {
 | 
						|
  MOZ_ASSERT(!mPresShell);
 | 
						|
  mPresShell = aPresShell;
 | 
						|
 | 
						|
  mRestyleManager = MakeUnique<mozilla::RestyleManager>(this);
 | 
						|
 | 
						|
  // Since CounterStyleManager is also the name of a method of
 | 
						|
  // nsPresContext, it is necessary to prefix the class with the mozilla
 | 
						|
  // namespace here.
 | 
						|
  mCounterStyleManager = new mozilla::CounterStyleManager(this);
 | 
						|
 | 
						|
  dom::Document* doc = mPresShell->GetDocument();
 | 
						|
  MOZ_ASSERT(doc);
 | 
						|
  // Have to update PresContext's mDocument before calling any other methods.
 | 
						|
  mDocument = doc;
 | 
						|
 | 
						|
  LookAndFeel::HandleGlobalThemeChange();
 | 
						|
 | 
						|
  // Initialize our state from the user preferences, now that we
 | 
						|
  // have a presshell, and hence a document.
 | 
						|
  GetUserPreferences();
 | 
						|
 | 
						|
  EnsureTheme();
 | 
						|
 | 
						|
  nsIURI* docURI = doc->GetDocumentURI();
 | 
						|
 | 
						|
  if (IsDynamic() && docURI) {
 | 
						|
    if (!docURI->SchemeIs("chrome") && !docURI->SchemeIs("resource"))
 | 
						|
      mImageAnimationMode = mImageAnimationModePref;
 | 
						|
    else
 | 
						|
      mImageAnimationMode = imgIContainer::kNormalAnimMode;
 | 
						|
  }
 | 
						|
 | 
						|
  UpdateCharSet(doc->GetDocumentCharacterSet());
 | 
						|
}
 | 
						|
 | 
						|
Maybe<ColorScheme> nsPresContext::GetOverriddenOrEmbedderColorScheme() const {
 | 
						|
  if (Medium() == nsGkAtoms::print) {
 | 
						|
    return Some(ColorScheme::Light);
 | 
						|
  }
 | 
						|
 | 
						|
  switch (mOverriddenOrEmbedderColorScheme) {
 | 
						|
    case dom::PrefersColorSchemeOverride::Dark:
 | 
						|
      return Some(ColorScheme::Dark);
 | 
						|
    case dom::PrefersColorSchemeOverride::Light:
 | 
						|
      return Some(ColorScheme::Light);
 | 
						|
    case dom::PrefersColorSchemeOverride::None:
 | 
						|
    case dom::PrefersColorSchemeOverride::EndGuard_:
 | 
						|
      break;
 | 
						|
  }
 | 
						|
 | 
						|
  return Nothing();
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SetColorSchemeOverride(
 | 
						|
    PrefersColorSchemeOverride aOverride) {
 | 
						|
  auto oldScheme = mDocument->PreferredColorScheme();
 | 
						|
 | 
						|
  mOverriddenOrEmbedderColorScheme = aOverride;
 | 
						|
 | 
						|
  if (mDocument->PreferredColorScheme() != oldScheme) {
 | 
						|
    MediaFeatureValuesChanged(
 | 
						|
        MediaFeatureChange::ForPreferredColorSchemeChange(),
 | 
						|
        MediaFeatureChangePropagation::JustThisDocument);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::RecomputeBrowsingContextDependentData() {
 | 
						|
  MOZ_ASSERT(mDocument);
 | 
						|
  dom::Document* doc = mDocument;
 | 
						|
  // Resource documents inherit all this state from their display document.
 | 
						|
  while (dom::Document* outer = doc->GetDisplayDocument()) {
 | 
						|
    doc = outer;
 | 
						|
  }
 | 
						|
  auto* browsingContext = doc->GetBrowsingContext();
 | 
						|
  if (!browsingContext) {
 | 
						|
    // This can legitimately happen for e.g. SVG images. Those just get scaled
 | 
						|
    // as a result of the zoom on the embedder document so it doesn't really
 | 
						|
    // matter... Medium also doesn't affect those.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  if (!IsPrintingOrPrintPreview()) {
 | 
						|
    auto systemZoom = LookAndFeel::SystemZoomSettings();
 | 
						|
    SetFullZoom(browsingContext->FullZoom() * systemZoom.mFullZoom);
 | 
						|
    SetTextZoom(browsingContext->TextZoom() * systemZoom.mTextZoom);
 | 
						|
    SetOverrideDPPX(browsingContext->OverrideDPPX());
 | 
						|
  }
 | 
						|
 | 
						|
  auto* top = browsingContext->Top();
 | 
						|
  SetColorSchemeOverride([&] {
 | 
						|
    auto overriden = top->PrefersColorSchemeOverride();
 | 
						|
    if (overriden != PrefersColorSchemeOverride::None) {
 | 
						|
      return overriden;
 | 
						|
    }
 | 
						|
    if (!StaticPrefs::
 | 
						|
            layout_css_iframe_embedder_prefers_color_scheme_content_enabled()) {
 | 
						|
      return top->GetEmbedderColorSchemes().mPreferred;
 | 
						|
    }
 | 
						|
    return browsingContext->GetEmbedderColorSchemes().mPreferred;
 | 
						|
  }());
 | 
						|
 | 
						|
  SetInRDMPane(top->GetInRDMPane());
 | 
						|
 | 
						|
  if (doc == mDocument) {
 | 
						|
    // Medium doesn't apply to resource documents, etc.
 | 
						|
    RefPtr<nsAtom> mediumToEmulate;
 | 
						|
    if (MOZ_UNLIKELY(!top->GetMediumOverride().IsEmpty())) {
 | 
						|
      nsAutoString lower;
 | 
						|
      nsContentUtils::ASCIIToLower(top->GetMediumOverride(), lower);
 | 
						|
      mediumToEmulate = NS_Atomize(lower);
 | 
						|
    }
 | 
						|
    EmulateMedium(mediumToEmulate);
 | 
						|
  }
 | 
						|
 | 
						|
  mDocument->EnumerateExternalResources([](dom::Document& aSubResource) {
 | 
						|
    if (nsPresContext* subResourcePc = aSubResource.GetPresContext()) {
 | 
						|
      subResourcePc->RecomputeBrowsingContextDependentData();
 | 
						|
    }
 | 
						|
    return CallState::Continue;
 | 
						|
  });
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::DetachPresShell() {
 | 
						|
  // The counter style manager's destructor needs to deallocate with the
 | 
						|
  // presshell arena. Disconnect it before nulling out the shell.
 | 
						|
  //
 | 
						|
  // XXXbholley: Given recent refactorings, it probably makes more sense to
 | 
						|
  // just null our mPresShell at the bottom of this function. I'm leaving it
 | 
						|
  // this way to preserve the old ordering, but I doubt anything would break.
 | 
						|
  if (mCounterStyleManager) {
 | 
						|
    mCounterStyleManager->Disconnect();
 | 
						|
    mCounterStyleManager = nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  mPresShell = nullptr;
 | 
						|
 | 
						|
  CancelManagedPostRefreshObservers();
 | 
						|
 | 
						|
  if (mAnimationEventDispatcher) {
 | 
						|
    mAnimationEventDispatcher->Disconnect();
 | 
						|
    mAnimationEventDispatcher = nullptr;
 | 
						|
  }
 | 
						|
  if (mEffectCompositor) {
 | 
						|
    mEffectCompositor->Disconnect();
 | 
						|
    mEffectCompositor = nullptr;
 | 
						|
  }
 | 
						|
  if (mTransitionManager) {
 | 
						|
    mTransitionManager->Disconnect();
 | 
						|
    mTransitionManager = nullptr;
 | 
						|
  }
 | 
						|
  if (mAnimationManager) {
 | 
						|
    mAnimationManager->Disconnect();
 | 
						|
    mAnimationManager = nullptr;
 | 
						|
  }
 | 
						|
  if (mTimelineManager) {
 | 
						|
    mTimelineManager->Disconnect();
 | 
						|
    mTimelineManager = nullptr;
 | 
						|
  }
 | 
						|
  if (mRestyleManager) {
 | 
						|
    mRestyleManager->Disconnect();
 | 
						|
    mRestyleManager = nullptr;
 | 
						|
  }
 | 
						|
  if (mRefreshDriver && mRefreshDriver->GetPresContext() == this) {
 | 
						|
    mRefreshDriver->Disconnect();
 | 
						|
    // Can't null out the refresh driver here.
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
struct QueryContainerState {
 | 
						|
  nsSize mSize;
 | 
						|
  WritingMode mWm;
 | 
						|
 | 
						|
  nscoord GetInlineSize() const { return LogicalSize(mWm, mSize).ISize(mWm); }
 | 
						|
 | 
						|
  bool Changed(const QueryContainerState& aNewState, StyleContainerType aType) {
 | 
						|
    switch (aType) {
 | 
						|
      case StyleContainerType::Normal:
 | 
						|
        break;
 | 
						|
      case StyleContainerType::Size:
 | 
						|
        return mSize != aNewState.mSize;
 | 
						|
      case StyleContainerType::InlineSize:
 | 
						|
        return GetInlineSize() != aNewState.GetInlineSize();
 | 
						|
    }
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
};
 | 
						|
NS_DECLARE_FRAME_PROPERTY_DELETABLE(ContainerState, QueryContainerState);
 | 
						|
 | 
						|
void nsPresContext::RegisterContainerQueryFrame(nsIFrame* aFrame) {
 | 
						|
  mContainerQueryFrames.Add(aFrame);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::UnregisterContainerQueryFrame(nsIFrame* aFrame) {
 | 
						|
  mContainerQueryFrames.Remove(aFrame);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::FinishedContainerQueryUpdate() {
 | 
						|
  mUpdatedContainerQueryContents.Clear();
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::UpdateContainerQueryStyles() {
 | 
						|
  if (mContainerQueryFrames.IsEmpty()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  AUTO_PROFILER_LABEL_RELEVANT_FOR_JS("Container Query Styles Update", LAYOUT);
 | 
						|
  AUTO_PROFILER_MARKER_TEXT("UpdateContainerQueryStyles", LAYOUT, {}, ""_ns);
 | 
						|
 | 
						|
  PresShell()->DoFlushLayout(/* aInterruptible = */ false);
 | 
						|
 | 
						|
  AutoTArray<nsIFrame*, 8> framesToUpdate;
 | 
						|
 | 
						|
  bool anyChanged = false;
 | 
						|
  for (nsIFrame* frame : mContainerQueryFrames.IterFromShallowest()) {
 | 
						|
    MOZ_ASSERT(frame->IsPrimaryFrame());
 | 
						|
 | 
						|
    auto type = frame->StyleDisplay()->mContainerType;
 | 
						|
    MOZ_ASSERT(type != StyleContainerType::Normal,
 | 
						|
               "Non-container frames shouldn't be in this type");
 | 
						|
 | 
						|
    const QueryContainerState newState{frame->GetSize(),
 | 
						|
                                       frame->GetWritingMode()};
 | 
						|
    QueryContainerState* oldState = frame->GetProperty(ContainerState());
 | 
						|
 | 
						|
    const bool changed = !oldState || oldState->Changed(newState, type);
 | 
						|
 | 
						|
    // Make sure to update the state regardless. It's cheap and it keeps tracks
 | 
						|
    // of both axes correctly even if only one axis is contained.
 | 
						|
    if (oldState) {
 | 
						|
      *oldState = newState;
 | 
						|
    } else {
 | 
						|
      frame->SetProperty(ContainerState(), new QueryContainerState(newState));
 | 
						|
    }
 | 
						|
 | 
						|
    if (!changed) {
 | 
						|
      continue;
 | 
						|
    }
 | 
						|
 | 
						|
    const bool updatingAncestor = [&] {
 | 
						|
      for (nsIFrame* f : framesToUpdate) {
 | 
						|
        if (nsLayoutUtils::IsProperAncestorFrame(f, frame)) {
 | 
						|
          return true;
 | 
						|
        }
 | 
						|
      }
 | 
						|
      return false;
 | 
						|
    }();
 | 
						|
 | 
						|
    if (updatingAncestor) {
 | 
						|
      // We're going to update an ancestor container of this frame already,
 | 
						|
      // avoid updating this one too until all our ancestor containers are
 | 
						|
      // updated.
 | 
						|
      continue;
 | 
						|
    }
 | 
						|
 | 
						|
    // To prevent unstable layout, only update once per-element per-flush.
 | 
						|
    if (NS_WARN_IF(!mUpdatedContainerQueryContents.EnsureInserted(
 | 
						|
            frame->GetContent()))) {
 | 
						|
      continue;
 | 
						|
    }
 | 
						|
 | 
						|
    framesToUpdate.AppendElement(frame);
 | 
						|
 | 
						|
    // TODO(emilio): More fine-grained invalidation rather than invalidating the
 | 
						|
    // whole subtree, probably!
 | 
						|
    RestyleManager()->PostRestyleEvent(frame->GetContent()->AsElement(),
 | 
						|
                                       RestyleHint::RestyleSubtree(),
 | 
						|
                                       nsChangeHint(0));
 | 
						|
    anyChanged = true;
 | 
						|
  }
 | 
						|
  return anyChanged;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::DocumentCharSetChanged(NotNull<const Encoding*> aCharSet) {
 | 
						|
  UpdateCharSet(aCharSet);
 | 
						|
  FlushFontCache();
 | 
						|
 | 
						|
  // If a document contains one or more <script> elements, frame construction
 | 
						|
  // might happen earlier than the UpdateCharSet(), so we need to restyle
 | 
						|
  // descendants to make their style data up-to-date.
 | 
						|
  //
 | 
						|
  // FIXME(emilio): Revisit whether this is true after bug 1438911.
 | 
						|
  RebuildAllStyleData(NS_STYLE_HINT_REFLOW, RestyleHint::RecascadeSubtree());
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::UpdateCharSet(NotNull<const Encoding*> aCharSet) {
 | 
						|
  switch (GET_BIDI_OPTION_TEXTTYPE(GetBidi())) {
 | 
						|
    case IBMBIDI_TEXTTYPE_LOGICAL:
 | 
						|
      SetVisualMode(false);
 | 
						|
      break;
 | 
						|
 | 
						|
    case IBMBIDI_TEXTTYPE_VISUAL:
 | 
						|
      SetVisualMode(true);
 | 
						|
      break;
 | 
						|
 | 
						|
    case IBMBIDI_TEXTTYPE_CHARSET:
 | 
						|
    default:
 | 
						|
      SetVisualMode(IsVisualCharset(aCharSet));
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
nsPresContext* nsPresContext::GetParentPresContext() const {
 | 
						|
  mozilla::PresShell* presShell = GetPresShell();
 | 
						|
  if (presShell) {
 | 
						|
    nsViewManager* viewManager = presShell->GetViewManager();
 | 
						|
    if (viewManager) {
 | 
						|
      nsView* view = viewManager->GetRootView();
 | 
						|
      if (view) {
 | 
						|
        view = view->GetParent();  // anonymous inner view
 | 
						|
        if (view) {
 | 
						|
          view = view->GetParent();  // subdocumentframe's view
 | 
						|
          if (view) {
 | 
						|
            nsIFrame* f = view->GetFrame();
 | 
						|
            if (f) {
 | 
						|
              return f->PresContext();
 | 
						|
            }
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return nullptr;
 | 
						|
}
 | 
						|
 | 
						|
nsPresContext* nsPresContext::GetInProcessRootContentDocumentPresContext() {
 | 
						|
  if (IsChrome()) return nullptr;
 | 
						|
  nsPresContext* pc = this;
 | 
						|
  for (;;) {
 | 
						|
    nsPresContext* parent = pc->GetParentPresContext();
 | 
						|
    if (!parent || parent->IsChrome()) return pc;
 | 
						|
    pc = parent;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
nsIWidget* nsPresContext::GetNearestWidget(nsPoint* aOffset) {
 | 
						|
  NS_ENSURE_TRUE(mPresShell, nullptr);
 | 
						|
  nsViewManager* vm = mPresShell->GetViewManager();
 | 
						|
  NS_ENSURE_TRUE(vm, nullptr);
 | 
						|
  nsView* rootView = vm->GetRootView();
 | 
						|
  NS_ENSURE_TRUE(rootView, nullptr);
 | 
						|
  return rootView->GetNearestWidget(aOffset);
 | 
						|
}
 | 
						|
 | 
						|
nsIWidget* nsPresContext::GetRootWidget() const {
 | 
						|
  NS_ENSURE_TRUE(mPresShell, nullptr);
 | 
						|
  nsViewManager* vm = mPresShell->GetViewManager();
 | 
						|
  if (!vm) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
  return vm->GetRootWidget();
 | 
						|
}
 | 
						|
 | 
						|
// We may want to replace this with something faster, maybe caching the root
 | 
						|
// prescontext
 | 
						|
nsRootPresContext* nsPresContext::GetRootPresContext() const {
 | 
						|
  nsPresContext* pc = const_cast<nsPresContext*>(this);
 | 
						|
  for (;;) {
 | 
						|
    nsPresContext* parent = pc->GetParentPresContext();
 | 
						|
    if (!parent) break;
 | 
						|
    pc = parent;
 | 
						|
  }
 | 
						|
  return pc->IsRoot() ? static_cast<nsRootPresContext*>(pc) : nullptr;
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::UserInputEventsAllowed() {
 | 
						|
  MOZ_ASSERT(IsRoot());
 | 
						|
  if (mUserInputEventsAllowed) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  // Special document
 | 
						|
  if (Document()->IsEverInitialDocument()) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mRefreshDriver->IsThrottled()) {
 | 
						|
    MOZ_ASSERT(!mPresShell->IsVisible());
 | 
						|
    // This implies that the BC is not visibile and users can't
 | 
						|
    // interact with it, so we are okay with handling user inputs here.
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mMeasuredTicksSinceLoading <
 | 
						|
      StaticPrefs::dom_input_events_security_minNumTicks()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!StaticPrefs::dom_input_events_security_minTimeElapsedInMS()) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  dom::Document* doc = Document();
 | 
						|
 | 
						|
  MOZ_ASSERT_IF(StaticPrefs::dom_input_events_security_minNumTicks(),
 | 
						|
                doc->GetReadyStateEnum() >= Document::READYSTATE_LOADING);
 | 
						|
 | 
						|
  TimeStamp loadingOrRestoredFromBFCacheTime =
 | 
						|
      doc->GetLoadingOrRestoredFromBFCacheTimeStamp();
 | 
						|
  MOZ_ASSERT(!loadingOrRestoredFromBFCacheTime.IsNull());
 | 
						|
 | 
						|
  TimeDuration elapsed = TimeStamp::Now() - loadingOrRestoredFromBFCacheTime;
 | 
						|
  if (elapsed.ToMilliseconds() >=
 | 
						|
      StaticPrefs::dom_input_events_security_minTimeElapsedInMS()) {
 | 
						|
    mUserInputEventsAllowed = true;
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::MaybeIncreaseMeasuredTicksSinceLoading() {
 | 
						|
  MOZ_ASSERT(IsRoot());
 | 
						|
  if (mMeasuredTicksSinceLoading >=
 | 
						|
      StaticPrefs::dom_input_events_security_minNumTicks()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // We consider READYSTATE_LOADING is the point when the page
 | 
						|
  // becomes interactive
 | 
						|
  if (Document()->GetReadyStateEnum() >= Document::READYSTATE_LOADING ||
 | 
						|
      Document()->IsInitialDocument()) {
 | 
						|
    ++mMeasuredTicksSinceLoading;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mMeasuredTicksSinceLoading <
 | 
						|
      StaticPrefs::dom_input_events_security_minNumTicks()) {
 | 
						|
    // Here we are forcing refresh driver to run because we can't always
 | 
						|
    // guarantee refresh driver will run enough times to meet the minNumTicks
 | 
						|
    // requirement. i.e. about:blank.
 | 
						|
    if (!RefreshDriver()->HasPendingTick()) {
 | 
						|
      RefreshDriver()->InitializeTimer();
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::NeedsMoreTicksForUserInput() const {
 | 
						|
  MOZ_ASSERT(IsRoot());
 | 
						|
  return mMeasuredTicksSinceLoading <
 | 
						|
         StaticPrefs::dom_input_events_security_minNumTicks();
 | 
						|
}
 | 
						|
 | 
						|
// Helper function for setting Anim Mode on image
 | 
						|
static void SetImgAnimModeOnImgReq(imgIRequest* aImgReq, uint16_t aMode) {
 | 
						|
  if (aImgReq) {
 | 
						|
    nsCOMPtr<imgIContainer> imgCon;
 | 
						|
    aImgReq->GetImage(getter_AddRefs(imgCon));
 | 
						|
    if (imgCon) {
 | 
						|
      imgCon->SetAnimationMode(aMode);
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// IMPORTANT: Assumption is that all images for a Presentation
 | 
						|
// have the same Animation Mode (pavlov said this was OK)
 | 
						|
//
 | 
						|
// Walks content and set the animation mode
 | 
						|
// this is a way to turn on/off image animations
 | 
						|
void nsPresContext::SetImgAnimations(nsIContent* aParent, uint16_t aMode) {
 | 
						|
  nsCOMPtr<nsIImageLoadingContent> imgContent(do_QueryInterface(aParent));
 | 
						|
  if (imgContent) {
 | 
						|
    nsCOMPtr<imgIRequest> imgReq;
 | 
						|
    imgContent->GetRequest(nsIImageLoadingContent::CURRENT_REQUEST,
 | 
						|
                           getter_AddRefs(imgReq));
 | 
						|
    SetImgAnimModeOnImgReq(imgReq, aMode);
 | 
						|
  }
 | 
						|
 | 
						|
  for (nsIContent* childContent = aParent->GetFirstChild(); childContent;
 | 
						|
       childContent = childContent->GetNextSibling()) {
 | 
						|
    SetImgAnimations(childContent, aMode);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SetSMILAnimations(dom::Document* aDoc, uint16_t aNewMode,
 | 
						|
                                      uint16_t aOldMode) {
 | 
						|
  if (aDoc->HasAnimationController()) {
 | 
						|
    SMILAnimationController* controller = aDoc->GetAnimationController();
 | 
						|
    switch (aNewMode) {
 | 
						|
      case imgIContainer::kNormalAnimMode:
 | 
						|
      case imgIContainer::kLoopOnceAnimMode:
 | 
						|
        if (aOldMode == imgIContainer::kDontAnimMode)
 | 
						|
          controller->Resume(SMILTimeContainer::PAUSE_USERPREF);
 | 
						|
        break;
 | 
						|
 | 
						|
      case imgIContainer::kDontAnimMode:
 | 
						|
        if (aOldMode != imgIContainer::kDontAnimMode)
 | 
						|
          controller->Pause(SMILTimeContainer::PAUSE_USERPREF);
 | 
						|
        break;
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SetImageAnimationMode(uint16_t aMode) {
 | 
						|
  NS_ASSERTION(aMode == imgIContainer::kNormalAnimMode ||
 | 
						|
                   aMode == imgIContainer::kDontAnimMode ||
 | 
						|
                   aMode == imgIContainer::kLoopOnceAnimMode,
 | 
						|
               "Wrong Animation Mode is being set!");
 | 
						|
 | 
						|
  // Image animation mode cannot be changed when rendering to a printer.
 | 
						|
  if (!IsDynamic()) return;
 | 
						|
 | 
						|
  // Now walk the content tree and set the animation mode
 | 
						|
  // on all the images.
 | 
						|
  if (mPresShell) {
 | 
						|
    dom::Document* doc = mPresShell->GetDocument();
 | 
						|
    if (doc) {
 | 
						|
      doc->StyleImageLoader()->SetAnimationMode(aMode);
 | 
						|
 | 
						|
      Element* rootElement = doc->GetRootElement();
 | 
						|
      if (rootElement) {
 | 
						|
        SetImgAnimations(rootElement, aMode);
 | 
						|
      }
 | 
						|
      SetSMILAnimations(doc, aMode, mImageAnimationMode);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  mImageAnimationMode = aMode;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SetTextZoom(float aTextZoom) {
 | 
						|
  float newZoom = aTextZoom;
 | 
						|
  float minZoom = StaticPrefs::zoom_minPercent() / 100.0f;
 | 
						|
  float maxZoom = StaticPrefs::zoom_maxPercent() / 100.0f;
 | 
						|
 | 
						|
  if (newZoom < minZoom) {
 | 
						|
    newZoom = minZoom;
 | 
						|
  } else if (newZoom > maxZoom) {
 | 
						|
    newZoom = maxZoom;
 | 
						|
  }
 | 
						|
 | 
						|
  if (newZoom == mTextZoom) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  mTextZoom = newZoom;
 | 
						|
 | 
						|
  // Media queries could have changed, since we changed the meaning
 | 
						|
  // of 'em' units in them.
 | 
						|
  MediaFeatureValuesChanged(
 | 
						|
      {RestyleHint::RecascadeSubtree(), NS_STYLE_HINT_REFLOW,
 | 
						|
       MediaFeatureChangeReason::ZoomChange},
 | 
						|
      MediaFeatureChangePropagation::JustThisDocument);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SetInRDMPane(bool aInRDMPane) {
 | 
						|
  if (mInRDMPane == aInRDMPane) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  mInRDMPane = aInRDMPane;
 | 
						|
  RecomputeTheme();
 | 
						|
}
 | 
						|
 | 
						|
float nsPresContext::GetDeviceFullZoom() {
 | 
						|
  return mDeviceContext->GetFullZoom();
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SetFullZoom(float aZoom) {
 | 
						|
  if (!mPresShell || mFullZoom == aZoom) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Re-fetch the view manager's window dimensions in case there's a deferred
 | 
						|
  // resize which hasn't affected our mVisibleArea yet
 | 
						|
  nscoord oldWidthAppUnits, oldHeightAppUnits;
 | 
						|
  mPresShell->GetViewManager()->GetWindowDimensions(&oldWidthAppUnits,
 | 
						|
                                                    &oldHeightAppUnits);
 | 
						|
  float oldWidthDevPixels = oldWidthAppUnits / float(mCurAppUnitsPerDevPixel);
 | 
						|
  float oldHeightDevPixels = oldHeightAppUnits / float(mCurAppUnitsPerDevPixel);
 | 
						|
  mDeviceContext->SetFullZoom(aZoom);
 | 
						|
 | 
						|
  mFullZoom = aZoom;
 | 
						|
 | 
						|
  AppUnitsPerDevPixelChanged();
 | 
						|
 | 
						|
  mPresShell->GetViewManager()->SetWindowDimensions(
 | 
						|
      NSToCoordRound(oldWidthDevPixels * AppUnitsPerDevPixel()),
 | 
						|
      NSToCoordRound(oldHeightDevPixels * AppUnitsPerDevPixel()));
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SetOverrideDPPX(float aDPPX) {
 | 
						|
  // SetOverrideDPPX is called during navigations, including history
 | 
						|
  // traversals.  In that case, it's typically called with our current value,
 | 
						|
  // and we don't need to actually do anything.
 | 
						|
  if (aDPPX == GetOverrideDPPX()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  mMediaEmulationData.mDPPX = aDPPX;
 | 
						|
  MediaFeatureValuesChanged({MediaFeatureChangeReason::ResolutionChange},
 | 
						|
                            MediaFeatureChangePropagation::JustThisDocument);
 | 
						|
}
 | 
						|
 | 
						|
gfxSize nsPresContext::ScreenSizeInchesForFontInflation(bool* aChanged) {
 | 
						|
  if (aChanged) {
 | 
						|
    *aChanged = false;
 | 
						|
  }
 | 
						|
 | 
						|
  nsDeviceContext* dx = DeviceContext();
 | 
						|
  nsRect clientRect;
 | 
						|
  dx->GetClientRect(clientRect);  // FIXME: GetClientRect looks expensive
 | 
						|
  float unitsPerInch = dx->AppUnitsPerPhysicalInch();
 | 
						|
  gfxSize deviceSizeInches(float(clientRect.width) / unitsPerInch,
 | 
						|
                           float(clientRect.height) / unitsPerInch);
 | 
						|
 | 
						|
  if (mLastFontInflationScreenSize == gfxSize(-1.0, -1.0)) {
 | 
						|
    mLastFontInflationScreenSize = deviceSizeInches;
 | 
						|
  }
 | 
						|
 | 
						|
  if (deviceSizeInches != mLastFontInflationScreenSize && aChanged) {
 | 
						|
    *aChanged = true;
 | 
						|
    mLastFontInflationScreenSize = deviceSizeInches;
 | 
						|
  }
 | 
						|
 | 
						|
  return deviceSizeInches;
 | 
						|
}
 | 
						|
 | 
						|
static bool CheckOverflow(const ComputedStyle* aComputedStyle,
 | 
						|
                          ScrollStyles* aStyles) {
 | 
						|
  // If they're not styled yet, we'll get around to it when constructing frames
 | 
						|
  // for the element.
 | 
						|
  if (!aComputedStyle) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  const nsStyleDisplay* display = aComputedStyle->StyleDisplay();
 | 
						|
 | 
						|
  // If they will generate no box, just don't.
 | 
						|
  if (display->mDisplay == StyleDisplay::None ||
 | 
						|
      display->mDisplay == StyleDisplay::Contents) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  // NOTE(emilio): This check needs to match the one in
 | 
						|
  // Document::IsPotentiallyScrollable.
 | 
						|
  if (display->OverflowIsVisibleInBothAxis()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  *aStyles =
 | 
						|
      ScrollStyles(*display, ScrollStyles::MapOverflowToValidScrollStyle);
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
// https://drafts.csswg.org/css-overflow/#overflow-propagation
 | 
						|
//
 | 
						|
// NOTE(emilio): We may need to use out-of-date styles for this, since this is
 | 
						|
// called from nsCSSFrameConstructor::ContentRemoved. We could refactor this a
 | 
						|
// bit to avoid doing that, and also fix correctness issues (we don't invalidate
 | 
						|
// properly when we insert a body element and there is a previous one, for
 | 
						|
// example).
 | 
						|
static Element* GetPropagatedScrollStylesForViewport(
 | 
						|
    nsPresContext* aPresContext, ScrollStyles* aStyles) {
 | 
						|
  Document* document = aPresContext->Document();
 | 
						|
  Element* docElement = document->GetRootElement();
 | 
						|
  // docElement might be null if we're doing this after removing it.
 | 
						|
  if (!docElement) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  // Check the style on the document root element
 | 
						|
  const auto* rootStyle = Servo_Element_GetMaybeOutOfDateStyle(docElement);
 | 
						|
  if (CheckOverflow(rootStyle, aStyles)) {
 | 
						|
    // tell caller we stole the overflow style from the root element
 | 
						|
    return docElement;
 | 
						|
  }
 | 
						|
 | 
						|
  if (rootStyle && rootStyle->StyleDisplay()->IsContainAny()) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  // Don't look in the BODY for non-HTML documents or HTML documents
 | 
						|
  // with non-HTML roots.
 | 
						|
  // XXX this should be earlier; we shouldn't even look at the document root
 | 
						|
  // for non-HTML documents. Fix this once we support explicit CSS styling
 | 
						|
  // of the viewport
 | 
						|
  if (!document->IsHTMLOrXHTML() || !docElement->IsHTMLElement()) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  Element* bodyElement = document->AsHTMLDocument()->GetBodyElement();
 | 
						|
  if (!bodyElement) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  MOZ_ASSERT(bodyElement->IsHTMLElement(nsGkAtoms::body),
 | 
						|
             "GetBodyElement returned something bogus");
 | 
						|
 | 
						|
  const auto* bodyStyle = Servo_Element_GetMaybeOutOfDateStyle(bodyElement);
 | 
						|
  if (bodyStyle && bodyStyle->StyleDisplay()->IsContainAny()) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  if (CheckOverflow(bodyStyle, aStyles)) {
 | 
						|
    // tell caller we stole the overflow style from the body element
 | 
						|
    return bodyElement;
 | 
						|
  }
 | 
						|
 | 
						|
  return nullptr;
 | 
						|
}
 | 
						|
 | 
						|
Element* nsPresContext::UpdateViewportScrollStylesOverride() {
 | 
						|
  ScrollStyles oldViewportScrollStyles = mViewportScrollStyles;
 | 
						|
 | 
						|
  // Start off with our default styles, and then update them as needed.
 | 
						|
  mViewportScrollStyles =
 | 
						|
      ScrollStyles(StyleOverflow::Auto, StyleOverflow::Auto);
 | 
						|
  mViewportScrollOverrideElement = nullptr;
 | 
						|
  // Don't propagate the scrollbar state in printing or print preview.
 | 
						|
  if (!IsPaginated()) {
 | 
						|
    mViewportScrollOverrideElement =
 | 
						|
        GetPropagatedScrollStylesForViewport(this, &mViewportScrollStyles);
 | 
						|
  }
 | 
						|
 | 
						|
  dom::Document* document = Document();
 | 
						|
  if (Element* fsElement = document->GetUnretargetedFullscreenElement()) {
 | 
						|
    // If the document is in fullscreen, but the fullscreen element is
 | 
						|
    // not the root element, we should explicitly suppress the scrollbar
 | 
						|
    // here. Note that, we still need to return the original element
 | 
						|
    // the styles are from, so that the state of those elements is not
 | 
						|
    // affected across fullscreen change.
 | 
						|
    if (fsElement != document->GetRootElement() &&
 | 
						|
        fsElement != mViewportScrollOverrideElement) {
 | 
						|
      mViewportScrollStyles =
 | 
						|
          ScrollStyles(StyleOverflow::Hidden, StyleOverflow::Hidden);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (mViewportScrollStyles != oldViewportScrollStyles) {
 | 
						|
    if (mPresShell) {
 | 
						|
      if (nsIFrame* frame = mPresShell->GetRootFrame()) {
 | 
						|
        frame->SchedulePaint();
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  return mViewportScrollOverrideElement;
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::ElementWouldPropagateScrollStyles(const Element& aElement) {
 | 
						|
  if (aElement.GetParent() && !aElement.IsHTMLElement(nsGkAtoms::body)) {
 | 
						|
    // We certainly won't be propagating from this element.
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  // Go ahead and just call GetPropagatedScrollStylesForViewport, but update
 | 
						|
  // a dummy ScrollStyles we don't care about.  It'll do a bit of extra work,
 | 
						|
  // but saves us having to have more complicated code or more code duplication;
 | 
						|
  // in practice we will make this call quite rarely, because we checked for all
 | 
						|
  // the common cases above.
 | 
						|
  ScrollStyles dummy(StyleOverflow::Auto, StyleOverflow::Auto);
 | 
						|
  return GetPropagatedScrollStylesForViewport(this, &dummy) == &aElement;
 | 
						|
}
 | 
						|
 | 
						|
nsISupports* nsPresContext::GetContainerWeak() const {
 | 
						|
  return mDocument->GetDocShell();
 | 
						|
}
 | 
						|
 | 
						|
ColorScheme nsPresContext::DefaultBackgroundColorScheme() const {
 | 
						|
  dom::Document* doc = Document();
 | 
						|
  // Use a dark background for top-level about:blank that is inaccessible to
 | 
						|
  // content JS.
 | 
						|
  if (doc->IsLikelyContentInaccessibleTopLevelAboutBlank()) {
 | 
						|
    return doc->PreferredColorScheme(Document::IgnoreRFP::Yes);
 | 
						|
  }
 | 
						|
  // Prefer the root color-scheme (since generally the default canvas
 | 
						|
  // background comes from the root element's background-color), and fall back
 | 
						|
  // to the default color-scheme if not available.
 | 
						|
  if (auto* frame = FrameConstructor()->GetRootElementStyleFrame()) {
 | 
						|
    return LookAndFeel::ColorSchemeForFrame(frame);
 | 
						|
  }
 | 
						|
  return doc->DefaultColorScheme();
 | 
						|
}
 | 
						|
 | 
						|
nscolor nsPresContext::DefaultBackgroundColor() const {
 | 
						|
  if (!GetBackgroundColorDraw()) {
 | 
						|
    return NS_RGB(255, 255, 255);
 | 
						|
  }
 | 
						|
  return PrefSheetPrefs()
 | 
						|
      .ColorsFor(DefaultBackgroundColorScheme())
 | 
						|
      .mDefaultBackground;
 | 
						|
}
 | 
						|
 | 
						|
nsDocShell* nsPresContext::GetDocShell() const {
 | 
						|
  return nsDocShell::Cast(mDocument->GetDocShell());
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::BidiEnabled() const { return Document()->GetBidiEnabled(); }
 | 
						|
 | 
						|
void nsPresContext::SetBidiEnabled() const { Document()->SetBidiEnabled(); }
 | 
						|
 | 
						|
void nsPresContext::SetBidi(uint32_t aSource) {
 | 
						|
  // Don't do all this stuff unless the options have changed.
 | 
						|
  if (aSource == GetBidi()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  Document()->SetBidiOptions(aSource);
 | 
						|
  if (IBMBIDI_TEXTDIRECTION_RTL == GET_BIDI_OPTION_DIRECTION(aSource) ||
 | 
						|
      IBMBIDI_NUMERAL_HINDI == GET_BIDI_OPTION_NUMERAL(aSource)) {
 | 
						|
    SetBidiEnabled();
 | 
						|
  }
 | 
						|
  if (IBMBIDI_TEXTTYPE_VISUAL == GET_BIDI_OPTION_TEXTTYPE(aSource)) {
 | 
						|
    SetVisualMode(true);
 | 
						|
  } else if (IBMBIDI_TEXTTYPE_LOGICAL == GET_BIDI_OPTION_TEXTTYPE(aSource)) {
 | 
						|
    SetVisualMode(false);
 | 
						|
  } else {
 | 
						|
    SetVisualMode(IsVisualCharset(Document()->GetDocumentCharacterSet()));
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
uint32_t nsPresContext::GetBidi() const { return Document()->GetBidiOptions(); }
 | 
						|
 | 
						|
void nsPresContext::RecordInteractionTime(InteractionType aType,
 | 
						|
                                          const TimeStamp& aTimeStamp) {
 | 
						|
  if (!mInteractionTimeEnabled || aTimeStamp.IsNull()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Array of references to the member variable of each time stamp
 | 
						|
  // for the different interaction types, keyed by InteractionType.
 | 
						|
  TimeStamp nsPresContext::*interactionTimes[] = {
 | 
						|
      &nsPresContext::mFirstClickTime, &nsPresContext::mFirstKeyTime,
 | 
						|
      &nsPresContext::mFirstMouseMoveTime, &nsPresContext::mFirstScrollTime};
 | 
						|
 | 
						|
  // Array of histogram IDs for the different interaction types,
 | 
						|
  // keyed by InteractionType.
 | 
						|
  Telemetry::HistogramID histogramIds[] = {
 | 
						|
      Telemetry::TIME_TO_FIRST_CLICK_MS, Telemetry::TIME_TO_FIRST_KEY_INPUT_MS,
 | 
						|
      Telemetry::TIME_TO_FIRST_MOUSE_MOVE_MS,
 | 
						|
      Telemetry::TIME_TO_FIRST_SCROLL_MS};
 | 
						|
 | 
						|
  TimeStamp& interactionTime =
 | 
						|
      this->*(interactionTimes[static_cast<uint32_t>(aType)]);
 | 
						|
  if (!interactionTime.IsNull()) {
 | 
						|
    // We have already recorded an interaction time.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Record the interaction time if it occurs after the first paint
 | 
						|
  // of the top level content document.
 | 
						|
  nsPresContext* inProcessRootPresContext =
 | 
						|
      GetInProcessRootContentDocumentPresContext();
 | 
						|
 | 
						|
  if (!inProcessRootPresContext ||
 | 
						|
      !inProcessRootPresContext->IsRootContentDocumentCrossProcess()) {
 | 
						|
    // There is no top content pres context, or we are in a cross process
 | 
						|
    // document so we don't care about the interaction time. Record a value
 | 
						|
    // anyways to avoid trying to find the top content pres context in future
 | 
						|
    // interactions.
 | 
						|
    interactionTime = TimeStamp::Now();
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (inProcessRootPresContext->mFirstNonBlankPaintTime.IsNull() ||
 | 
						|
      inProcessRootPresContext->mFirstNonBlankPaintTime > aTimeStamp) {
 | 
						|
    // Top content pres context has not had a non-blank paint yet
 | 
						|
    // or the event timestamp is before the first non-blank paint,
 | 
						|
    // so don't record interaction time.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Check if we are recording the first of any of the interaction types.
 | 
						|
  bool isFirstInteraction = true;
 | 
						|
  for (TimeStamp nsPresContext::*memberPtr : interactionTimes) {
 | 
						|
    TimeStamp& timeStamp = this->*(memberPtr);
 | 
						|
    if (!timeStamp.IsNull()) {
 | 
						|
      isFirstInteraction = false;
 | 
						|
      break;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  interactionTime = TimeStamp::Now();
 | 
						|
  // Only the top level content pres context reports first interaction
 | 
						|
  // time to telemetry (if it hasn't already done so).
 | 
						|
  if (this == inProcessRootPresContext) {
 | 
						|
    if (Telemetry::CanRecordExtended()) {
 | 
						|
      double millis =
 | 
						|
          (interactionTime - mFirstNonBlankPaintTime).ToMilliseconds();
 | 
						|
      Telemetry::Accumulate(histogramIds[static_cast<uint32_t>(aType)], millis);
 | 
						|
 | 
						|
      if (isFirstInteraction) {
 | 
						|
        Telemetry::Accumulate(Telemetry::TIME_TO_FIRST_INTERACTION_MS, millis);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    inProcessRootPresContext->RecordInteractionTime(aType, aTimeStamp);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
nsITheme* nsPresContext::Theme() const {
 | 
						|
  MOZ_ASSERT(mTheme);
 | 
						|
  return mTheme;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::EnsureTheme() {
 | 
						|
  MOZ_ASSERT(!mTheme);
 | 
						|
  if (Document()->ShouldAvoidNativeTheme()) {
 | 
						|
    if (mInRDMPane) {
 | 
						|
      mTheme = do_GetRDMThemeDoNotUseDirectly();
 | 
						|
    } else {
 | 
						|
      mTheme = do_GetBasicNativeThemeDoNotUseDirectly();
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    mTheme = do_GetNativeThemeDoNotUseDirectly();
 | 
						|
  }
 | 
						|
  MOZ_RELEASE_ASSERT(mTheme);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::RecomputeTheme() {
 | 
						|
  if (!mTheme) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  nsCOMPtr<nsITheme> oldTheme = std::move(mTheme);
 | 
						|
  EnsureTheme();
 | 
						|
  if (oldTheme == mTheme) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  // Theme affects layout information (as it affects whether we create
 | 
						|
  // scrollbar buttons for example) and also style (affects the
 | 
						|
  // scrollbar-inline-size env var).
 | 
						|
  RebuildAllStyleData(nsChangeHint_ReconstructFrame,
 | 
						|
                      RestyleHint::RecascadeSubtree());
 | 
						|
  // This is a bit of a lie, but this affects the overlay-scrollbars
 | 
						|
  // media query and it's the code-path that gets taken for regular system
 | 
						|
  // metrics changes via ThemeChanged().
 | 
						|
  MediaFeatureValuesChanged({MediaFeatureChangeReason::SystemMetricsChange},
 | 
						|
                            MediaFeatureChangePropagation::JustThisDocument);
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::UseOverlayScrollbars() const {
 | 
						|
  return LookAndFeel::GetInt(LookAndFeel::IntID::UseOverlayScrollbars) ||
 | 
						|
         mInRDMPane;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::ThemeChanged(widget::ThemeChangeKind aKind) {
 | 
						|
  PROFILER_MARKER_UNTYPED("ThemeChanged", LAYOUT, MarkerStack::Capture());
 | 
						|
 | 
						|
  mPendingThemeChangeKind |= unsigned(aKind);
 | 
						|
 | 
						|
  if (!mPendingThemeChanged) {
 | 
						|
    nsCOMPtr<nsIRunnable> ev =
 | 
						|
        new WeakRunnableMethod("nsPresContext::ThemeChangedInternal", this,
 | 
						|
                               &nsPresContext::ThemeChangedInternal);
 | 
						|
    RefreshDriver()->AddEarlyRunner(ev);
 | 
						|
    mPendingThemeChanged = true;
 | 
						|
  }
 | 
						|
  MOZ_ASSERT(LookAndFeel::HasPendingGlobalThemeChange());
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::ThemeChangedInternal() {
 | 
						|
  MOZ_ASSERT(mPendingThemeChanged);
 | 
						|
 | 
						|
  mPendingThemeChanged = false;
 | 
						|
 | 
						|
  const auto kind = widget::ThemeChangeKind(mPendingThemeChangeKind);
 | 
						|
  mPendingThemeChangeKind = 0;
 | 
						|
 | 
						|
  LookAndFeel::HandleGlobalThemeChange();
 | 
						|
 | 
						|
  // Full zoom might have changed as a result of the text scale factor.
 | 
						|
  RecomputeBrowsingContextDependentData();
 | 
						|
 | 
						|
  // Changes to system metrics and other look and feel values can change media
 | 
						|
  // queries on them.
 | 
						|
  //
 | 
						|
  // Changes in theme can change system colors (whose changes are properly
 | 
						|
  // reflected in computed style data), system fonts (whose changes are
 | 
						|
  // some reflected (like sizes and such) and some not), and -moz-appearance
 | 
						|
  // (whose changes are not), so we need to recascade for the first, and reflow
 | 
						|
  // for the rest.
 | 
						|
  auto restyleHint = (kind & widget::ThemeChangeKind::Style)
 | 
						|
                         ? RestyleHint::RecascadeSubtree()
 | 
						|
                         : RestyleHint{0};
 | 
						|
  auto changeHint = (kind & widget::ThemeChangeKind::Layout)
 | 
						|
                        ? NS_STYLE_HINT_REFLOW
 | 
						|
                        : nsChangeHint(0);
 | 
						|
  MediaFeatureValuesChanged(
 | 
						|
      {restyleHint, changeHint, MediaFeatureChangeReason::SystemMetricsChange},
 | 
						|
      MediaFeatureChangePropagation::All);
 | 
						|
 | 
						|
  if (Document()->IsInChromeDocShell()) {
 | 
						|
    if (RefPtr<nsPIDOMWindowInner> win = Document()->GetInnerWindow()) {
 | 
						|
      nsContentUtils::DispatchEventOnlyToChrome(
 | 
						|
          Document(), nsGlobalWindowInner::Cast(win), u"nativethemechange"_ns,
 | 
						|
          CanBubble::eYes, Cancelable::eYes, nullptr);
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::UIResolutionChanged() {
 | 
						|
  if (!mPendingUIResolutionChanged) {
 | 
						|
    nsCOMPtr<nsIRunnable> ev =
 | 
						|
        NewRunnableMethod("nsPresContext::UIResolutionChangedInternal", this,
 | 
						|
                          &nsPresContext::UIResolutionChangedInternal);
 | 
						|
    nsresult rv = Document()->Dispatch(ev.forget());
 | 
						|
    if (NS_SUCCEEDED(rv)) {
 | 
						|
      mPendingUIResolutionChanged = true;
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::UIResolutionChangedSync() {
 | 
						|
  if (!mPendingUIResolutionChanged) {
 | 
						|
    mPendingUIResolutionChanged = true;
 | 
						|
    UIResolutionChangedInternal();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
static void NotifyTabUIResolutionChanged(nsIRemoteTab* aTab, void* aArg) {
 | 
						|
  aTab->NotifyResolutionChanged();
 | 
						|
}
 | 
						|
 | 
						|
static void NotifyChildrenUIResolutionChanged(nsPIDOMWindowOuter* aWindow) {
 | 
						|
  nsCOMPtr<Document> doc = aWindow->GetExtantDoc();
 | 
						|
  RefPtr<nsPIWindowRoot> topLevelWin = nsContentUtils::GetWindowRoot(doc);
 | 
						|
  if (!topLevelWin) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  topLevelWin->EnumerateBrowsers(NotifyTabUIResolutionChanged, nullptr);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::UIResolutionChangedInternal() {
 | 
						|
  mPendingUIResolutionChanged = false;
 | 
						|
 | 
						|
  mDeviceContext->CheckDPIChange();
 | 
						|
  if (mCurAppUnitsPerDevPixel != mDeviceContext->AppUnitsPerDevPixel()) {
 | 
						|
    AppUnitsPerDevPixelChanged();
 | 
						|
  }
 | 
						|
 | 
						|
  if (mPresShell) {
 | 
						|
    mPresShell->RefreshZoomConstraintsForScreenSizeChange();
 | 
						|
    if (RefPtr<MobileViewportManager> mvm =
 | 
						|
            mPresShell->GetMobileViewportManager()) {
 | 
						|
      mvm->UpdateSizesBeforeReflow();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Recursively notify all remote leaf descendants of the change.
 | 
						|
  if (nsPIDOMWindowOuter* window = mDocument->GetWindow()) {
 | 
						|
    NotifyChildrenUIResolutionChanged(window);
 | 
						|
  }
 | 
						|
 | 
						|
  auto recurse = [](dom::Document& aSubDoc) {
 | 
						|
    if (nsPresContext* pc = aSubDoc.GetPresContext()) {
 | 
						|
      pc->UIResolutionChangedInternal();
 | 
						|
    }
 | 
						|
    return CallState::Continue;
 | 
						|
  };
 | 
						|
  mDocument->EnumerateSubDocuments(recurse);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::EmulateMedium(nsAtom* aMediaType) {
 | 
						|
  MOZ_ASSERT(!aMediaType || aMediaType->IsAsciiLowercase());
 | 
						|
 | 
						|
  RefPtr<const nsAtom> oldMedium = Medium();
 | 
						|
  auto oldScheme = mDocument->PreferredColorScheme();
 | 
						|
 | 
						|
  mMediaEmulationData.mMedium = aMediaType;
 | 
						|
 | 
						|
  if (Medium() == oldMedium) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  MediaFeatureChange change(MediaFeatureChangeReason::MediumChange);
 | 
						|
  if (oldScheme != mDocument->PreferredColorScheme()) {
 | 
						|
    change |= MediaFeatureChange::ForPreferredColorSchemeChange();
 | 
						|
  }
 | 
						|
  MediaFeatureValuesChanged(change,
 | 
						|
                            MediaFeatureChangePropagation::JustThisDocument);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::ContentLanguageChanged() {
 | 
						|
  PostRebuildAllStyleDataEvent(nsChangeHint(0),
 | 
						|
                               RestyleHint::RecascadeSubtree());
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::RegisterManagedPostRefreshObserver(
 | 
						|
    ManagedPostRefreshObserver* aObserver) {
 | 
						|
  if (MOZ_UNLIKELY(!mPresShell)) {
 | 
						|
    // If we're detached from our pres shell already, refuse to keep observer
 | 
						|
    // around, as that'd create a cycle.
 | 
						|
    RefPtr<ManagedPostRefreshObserver> obs = aObserver;
 | 
						|
    obs->Cancel();
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  RefreshDriver()->AddPostRefreshObserver(
 | 
						|
      static_cast<nsAPostRefreshObserver*>(aObserver));
 | 
						|
  mManagedPostRefreshObservers.AppendElement(aObserver);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::UnregisterManagedPostRefreshObserver(
 | 
						|
    ManagedPostRefreshObserver* aObserver) {
 | 
						|
  RefreshDriver()->RemovePostRefreshObserver(
 | 
						|
      static_cast<nsAPostRefreshObserver*>(aObserver));
 | 
						|
  DebugOnly<bool> removed =
 | 
						|
      mManagedPostRefreshObservers.RemoveElement(aObserver);
 | 
						|
  MOZ_ASSERT(removed,
 | 
						|
             "ManagedPostRefreshObserver should be owned by PresContext");
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::CancelManagedPostRefreshObservers() {
 | 
						|
  auto observers = std::move(mManagedPostRefreshObservers);
 | 
						|
  nsRefreshDriver* driver = RefreshDriver();
 | 
						|
  for (const auto& observer : observers) {
 | 
						|
    observer->Cancel();
 | 
						|
    driver->RemovePostRefreshObserver(
 | 
						|
        static_cast<nsAPostRefreshObserver*>(observer));
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::RebuildAllStyleData(nsChangeHint aExtraHint,
 | 
						|
                                        const RestyleHint& aRestyleHint) {
 | 
						|
  if (!mPresShell) {
 | 
						|
    // We must have been torn down. Nothing to do here.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // TODO(emilio): It's unclear to me why would these three calls below be
 | 
						|
  // needed. In particular, RebuildAllStyleData doesn't rebuild rules or
 | 
						|
  // specified style information and such (note the comment in
 | 
						|
  // RestyleManager::RebuildAllStyleData re. the funny semantics), so I
 | 
						|
  // don't know why should we rebuild the user font set / counter styles /
 | 
						|
  // etc...
 | 
						|
  mDocument->MarkUserFontSetDirty();
 | 
						|
  MarkCounterStylesDirty();
 | 
						|
  MarkFontFeatureValuesDirty();
 | 
						|
  MarkFontPaletteValuesDirty();
 | 
						|
  PostRebuildAllStyleDataEvent(aExtraHint, aRestyleHint);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::PostRebuildAllStyleDataEvent(
 | 
						|
    nsChangeHint aExtraHint, const RestyleHint& aRestyleHint) {
 | 
						|
  if (!mPresShell) {
 | 
						|
    // We must have been torn down. Nothing to do here.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  RestyleManager()->RebuildAllStyleData(aExtraHint, aRestyleHint);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::MediaFeatureValuesChanged(
 | 
						|
    const MediaFeatureChange& aChange,
 | 
						|
    MediaFeatureChangePropagation aPropagation) {
 | 
						|
  if (mPresShell) {
 | 
						|
    mPresShell->EnsureStyleFlush();
 | 
						|
  }
 | 
						|
 | 
						|
  if (!mDocument->MediaQueryLists().isEmpty()) {
 | 
						|
    RefreshDriver()->ScheduleMediaQueryListenerUpdate();
 | 
						|
  }
 | 
						|
 | 
						|
  if (!mPendingMediaFeatureValuesChange) {
 | 
						|
    mPendingMediaFeatureValuesChange = MakeUnique<MediaFeatureChange>(aChange);
 | 
						|
  } else {
 | 
						|
    *mPendingMediaFeatureValuesChange |= aChange;
 | 
						|
  }
 | 
						|
 | 
						|
  if (aPropagation & MediaFeatureChangePropagation::Images) {
 | 
						|
    // Propagate the media feature value change down to any SVG images the
 | 
						|
    // document is using.
 | 
						|
    mDocument->ImageTracker()->MediaFeatureValuesChangedAllDocuments(aChange);
 | 
						|
  }
 | 
						|
 | 
						|
  if (aPropagation & MediaFeatureChangePropagation::SubDocuments) {
 | 
						|
    // And then into any subdocuments.
 | 
						|
    auto recurse = [&aChange, aPropagation](dom::Document& aSubDoc) {
 | 
						|
      if (nsPresContext* pc = aSubDoc.GetPresContext()) {
 | 
						|
        pc->MediaFeatureValuesChanged(aChange, aPropagation);
 | 
						|
      }
 | 
						|
      return CallState::Continue;
 | 
						|
    };
 | 
						|
    mDocument->EnumerateSubDocuments(recurse);
 | 
						|
  }
 | 
						|
 | 
						|
  // We notify the media feature values changed for the responsive content of
 | 
						|
  // HTMLImageElements synchronously, so their image sources are always
 | 
						|
  // up-to-date when running the image load tasks in the microtasks.
 | 
						|
  mDocument->NotifyMediaFeatureValuesChanged();
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::FlushPendingMediaFeatureValuesChanged() {
 | 
						|
  if (!mPendingMediaFeatureValuesChange) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  MediaFeatureChange change = *mPendingMediaFeatureValuesChange;
 | 
						|
  mPendingMediaFeatureValuesChange.reset();
 | 
						|
 | 
						|
  // MediumFeaturesChanged updates the applied rules, so it always gets called.
 | 
						|
  if (mPresShell) {
 | 
						|
    change.mRestyleHint |=
 | 
						|
        mPresShell->StyleSet()->MediumFeaturesChanged(change.mReason);
 | 
						|
  }
 | 
						|
 | 
						|
  const bool changedStyle = change.mRestyleHint || change.mChangeHint;
 | 
						|
  if (changedStyle) {
 | 
						|
    RebuildAllStyleData(change.mChangeHint, change.mRestyleHint);
 | 
						|
  }
 | 
						|
 | 
						|
  for (MediaQueryList* mql : mDocument->MediaQueryLists()) {
 | 
						|
    mql->MediaFeatureValuesChanged();
 | 
						|
  }
 | 
						|
 | 
						|
  return changedStyle;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SizeModeChanged(nsSizeMode aSizeMode) {
 | 
						|
  if (nsPIDOMWindowOuter* window = mDocument->GetWindow()) {
 | 
						|
    nsContentUtils::CallOnAllRemoteChildren(
 | 
						|
        window, [&aSizeMode](BrowserParent* aBrowserParent) -> CallState {
 | 
						|
          aBrowserParent->SizeModeChanged(aSizeMode);
 | 
						|
          return CallState::Continue;
 | 
						|
        });
 | 
						|
  }
 | 
						|
  MediaFeatureValuesChanged({MediaFeatureChangeReason::SizeModeChange},
 | 
						|
                            MediaFeatureChangePropagation::SubDocuments);
 | 
						|
}
 | 
						|
 | 
						|
nsCompatibility nsPresContext::CompatibilityMode() const {
 | 
						|
  return Document()->GetCompatibilityMode();
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SetPaginatedScrolling(bool aPaginated) {
 | 
						|
  if (mType == eContext_PrintPreview || mType == eContext_PageLayout) {
 | 
						|
    mCanPaginatedScroll = aPaginated;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SetPrintSettings(nsIPrintSettings* aPrintSettings) {
 | 
						|
  if (mMedium != nsGkAtoms::print) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  mPrintSettings = aPrintSettings;
 | 
						|
  mDefaultPageMargin = nsMargin();
 | 
						|
  if (!mPrintSettings) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Set the presentation context to the value in the print settings.
 | 
						|
  mDrawColorBackground = mPrintSettings->GetPrintBGColors();
 | 
						|
  mDrawImageBackground = mPrintSettings->GetPrintBGImages();
 | 
						|
 | 
						|
  nsIntMargin marginTwips = mPrintSettings->GetMarginInTwips();
 | 
						|
  if (!mPrintSettings->GetIgnoreUnwriteableMargins()) {
 | 
						|
    nsIntMargin unwriteableTwips =
 | 
						|
        mPrintSettings->GetUnwriteableMarginInTwips();
 | 
						|
    NS_ASSERTION(unwriteableTwips.top >= 0 && unwriteableTwips.right >= 0 &&
 | 
						|
                     unwriteableTwips.bottom >= 0 && unwriteableTwips.left >= 0,
 | 
						|
                 "Unwriteable twips should be non-negative");
 | 
						|
    marginTwips.EnsureAtLeast(unwriteableTwips);
 | 
						|
  }
 | 
						|
  mDefaultPageMargin = nsPresContext::CSSTwipsToAppUnits(marginTwips);
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::EnsureVisible() {
 | 
						|
  BrowsingContext* browsingContext =
 | 
						|
      mDocument ? mDocument->GetBrowsingContext() : nullptr;
 | 
						|
  if (!browsingContext || browsingContext->IsInBFCache()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  nsCOMPtr<nsIDocShell> docShell(GetDocShell());
 | 
						|
  if (!docShell) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  nsCOMPtr<nsIDocumentViewer> viewer;
 | 
						|
  docShell->GetDocViewer(getter_AddRefs(viewer));
 | 
						|
  // Make sure this is the content viewer we belong with
 | 
						|
  if (!viewer || viewer->GetPresContext() != this) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  // OK, this is us.  We want to call Show() on the content viewer.
 | 
						|
  nsresult result = viewer->Show();
 | 
						|
  return NS_SUCCEEDED(result);
 | 
						|
}
 | 
						|
 | 
						|
#ifdef MOZ_REFLOW_PERF
 | 
						|
void nsPresContext::CountReflows(const char* aName, nsIFrame* aFrame) {
 | 
						|
  if (mPresShell) {
 | 
						|
    mPresShell->CountReflows(aName, aFrame);
 | 
						|
  }
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
gfxUserFontSet* nsPresContext::GetUserFontSet() {
 | 
						|
  return mDocument->GetUserFontSet();
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::UserFontSetUpdated(gfxUserFontEntry* aUpdatedFont) {
 | 
						|
  if (!mPresShell) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Note: this method is called without a font when rules in the userfont set
 | 
						|
  // are updated.
 | 
						|
  //
 | 
						|
  // We can avoid a full restyle if font-metric-dependent units are not in use,
 | 
						|
  // since we know there's no style resolution that would depend on this font
 | 
						|
  // and trigger its load.
 | 
						|
  //
 | 
						|
  // TODO(emilio): We could be more granular if we knew which families have
 | 
						|
  // potentially changed.
 | 
						|
  if (!aUpdatedFont) {
 | 
						|
    auto hint = StyleSet()->UsesFontMetrics() ? RestyleHint::RecascadeSubtree()
 | 
						|
                                              : RestyleHint{0};
 | 
						|
    PostRebuildAllStyleDataEvent(NS_STYLE_HINT_REFLOW, hint);
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Iterate over the frame tree looking for frames associated with the
 | 
						|
  // downloadable font family in question. If a frame's nsStyleFont has
 | 
						|
  // the name, check the font group associated with the metrics to see if
 | 
						|
  // it contains that specific font (i.e. the one chosen within the family
 | 
						|
  // given the weight, width, and slant from the nsStyleFont). If it does,
 | 
						|
  // mark that frame dirty and skip inspecting its descendants.
 | 
						|
  if (nsIFrame* root = mPresShell->GetRootFrame()) {
 | 
						|
    nsFontFaceUtils::MarkDirtyForFontChange(root, aUpdatedFont);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
class CounterStyleCleaner final : public nsAPostRefreshObserver {
 | 
						|
 public:
 | 
						|
  CounterStyleCleaner(nsRefreshDriver* aRefreshDriver,
 | 
						|
                      CounterStyleManager* aCounterStyleManager)
 | 
						|
      : mRefreshDriver(aRefreshDriver),
 | 
						|
        mCounterStyleManager(aCounterStyleManager) {}
 | 
						|
  virtual ~CounterStyleCleaner() = default;
 | 
						|
 | 
						|
  void DidRefresh() final {
 | 
						|
    mRefreshDriver->RemovePostRefreshObserver(this);
 | 
						|
    mCounterStyleManager->CleanRetiredStyles();
 | 
						|
    delete this;
 | 
						|
  }
 | 
						|
 | 
						|
 private:
 | 
						|
  RefPtr<nsRefreshDriver> mRefreshDriver;
 | 
						|
  RefPtr<CounterStyleManager> mCounterStyleManager;
 | 
						|
};
 | 
						|
 | 
						|
void nsPresContext::FlushCounterStyles() {
 | 
						|
  if (!mPresShell) {
 | 
						|
    return;  // we've been torn down
 | 
						|
  }
 | 
						|
  if (mCounterStyleManager->IsInitial()) {
 | 
						|
    // Still in its initial state, no need to clean.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mCounterStylesDirty) {
 | 
						|
    bool changed = mCounterStyleManager->NotifyRuleChanged();
 | 
						|
    if (changed) {
 | 
						|
      PresShell()->NotifyCounterStylesAreDirty();
 | 
						|
      PostRebuildAllStyleDataEvent(NS_STYLE_HINT_REFLOW, RestyleHint{0});
 | 
						|
      RefreshDriver()->AddPostRefreshObserver(
 | 
						|
          new CounterStyleCleaner(RefreshDriver(), mCounterStyleManager));
 | 
						|
    }
 | 
						|
    mCounterStylesDirty = false;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::MarkCounterStylesDirty() {
 | 
						|
  if (mCounterStyleManager->IsInitial()) {
 | 
						|
    // Still in its initial state, no need to touch anything.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  mCounterStylesDirty = true;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::NotifyMissingFonts() {
 | 
						|
  if (mMissingFonts) {
 | 
						|
    mMissingFonts->Flush();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::EnsureSafeToHandOutCSSRules() {
 | 
						|
  if (!mPresShell->StyleSet()->EnsureUniqueInnerOnCSSSheets()) {
 | 
						|
    // Nothing to do.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  RebuildAllStyleData(nsChangeHint(0), RestyleHint::RestyleSubtree());
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::FireDOMPaintEvent(
 | 
						|
    nsTArray<nsRect>* aList, TransactionId aTransactionId,
 | 
						|
    mozilla::TimeStamp aTimeStamp /* = mozilla::TimeStamp() */) {
 | 
						|
  nsPIDOMWindowInner* ourWindow = mDocument->GetInnerWindow();
 | 
						|
  if (!ourWindow) return;
 | 
						|
 | 
						|
  nsCOMPtr<EventTarget> dispatchTarget = do_QueryInterface(ourWindow);
 | 
						|
  nsCOMPtr<EventTarget> eventTarget = dispatchTarget;
 | 
						|
  if (!IsChrome() && !mSendAfterPaintToContent) {
 | 
						|
    // Don't tell the window about this event, it should not know that
 | 
						|
    // something happened in a subdocument. Tell only the chrome event handler.
 | 
						|
    // (Events sent to the window get propagated to the chrome event handler
 | 
						|
    // automatically.)
 | 
						|
    dispatchTarget = ourWindow->GetParentTarget();
 | 
						|
    if (!dispatchTarget) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (aTimeStamp.IsNull()) {
 | 
						|
    aTimeStamp = mozilla::TimeStamp::Now();
 | 
						|
  }
 | 
						|
  DOMHighResTimeStamp timeStamp = 0;
 | 
						|
  if (ourWindow) {
 | 
						|
    mozilla::dom::Performance* perf = ourWindow->GetPerformance();
 | 
						|
    if (perf) {
 | 
						|
      timeStamp = perf->GetDOMTiming()->TimeStampToDOMHighRes(aTimeStamp);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Events sent to the window get propagated to the chrome event handler
 | 
						|
  // automatically.
 | 
						|
  //
 | 
						|
  // This will empty our list in case dispatching the event causes more damage
 | 
						|
  // (hopefully it won't, or we're likely to get an infinite loop! At least
 | 
						|
  // it won't be blocking app execution though).
 | 
						|
  RefPtr<NotifyPaintEvent> event =
 | 
						|
      NS_NewDOMNotifyPaintEvent(eventTarget, this, nullptr, eAfterPaint, aList,
 | 
						|
                                uint64_t(aTransactionId), timeStamp);
 | 
						|
 | 
						|
  // Even if we're not telling the window about the event (so eventTarget is
 | 
						|
  // the chrome event handler, not the window), the window is still
 | 
						|
  // logically the event target.
 | 
						|
  event->SetTarget(eventTarget);
 | 
						|
  event->SetTrusted(true);
 | 
						|
  EventDispatcher::DispatchDOMEvent(dispatchTarget, nullptr,
 | 
						|
                                    static_cast<Event*>(event), this, nullptr);
 | 
						|
}
 | 
						|
 | 
						|
static bool MayHavePaintEventListener(nsPIDOMWindowInner* aInnerWindow) {
 | 
						|
  if (!aInnerWindow) return false;
 | 
						|
  if (aInnerWindow->HasPaintEventListeners()) return true;
 | 
						|
 | 
						|
  EventTarget* parentTarget = aInnerWindow->GetParentTarget();
 | 
						|
  if (!parentTarget) return false;
 | 
						|
 | 
						|
  EventListenerManager* manager = nullptr;
 | 
						|
  if ((manager = parentTarget->GetExistingListenerManager()) &&
 | 
						|
      manager->MayHavePaintEventListener()) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  nsCOMPtr<nsINode> node;
 | 
						|
  if (parentTarget != aInnerWindow->GetChromeEventHandler()) {
 | 
						|
    nsCOMPtr<nsIInProcessContentFrameMessageManager> mm =
 | 
						|
        do_QueryInterface(parentTarget);
 | 
						|
    if (mm) {
 | 
						|
      node = mm->GetOwnerContent();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (!node) {
 | 
						|
    node = nsINode::FromEventTarget(parentTarget);
 | 
						|
  }
 | 
						|
  if (node) {
 | 
						|
    return MayHavePaintEventListener(node->OwnerDoc()->GetInnerWindow());
 | 
						|
  }
 | 
						|
 | 
						|
  if (nsCOMPtr<nsPIDOMWindowInner> window =
 | 
						|
          nsPIDOMWindowInner::FromEventTarget(parentTarget)) {
 | 
						|
    return MayHavePaintEventListener(window);
 | 
						|
  }
 | 
						|
 | 
						|
  if (nsCOMPtr<nsPIWindowRoot> root =
 | 
						|
          nsPIWindowRoot::FromEventTarget(parentTarget)) {
 | 
						|
    EventTarget* browserChildGlobal;
 | 
						|
    return root && (browserChildGlobal = root->GetParentTarget()) &&
 | 
						|
           (manager = browserChildGlobal->GetExistingListenerManager()) &&
 | 
						|
           manager->MayHavePaintEventListener();
 | 
						|
  }
 | 
						|
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::MayHavePaintEventListener() {
 | 
						|
  return ::MayHavePaintEventListener(mDocument->GetInnerWindow());
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::NotifyInvalidation(TransactionId aTransactionId,
 | 
						|
                                       const nsIntRect& aRect) {
 | 
						|
  // Prevent values from overflow after DevPixelsToAppUnits().
 | 
						|
  //
 | 
						|
  // DevPixelsTopAppUnits() will multiple a factor (60) to the value,
 | 
						|
  // it may make the result value over the edge (overflow) of max or
 | 
						|
  // min value of int32_t. Compute the max sized dev pixel rect that
 | 
						|
  // we can support and intersect with it.
 | 
						|
  nsIntRect clampedRect = nsIntRect::MaxIntRect();
 | 
						|
  clampedRect.ScaleInverseRoundIn(AppUnitsPerDevPixel());
 | 
						|
 | 
						|
  clampedRect = clampedRect.Intersect(aRect);
 | 
						|
 | 
						|
  nsRect rect(DevPixelsToAppUnits(clampedRect.x),
 | 
						|
              DevPixelsToAppUnits(clampedRect.y),
 | 
						|
              DevPixelsToAppUnits(clampedRect.width),
 | 
						|
              DevPixelsToAppUnits(clampedRect.height));
 | 
						|
  NotifyInvalidation(aTransactionId, rect);
 | 
						|
}
 | 
						|
 | 
						|
nsPresContext::TransactionInvalidations* nsPresContext::GetInvalidations(
 | 
						|
    TransactionId aTransactionId) {
 | 
						|
  for (TransactionInvalidations& t : mTransactions) {
 | 
						|
    if (t.mTransactionId == aTransactionId) {
 | 
						|
      return &t;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return nullptr;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::NotifyInvalidation(TransactionId aTransactionId,
 | 
						|
                                       const nsRect& aRect) {
 | 
						|
  MOZ_ASSERT(GetContainerWeak(), "Invalidation in detached pres context");
 | 
						|
 | 
						|
  // If there is no paint event listener, then we don't need to fire
 | 
						|
  // the asynchronous event. We don't even need to record invalidation.
 | 
						|
  // MayHavePaintEventListener is pretty cheap and we could make it
 | 
						|
  // even cheaper by providing a more efficient
 | 
						|
  // nsPIDOMWindow::GetListenerManager.
 | 
						|
 | 
						|
  nsPresContext* pc;
 | 
						|
  for (pc = this; pc; pc = pc->GetParentPresContext()) {
 | 
						|
    TransactionInvalidations* transaction =
 | 
						|
        pc->GetInvalidations(aTransactionId);
 | 
						|
    if (transaction) {
 | 
						|
      break;
 | 
						|
    } else {
 | 
						|
      transaction = pc->mTransactions.AppendElement();
 | 
						|
      transaction->mTransactionId = aTransactionId;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  TransactionInvalidations* transaction = GetInvalidations(aTransactionId);
 | 
						|
  MOZ_ASSERT(transaction);
 | 
						|
  transaction->mInvalidations.AppendElement(aRect);
 | 
						|
}
 | 
						|
 | 
						|
class DelayedFireDOMPaintEvent : public Runnable {
 | 
						|
 public:
 | 
						|
  DelayedFireDOMPaintEvent(
 | 
						|
      nsPresContext* aPresContext, nsTArray<nsRect>&& aList,
 | 
						|
      TransactionId aTransactionId,
 | 
						|
      const mozilla::TimeStamp& aTimeStamp = mozilla::TimeStamp())
 | 
						|
      : mozilla::Runnable("DelayedFireDOMPaintEvent"),
 | 
						|
        mPresContext(aPresContext),
 | 
						|
        mTransactionId(aTransactionId),
 | 
						|
        mTimeStamp(aTimeStamp),
 | 
						|
        mList(std::move(aList)) {
 | 
						|
    MOZ_ASSERT(mPresContext->GetContainerWeak(),
 | 
						|
               "DOMPaintEvent requested for a detached pres context");
 | 
						|
  }
 | 
						|
  NS_IMETHOD Run() override {
 | 
						|
    // The pres context might have been detached during the delay -
 | 
						|
    // that's fine, just don't fire the event.
 | 
						|
    if (mPresContext->GetContainerWeak()) {
 | 
						|
      mPresContext->FireDOMPaintEvent(&mList, mTransactionId, mTimeStamp);
 | 
						|
    }
 | 
						|
    return NS_OK;
 | 
						|
  }
 | 
						|
 | 
						|
  RefPtr<nsPresContext> mPresContext;
 | 
						|
  TransactionId mTransactionId;
 | 
						|
  const mozilla::TimeStamp mTimeStamp;
 | 
						|
  nsTArray<nsRect> mList;
 | 
						|
};
 | 
						|
 | 
						|
void nsPresContext::NotifyRevokingDidPaint(TransactionId aTransactionId) {
 | 
						|
  if ((IsRoot() || !PresShell()->IsVisible()) && mTransactions.IsEmpty()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  TransactionInvalidations* transaction = nullptr;
 | 
						|
  for (auto& t : mTransactions) {
 | 
						|
    if (t.mTransactionId == aTransactionId) {
 | 
						|
      transaction = &t;
 | 
						|
      break;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  // If there are no transaction invalidations (which imply callers waiting
 | 
						|
  // on the event) for this revoked id, then we don't need to fire a
 | 
						|
  // MozAfterPaint.
 | 
						|
  if (!transaction) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // If there are queued transactions with an earlier id, we can't send
 | 
						|
  // our event now since it will arrive out of order. Set the waiting for
 | 
						|
  // previous transaction flag to true, and we'll send the event when
 | 
						|
  // the others are completed.
 | 
						|
  // If this is the only transaction, then we can send it immediately.
 | 
						|
  if (mTransactions.Length() == 1) {
 | 
						|
    nsCOMPtr<nsIRunnable> ev = new DelayedFireDOMPaintEvent(
 | 
						|
        this, std::move(transaction->mInvalidations),
 | 
						|
        transaction->mTransactionId, mozilla::TimeStamp());
 | 
						|
    nsContentUtils::AddScriptRunner(ev);
 | 
						|
    mTransactions.RemoveElementAt(0);
 | 
						|
  } else {
 | 
						|
    transaction->mIsWaitingForPreviousTransaction = true;
 | 
						|
  }
 | 
						|
 | 
						|
  auto recurse = [&aTransactionId](dom::Document& aSubDoc) {
 | 
						|
    if (nsPresContext* pc = aSubDoc.GetPresContext()) {
 | 
						|
      pc->NotifyRevokingDidPaint(aTransactionId);
 | 
						|
    }
 | 
						|
    return CallState::Continue;
 | 
						|
  };
 | 
						|
  mDocument->EnumerateSubDocuments(recurse);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::NotifyDidPaintForSubtree(
 | 
						|
    TransactionId aTransactionId, const mozilla::TimeStamp& aTimeStamp) {
 | 
						|
  if (mFirstContentfulPaintTransactionId && !mHadContentfulPaintComposite) {
 | 
						|
    if (aTransactionId >= *mFirstContentfulPaintTransactionId) {
 | 
						|
      mHadContentfulPaintComposite = true;
 | 
						|
      RefPtr<nsDOMNavigationTiming> timing = mDocument->GetNavigationTiming();
 | 
						|
      if (timing && !IsPrintingOrPrintPreview()) {
 | 
						|
        timing->NotifyContentfulCompositeForRootContentDocument(aTimeStamp);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (IsRoot() && mTransactions.IsEmpty()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!PresShell()->IsVisible() && mTransactions.IsEmpty()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Non-root prescontexts fire MozAfterPaint to all their descendants
 | 
						|
  // unconditionally, even if no invalidations have been collected. This is
 | 
						|
  // because we don't want to eat the cost of collecting invalidations for
 | 
						|
  // every subdocument (which would require putting every subdocument in its
 | 
						|
  // own layer).
 | 
						|
 | 
						|
  bool sent = false;
 | 
						|
  uint32_t i = 0;
 | 
						|
  while (i < mTransactions.Length()) {
 | 
						|
    if (mTransactions[i].mTransactionId <= aTransactionId) {
 | 
						|
      if (!mTransactions[i].mInvalidations.IsEmpty()) {
 | 
						|
        nsCOMPtr<nsIRunnable> ev = new DelayedFireDOMPaintEvent(
 | 
						|
            this, std::move(mTransactions[i].mInvalidations),
 | 
						|
            mTransactions[i].mTransactionId, aTimeStamp);
 | 
						|
        NS_DispatchToCurrentThreadQueue(ev.forget(),
 | 
						|
                                        EventQueuePriority::MediumHigh);
 | 
						|
        sent = true;
 | 
						|
      }
 | 
						|
      mTransactions.RemoveElementAt(i);
 | 
						|
    } else {
 | 
						|
      // If there are transaction which is waiting for this transaction,
 | 
						|
      // we should fire a MozAfterPaint immediately.
 | 
						|
      if (sent && mTransactions[i].mIsWaitingForPreviousTransaction) {
 | 
						|
        nsCOMPtr<nsIRunnable> ev = new DelayedFireDOMPaintEvent(
 | 
						|
            this, std::move(mTransactions[i].mInvalidations),
 | 
						|
            mTransactions[i].mTransactionId, aTimeStamp);
 | 
						|
        NS_DispatchToCurrentThreadQueue(ev.forget(),
 | 
						|
                                        EventQueuePriority::MediumHigh);
 | 
						|
        sent = true;
 | 
						|
        mTransactions.RemoveElementAt(i);
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      i++;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (!sent) {
 | 
						|
    nsTArray<nsRect> dummy;
 | 
						|
    nsCOMPtr<nsIRunnable> ev = new DelayedFireDOMPaintEvent(
 | 
						|
        this, std::move(dummy), aTransactionId, aTimeStamp);
 | 
						|
    NS_DispatchToCurrentThreadQueue(ev.forget(),
 | 
						|
                                    EventQueuePriority::MediumHigh);
 | 
						|
  }
 | 
						|
 | 
						|
  auto recurse = [&aTransactionId, &aTimeStamp](dom::Document& aSubDoc) {
 | 
						|
    if (nsPresContext* pc = aSubDoc.GetPresContext()) {
 | 
						|
      pc->NotifyDidPaintForSubtree(aTransactionId, aTimeStamp);
 | 
						|
    }
 | 
						|
    return CallState::Continue;
 | 
						|
  };
 | 
						|
  mDocument->EnumerateSubDocuments(recurse);
 | 
						|
}
 | 
						|
 | 
						|
already_AddRefed<nsITimer> nsPresContext::CreateTimer(
 | 
						|
    nsTimerCallbackFunc aCallback, const char* aName, uint32_t aDelay) {
 | 
						|
  nsCOMPtr<nsITimer> timer;
 | 
						|
  NS_NewTimerWithFuncCallback(getter_AddRefs(timer), aCallback, this, aDelay,
 | 
						|
                              nsITimer::TYPE_ONE_SHOT, aName,
 | 
						|
                              GetMainThreadSerialEventTarget());
 | 
						|
  return timer.forget();
 | 
						|
}
 | 
						|
 | 
						|
static bool sGotInterruptEnv = false;
 | 
						|
enum InterruptMode { ModeRandom, ModeCounter, ModeEvent };
 | 
						|
// Controlled by the GECKO_REFLOW_INTERRUPT_MODE env var; allowed values are
 | 
						|
// "random" (except on Windows) or "counter".  If neither is used, the mode is
 | 
						|
// ModeEvent.
 | 
						|
static InterruptMode sInterruptMode = ModeEvent;
 | 
						|
#ifndef XP_WIN
 | 
						|
// Used for the "random" mode.  Controlled by the GECKO_REFLOW_INTERRUPT_SEED
 | 
						|
// env var.
 | 
						|
static uint32_t sInterruptSeed = 1;
 | 
						|
#endif
 | 
						|
// Used for the "counter" mode.  This is the number of unskipped interrupt
 | 
						|
// checks that have to happen before we interrupt.  Controlled by the
 | 
						|
// GECKO_REFLOW_INTERRUPT_FREQUENCY env var.
 | 
						|
static uint32_t sInterruptMaxCounter = 10;
 | 
						|
// Used for the "counter" mode.  This counts up to sInterruptMaxCounter and is
 | 
						|
// then reset to 0.
 | 
						|
static uint32_t sInterruptCounter;
 | 
						|
// Number of interrupt checks to skip before really trying to interrupt.
 | 
						|
// Controlled by the GECKO_REFLOW_INTERRUPT_CHECKS_TO_SKIP env var.
 | 
						|
static uint32_t sInterruptChecksToSkip = 200;
 | 
						|
// Number of milliseconds that a reflow should be allowed to run for before we
 | 
						|
// actually allow interruption.  Controlled by the
 | 
						|
// GECKO_REFLOW_MIN_NOINTERRUPT_DURATION env var.  Can't be initialized here,
 | 
						|
// because TimeDuration/TimeStamp is not safe to use in static constructors..
 | 
						|
static TimeDuration sInterruptTimeout;
 | 
						|
 | 
						|
static void GetInterruptEnv() {
 | 
						|
  char* ev = PR_GetEnv("GECKO_REFLOW_INTERRUPT_MODE");
 | 
						|
  if (ev) {
 | 
						|
#ifndef XP_WIN
 | 
						|
    if (nsCRT::strcasecmp(ev, "random") == 0) {
 | 
						|
      ev = PR_GetEnv("GECKO_REFLOW_INTERRUPT_SEED");
 | 
						|
      if (ev) {
 | 
						|
        sInterruptSeed = atoi(ev);
 | 
						|
      }
 | 
						|
      srandom(sInterruptSeed);
 | 
						|
      sInterruptMode = ModeRandom;
 | 
						|
    } else
 | 
						|
#endif
 | 
						|
        if (PL_strcasecmp(ev, "counter") == 0) {
 | 
						|
      ev = PR_GetEnv("GECKO_REFLOW_INTERRUPT_FREQUENCY");
 | 
						|
      if (ev) {
 | 
						|
        sInterruptMaxCounter = atoi(ev);
 | 
						|
      }
 | 
						|
      sInterruptCounter = 0;
 | 
						|
      sInterruptMode = ModeCounter;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  ev = PR_GetEnv("GECKO_REFLOW_INTERRUPT_CHECKS_TO_SKIP");
 | 
						|
  if (ev) {
 | 
						|
    sInterruptChecksToSkip = atoi(ev);
 | 
						|
  }
 | 
						|
 | 
						|
  ev = PR_GetEnv("GECKO_REFLOW_MIN_NOINTERRUPT_DURATION");
 | 
						|
  int duration_ms = ev ? atoi(ev) : 100;
 | 
						|
  sInterruptTimeout = TimeDuration::FromMilliseconds(duration_ms);
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::HavePendingInputEvent() {
 | 
						|
  switch (sInterruptMode) {
 | 
						|
#ifndef XP_WIN
 | 
						|
    case ModeRandom:
 | 
						|
      return (random() & 1);
 | 
						|
#endif
 | 
						|
    case ModeCounter:
 | 
						|
      if (sInterruptCounter < sInterruptMaxCounter) {
 | 
						|
        ++sInterruptCounter;
 | 
						|
        return false;
 | 
						|
      }
 | 
						|
      sInterruptCounter = 0;
 | 
						|
      return true;
 | 
						|
    default:
 | 
						|
    case ModeEvent: {
 | 
						|
      nsIFrame* f = PresShell()->GetRootFrame();
 | 
						|
      if (f) {
 | 
						|
        nsIWidget* w = f->GetNearestWidget();
 | 
						|
        if (w) {
 | 
						|
          return w->HasPendingInputEvent();
 | 
						|
        }
 | 
						|
      }
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::HasPendingRestyleOrReflow() {
 | 
						|
  mozilla::PresShell* presShell = PresShell();
 | 
						|
  return presShell->NeedStyleFlush() || presShell->HasPendingReflow();
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::ReflowStarted(bool aInterruptible) {
 | 
						|
#ifdef NOISY_INTERRUPTIBLE_REFLOW
 | 
						|
  if (!aInterruptible) {
 | 
						|
    printf("STARTING NONINTERRUPTIBLE REFLOW\n");
 | 
						|
  }
 | 
						|
#endif
 | 
						|
  // We don't support interrupting in paginated contexts, since page
 | 
						|
  // sequences only handle initial reflow
 | 
						|
  mInterruptsEnabled = aInterruptible && !IsPaginated() &&
 | 
						|
                       StaticPrefs::layout_interruptible_reflow_enabled();
 | 
						|
 | 
						|
  // Don't set mHasPendingInterrupt based on HavePendingInputEvent() here.  If
 | 
						|
  // we ever change that, then we need to update the code in
 | 
						|
  // PresShell::DoReflow to only add the just-reflown root to dirty roots if
 | 
						|
  // it's actually dirty.  Otherwise we can end up adding a root that has no
 | 
						|
  // interruptible descendants, just because we detected an interrupt at reflow
 | 
						|
  // start.
 | 
						|
  mHasPendingInterrupt = false;
 | 
						|
 | 
						|
  mInterruptChecksToSkip = sInterruptChecksToSkip;
 | 
						|
 | 
						|
  if (mInterruptsEnabled) {
 | 
						|
    mReflowStartTime = TimeStamp::Now();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::CheckForInterrupt(nsIFrame* aFrame) {
 | 
						|
  if (mHasPendingInterrupt) {
 | 
						|
    mPresShell->FrameNeedsToContinueReflow(aFrame);
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!sGotInterruptEnv) {
 | 
						|
    sGotInterruptEnv = true;
 | 
						|
    GetInterruptEnv();
 | 
						|
  }
 | 
						|
 | 
						|
  if (!mInterruptsEnabled) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mInterruptChecksToSkip > 0) {
 | 
						|
    --mInterruptChecksToSkip;
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  mInterruptChecksToSkip = sInterruptChecksToSkip;
 | 
						|
 | 
						|
  // Don't interrupt if it's been less than sInterruptTimeout since we started
 | 
						|
  // the reflow.
 | 
						|
  mHasPendingInterrupt =
 | 
						|
      TimeStamp::Now() - mReflowStartTime > sInterruptTimeout &&
 | 
						|
      HavePendingInputEvent() && !IsChrome();
 | 
						|
 | 
						|
  if (mPendingInterruptFromTest) {
 | 
						|
    mPendingInterruptFromTest = false;
 | 
						|
    mHasPendingInterrupt = true;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mHasPendingInterrupt) {
 | 
						|
#ifdef NOISY_INTERRUPTIBLE_REFLOW
 | 
						|
    printf("*** DETECTED pending interrupt (time=%lld)\n", PR_Now());
 | 
						|
#endif /* NOISY_INTERRUPTIBLE_REFLOW */
 | 
						|
    mPresShell->FrameNeedsToContinueReflow(aFrame);
 | 
						|
  }
 | 
						|
  return mHasPendingInterrupt;
 | 
						|
}
 | 
						|
 | 
						|
nsIFrame* nsPresContext::GetPrimaryFrameFor(nsIContent* aContent) {
 | 
						|
  MOZ_ASSERT(aContent, "Don't do that");
 | 
						|
  if (GetPresShell() &&
 | 
						|
      GetPresShell()->GetDocument() == aContent->GetComposedDoc()) {
 | 
						|
    return aContent->GetPrimaryFrame();
 | 
						|
  }
 | 
						|
  return nullptr;
 | 
						|
}
 | 
						|
 | 
						|
size_t nsPresContext::SizeOfExcludingThis(MallocSizeOf aMallocSizeOf) const {
 | 
						|
  // Measurement may be added later if DMD finds it is worthwhile.
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::IsRootContentDocumentInProcess() const {
 | 
						|
  if (mDocument->IsResourceDoc()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  if (IsChrome()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  // We may not have a root frame, so use views.
 | 
						|
  nsView* view = PresShell()->GetViewManager()->GetRootView();
 | 
						|
  if (!view) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
  view = view->GetParent();  // anonymous inner view
 | 
						|
  if (!view) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
  view = view->GetParent();  // subdocumentframe's view
 | 
						|
  if (!view) {
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  nsIFrame* f = view->GetFrame();
 | 
						|
  return (f && f->PresContext()->IsChrome());
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::IsRootContentDocumentCrossProcess() const {
 | 
						|
  if (mDocument->IsResourceDoc()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  if (BrowsingContext* browsingContext = mDocument->GetBrowsingContext()) {
 | 
						|
    if (browsingContext->GetEmbeddedInContentDocument()) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return mDocument->IsTopLevelContentDocument();
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::NotifyNonBlankPaint() {
 | 
						|
  MOZ_ASSERT(!mHadNonBlankPaint);
 | 
						|
  mHadNonBlankPaint = true;
 | 
						|
  if (IsRootContentDocumentCrossProcess()) {
 | 
						|
    RefPtr<nsDOMNavigationTiming> timing = mDocument->GetNavigationTiming();
 | 
						|
    if (timing && !IsPrintingOrPrintPreview()) {
 | 
						|
      timing->NotifyNonBlankPaintForRootContentDocument();
 | 
						|
    }
 | 
						|
 | 
						|
    mFirstNonBlankPaintTime = TimeStamp::Now();
 | 
						|
  }
 | 
						|
  if (IsChrome() && IsRoot()) {
 | 
						|
    if (nsCOMPtr<nsIWidget> rootWidget = GetRootWidget()) {
 | 
						|
      rootWidget->DidGetNonBlankPaint();
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
bool nsPresContext::HasStoppedGeneratingLCP() const {
 | 
						|
  if (auto* perf = GetPerformanceMainThread()) {
 | 
						|
    return perf->HasDispatchedInputEvent() || perf->HasDispatchedScrollEvent();
 | 
						|
  }
 | 
						|
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::NotifyContentfulPaint() {
 | 
						|
  if (mHadFirstContentfulPaint && HasStoppedGeneratingLCP()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  nsRootPresContext* rootPresContext = GetRootPresContext();
 | 
						|
  if (!rootPresContext) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  if (!mHadNonTickContentfulPaint) {
 | 
						|
#ifdef MOZ_WIDGET_ANDROID
 | 
						|
    (new AsyncEventDispatcher(mDocument, u"MozFirstContentfulPaint"_ns,
 | 
						|
                              CanBubble::eYes, ChromeOnlyDispatch::eYes))
 | 
						|
        ->PostDOMEvent();
 | 
						|
#endif
 | 
						|
  }
 | 
						|
  if (!rootPresContext->RefreshDriver()->IsInRefresh()) {
 | 
						|
    if (!mHadNonTickContentfulPaint) {
 | 
						|
      rootPresContext->RefreshDriver()
 | 
						|
          ->AddForceNotifyContentfulPaintPresContext(this);
 | 
						|
      mHadNonTickContentfulPaint = true;
 | 
						|
    }
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!mHadFirstContentfulPaint) {
 | 
						|
    mHadFirstContentfulPaint = true;
 | 
						|
    mFirstContentfulPaintTransactionId =
 | 
						|
        Some(rootPresContext->mRefreshDriver->LastTransactionId().Next());
 | 
						|
  }
 | 
						|
 | 
						|
  if (auto* perf = GetPerformanceMainThread()) {
 | 
						|
    mMarkPaintTimingStart = TimeStamp::Now();
 | 
						|
    MOZ_ASSERT(rootPresContext->RefreshDriver()->IsInRefresh(),
 | 
						|
               "We should only notify contentful paint during refresh "
 | 
						|
               "driver ticks");
 | 
						|
    if (!perf->HadFCPTimingEntry()) {
 | 
						|
      TimeStamp nowTime = rootPresContext->RefreshDriver()->MostRecentRefresh(
 | 
						|
          /* aEnsureTimerStarted */ false);
 | 
						|
      MOZ_ASSERT(!nowTime.IsNull(),
 | 
						|
                 "Most recent refresh timestamp should exist since we are in "
 | 
						|
                 "a refresh driver tick");
 | 
						|
      RefPtr<PerformancePaintTiming> paintTiming = new PerformancePaintTiming(
 | 
						|
          perf, u"first-contentful-paint"_ns, nowTime);
 | 
						|
      perf->SetFCPTimingEntry(paintTiming);
 | 
						|
 | 
						|
      if (profiler_thread_is_being_profiled_for_markers()) {
 | 
						|
        RefPtr<nsDOMNavigationTiming> timing = mDocument->GetNavigationTiming();
 | 
						|
        if (timing) {
 | 
						|
          TimeStamp navigationStart = timing->GetNavigationStartTimeStamp();
 | 
						|
          TimeDuration elapsed = nowTime - navigationStart;
 | 
						|
          nsIURI* docURI = Document()->GetDocumentURI();
 | 
						|
          nsPrintfCString marker(
 | 
						|
              "Contentful paint after %dms for URL %s",
 | 
						|
              int(elapsed.ToMilliseconds()),
 | 
						|
              nsContentUtils::TruncatedURLForDisplay(docURI).get());
 | 
						|
          PROFILER_MARKER_TEXT(
 | 
						|
              "FirstContentfulPaint", DOM,
 | 
						|
              MarkerOptions(
 | 
						|
                  MarkerTiming::Interval(navigationStart, nowTime),
 | 
						|
                  MarkerInnerWindowId(mDocument->GetInnerWindow()->WindowID())),
 | 
						|
              marker);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    perf->ProcessElementTiming();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::NotifyPaintStatusReset() {
 | 
						|
  mHadNonBlankPaint = false;
 | 
						|
  mHadFirstContentfulPaint = false;
 | 
						|
#if defined(MOZ_WIDGET_ANDROID)
 | 
						|
  (new AsyncEventDispatcher(mDocument, u"MozPaintStatusReset"_ns,
 | 
						|
                            CanBubble::eYes, ChromeOnlyDispatch::eYes))
 | 
						|
      ->PostDOMEvent();
 | 
						|
#endif
 | 
						|
  mHadNonTickContentfulPaint = false;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::NotifyDOMContentFlushed() {
 | 
						|
  NS_ENSURE_TRUE_VOID(mPresShell);
 | 
						|
  if (IsRootContentDocumentCrossProcess()) {
 | 
						|
    RefPtr<nsDOMNavigationTiming> timing = mDocument->GetNavigationTiming();
 | 
						|
    if (timing) {
 | 
						|
      timing->NotifyDOMContentFlushedForRootContentDocument();
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
nscoord nsPresContext::GfxUnitsToAppUnits(gfxFloat aGfxUnits) const {
 | 
						|
  return mDeviceContext->GfxUnitsToAppUnits(aGfxUnits);
 | 
						|
}
 | 
						|
 | 
						|
gfxFloat nsPresContext::AppUnitsToGfxUnits(nscoord aAppUnits) const {
 | 
						|
  return mDeviceContext->AppUnitsToGfxUnits(aAppUnits);
 | 
						|
}
 | 
						|
 | 
						|
nscoord nsPresContext::PhysicalMillimetersToAppUnits(float aMM) const {
 | 
						|
  float inches = aMM / MM_PER_INCH_FLOAT;
 | 
						|
  return NSToCoordFloorClamped(
 | 
						|
      inches * float(DeviceContext()->AppUnitsPerPhysicalInch()));
 | 
						|
}
 | 
						|
 | 
						|
uint64_t nsPresContext::GetRestyleGeneration() const {
 | 
						|
  if (!mRestyleManager) {
 | 
						|
    return 0;
 | 
						|
  }
 | 
						|
  return mRestyleManager->GetRestyleGeneration();
 | 
						|
}
 | 
						|
 | 
						|
uint64_t nsPresContext::GetUndisplayedRestyleGeneration() const {
 | 
						|
  if (!mRestyleManager) {
 | 
						|
    return 0;
 | 
						|
  }
 | 
						|
  return mRestyleManager->GetUndisplayedRestyleGeneration();
 | 
						|
}
 | 
						|
 | 
						|
mozilla::intl::Bidi& nsPresContext::BidiEngine() {
 | 
						|
  MOZ_ASSERT(NS_IsMainThread());
 | 
						|
 | 
						|
  if (!mBidiEngine) {
 | 
						|
    mBidiEngine = MakeUnique<mozilla::intl::Bidi>();
 | 
						|
  }
 | 
						|
  return *mBidiEngine;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::FlushFontFeatureValues() {
 | 
						|
  if (!mPresShell) {
 | 
						|
    return;  // we've been torn down
 | 
						|
  }
 | 
						|
 | 
						|
  if (!mFontFeatureValuesDirty) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  ServoStyleSet* styleSet = mPresShell->StyleSet();
 | 
						|
  mFontFeatureValuesLookup = styleSet->BuildFontFeatureValueSet();
 | 
						|
  mFontFeatureValuesDirty = false;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::FlushFontPaletteValues() {
 | 
						|
  if (!mPresShell) {
 | 
						|
    return;  // we've been torn down
 | 
						|
  }
 | 
						|
 | 
						|
  if (!mFontPaletteValuesDirty) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  ServoStyleSet* styleSet = mPresShell->StyleSet();
 | 
						|
  mFontPaletteValueSet = styleSet->BuildFontPaletteValueSet();
 | 
						|
  mFontPaletteValuesDirty = false;
 | 
						|
 | 
						|
  if (mFontPaletteCache) {
 | 
						|
    mFontPaletteCache->SetPaletteValueSet(mFontPaletteValueSet);
 | 
						|
  }
 | 
						|
 | 
						|
  // Even if we're not reflowing anything, a change to the palette means we
 | 
						|
  // need to repaint in order to show the new colors.
 | 
						|
  InvalidatePaintedLayers();
 | 
						|
}
 | 
						|
 | 
						|
gfx::PaletteCache& nsPresContext::FontPaletteCache() {
 | 
						|
  if (!mFontPaletteCache) {
 | 
						|
    mFontPaletteCache = MakeUnique<gfx::PaletteCache>(mFontPaletteValueSet);
 | 
						|
  }
 | 
						|
  return *mFontPaletteCache.get();
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SetVisibleArea(const nsRect& r) {
 | 
						|
  if (!r.IsEqualEdges(mVisibleArea)) {
 | 
						|
    mVisibleArea = r;
 | 
						|
    mSizeForViewportUnits = mVisibleArea.Size();
 | 
						|
    if (IsRootContentDocumentCrossProcess()) {
 | 
						|
      AdjustSizeForViewportUnits();
 | 
						|
    }
 | 
						|
    // Visible area does not affect media queries when paginated.
 | 
						|
    if (!IsRootPaginatedDocument()) {
 | 
						|
      MediaFeatureValuesChanged(
 | 
						|
          {mozilla::MediaFeatureChangeReason::ViewportChange},
 | 
						|
          MediaFeatureChangePropagation::JustThisDocument);
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SetDynamicToolbarMaxHeight(ScreenIntCoord aHeight) {
 | 
						|
  MOZ_ASSERT(IsRootContentDocumentCrossProcess());
 | 
						|
 | 
						|
  if (mDynamicToolbarMaxHeight == aHeight) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  mDynamicToolbarMaxHeight = aHeight;
 | 
						|
  mDynamicToolbarHeight = aHeight;
 | 
						|
 | 
						|
  AdjustSizeForViewportUnits();
 | 
						|
 | 
						|
  if (RefPtr<mozilla::PresShell> presShell = mPresShell) {
 | 
						|
    // Changing the max height of the dynamic toolbar changes the ICB size, we
 | 
						|
    // need to kick a reflow with the current window dimensions since the max
 | 
						|
    // height change doesn't change the window dimensions but
 | 
						|
    // PresShell::ResizeReflow ends up subtracting the new dynamic toolbar
 | 
						|
    // height from the window dimensions and kick a reflow with the proper ICB
 | 
						|
    // size.
 | 
						|
    presShell->ForceResizeReflowWithCurrentDimensions();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::AdjustSizeForViewportUnits() {
 | 
						|
  MOZ_ASSERT(IsRootContentDocumentCrossProcess());
 | 
						|
  if (mVisibleArea.height == NS_UNCONSTRAINEDSIZE) {
 | 
						|
    // Ignore `NS_UNCONSTRAINEDSIZE` since it's a temporary state during a
 | 
						|
    // reflow. We will end up calling this function again with a proper size in
 | 
						|
    // the same reflow.
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (MOZ_UNLIKELY(mVisibleArea.height +
 | 
						|
                       NSIntPixelsToAppUnits(mDynamicToolbarMaxHeight,
 | 
						|
                                             mCurAppUnitsPerDevPixel) >
 | 
						|
                   nscoord_MAX)) {
 | 
						|
    MOZ_ASSERT_UNREACHABLE("The dynamic toolbar max height is probably wrong");
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  mSizeForViewportUnits.height =
 | 
						|
      mVisibleArea.height +
 | 
						|
      NSIntPixelsToAppUnits(mDynamicToolbarMaxHeight, mCurAppUnitsPerDevPixel);
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::UpdateDynamicToolbarOffset(ScreenIntCoord aOffset) {
 | 
						|
  MOZ_ASSERT(IsRootContentDocumentCrossProcess());
 | 
						|
  if (!mPresShell) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (!HasDynamicToolbar()) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  MOZ_ASSERT(-mDynamicToolbarMaxHeight <= aOffset && aOffset <= 0);
 | 
						|
  if (mDynamicToolbarHeight == mDynamicToolbarMaxHeight + aOffset) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // Forcibly flush position:fixed elements in the case where the dynamic
 | 
						|
  // toolbar is going to be completely hidden or starts to be visible so that
 | 
						|
  // %-based style values will be recomputed with the visual viewport size which
 | 
						|
  // is including the area covered by the dynamic toolbar.
 | 
						|
  if (mDynamicToolbarHeight == 0 || aOffset == -mDynamicToolbarMaxHeight) {
 | 
						|
    mPresShell->MarkFixedFramesForReflow(IntrinsicDirty::None);
 | 
						|
    mPresShell->AddResizeEventFlushObserverIfNeeded();
 | 
						|
  }
 | 
						|
 | 
						|
  mDynamicToolbarHeight = mDynamicToolbarMaxHeight + aOffset;
 | 
						|
 | 
						|
  if (RefPtr<MobileViewportManager> mvm =
 | 
						|
          mPresShell->GetMobileViewportManager()) {
 | 
						|
    mvm->UpdateVisualViewportSizeByDynamicToolbar(-aOffset);
 | 
						|
  }
 | 
						|
 | 
						|
  mPresShell->StyleSet()->InvalidateForViewportUnits(
 | 
						|
      ServoStyleSet::OnlyDynamic::Yes);
 | 
						|
}
 | 
						|
 | 
						|
DynamicToolbarState nsPresContext::GetDynamicToolbarState() const {
 | 
						|
  if (!IsRootContentDocumentCrossProcess() || !HasDynamicToolbar()) {
 | 
						|
    return DynamicToolbarState::None;
 | 
						|
  }
 | 
						|
 | 
						|
  if (mDynamicToolbarMaxHeight == mDynamicToolbarHeight) {
 | 
						|
    return DynamicToolbarState::Expanded;
 | 
						|
  } else if (mDynamicToolbarHeight == 0) {
 | 
						|
    return DynamicToolbarState::Collapsed;
 | 
						|
  }
 | 
						|
  return DynamicToolbarState::InTransition;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::SetSafeAreaInsets(const ScreenIntMargin& aSafeAreaInsets) {
 | 
						|
  if (mSafeAreaInsets == aSafeAreaInsets) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  mSafeAreaInsets = aSafeAreaInsets;
 | 
						|
 | 
						|
  PostRebuildAllStyleDataEvent(nsChangeHint(0),
 | 
						|
                               RestyleHint::RecascadeSubtree());
 | 
						|
}
 | 
						|
 | 
						|
PerformanceMainThread* nsPresContext::GetPerformanceMainThread() const {
 | 
						|
  if (nsPIDOMWindowInner* innerWindow = mDocument->GetInnerWindow()) {
 | 
						|
    if (auto* perf = static_cast<PerformanceMainThread*>(
 | 
						|
            innerWindow->GetPerformance())) {
 | 
						|
      return perf;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return nullptr;
 | 
						|
}
 | 
						|
 | 
						|
void nsPresContext::UpdateHiddenByContentVisibilityForAnimations() {
 | 
						|
  mDocument->UpdateHiddenByContentVisibilityForAnimations();
 | 
						|
  TimelineManager()->UpdateHiddenByContentVisibilityForAnimations();
 | 
						|
}
 | 
						|
 | 
						|
#ifdef DEBUG
 | 
						|
 | 
						|
void nsPresContext::ValidatePresShellAndDocumentReleation() const {
 | 
						|
  NS_ASSERTION(!mPresShell || !mPresShell->GetDocument() ||
 | 
						|
                   mPresShell->GetDocument() == mDocument,
 | 
						|
               "nsPresContext doesn't have the same document as nsPresShell!");
 | 
						|
}
 | 
						|
 | 
						|
#endif  // #ifdef DEBUG
 | 
						|
 | 
						|
nsRootPresContext::nsRootPresContext(dom::Document* aDocument,
 | 
						|
                                     nsPresContextType aType)
 | 
						|
    : nsPresContext(aDocument, aType) {}
 | 
						|
 | 
						|
void nsRootPresContext::AddWillPaintObserver(nsIRunnable* aRunnable) {
 | 
						|
  if (!mWillPaintFallbackEvent.IsPending()) {
 | 
						|
    mWillPaintFallbackEvent = new RunWillPaintObservers(this);
 | 
						|
    Document()->Dispatch(do_AddRef(mWillPaintFallbackEvent));
 | 
						|
  }
 | 
						|
  mWillPaintObservers.AppendElement(aRunnable);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Run all runnables that need to get called before the next paint.
 | 
						|
 */
 | 
						|
void nsRootPresContext::FlushWillPaintObservers() {
 | 
						|
  mWillPaintFallbackEvent = nullptr;
 | 
						|
  nsTArray<nsCOMPtr<nsIRunnable>> observers = std::move(mWillPaintObservers);
 | 
						|
  for (uint32_t i = 0; i < observers.Length(); ++i) {
 | 
						|
    observers[i]->Run();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
size_t nsRootPresContext::SizeOfExcludingThis(
 | 
						|
    MallocSizeOf aMallocSizeOf) const {
 | 
						|
  return nsPresContext::SizeOfExcludingThis(aMallocSizeOf);
 | 
						|
 | 
						|
  // Measurement of the following members may be added later if DMD finds it is
 | 
						|
  // worthwhile:
 | 
						|
  // - mWillPaintObservers
 | 
						|
  // - mWillPaintFallbackEvent
 | 
						|
}
 |