Blame layout/base/RestyleManager.h

Packit f0b94e
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
Packit f0b94e
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
Packit f0b94e
/* This Source Code Form is subject to the terms of the Mozilla Public
Packit f0b94e
 * License, v. 2.0. If a copy of the MPL was not distributed with this
Packit f0b94e
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
Packit f0b94e
Packit f0b94e
#ifndef mozilla_RestyleManager_h
Packit f0b94e
#define mozilla_RestyleManager_h
Packit f0b94e
Packit f0b94e
#include "mozilla/OverflowChangedTracker.h"
Packit f0b94e
#include "nsChangeHint.h"
Packit f0b94e
#include "nsPresContext.h"
Packit f0b94e
#include "nsStringFwd.h"
Packit f0b94e
Packit f0b94e
class nsCSSFrameConstructor;
Packit f0b94e
class nsStyleChangeList;
Packit f0b94e
Packit f0b94e
namespace mozilla {
Packit f0b94e
Packit f0b94e
class EventStates;
Packit f0b94e
class GeckoRestyleManager;
Packit f0b94e
class ServoRestyleManager;
Packit f0b94e
Packit f0b94e
namespace dom {
Packit f0b94e
class Element;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
/**
Packit f0b94e
 * Class for sharing data and logic common to both GeckoRestyleManager and
Packit f0b94e
 * ServoRestyleManager.
Packit f0b94e
 */
Packit f0b94e
class RestyleManager {
Packit f0b94e
 public:
Packit f0b94e
  typedef mozilla::dom::Element Element;
Packit f0b94e
Packit f0b94e
  NS_INLINE_DECL_REFCOUNTING(mozilla::RestyleManager)
Packit f0b94e
Packit f0b94e
  // Get an integer that increments every time we process pending restyles.
Packit f0b94e
  // The value is never 0.
Packit f0b94e
  uint64_t GetRestyleGeneration() const { return mRestyleGeneration; }
Packit f0b94e
  // Unlike GetRestyleGeneration, which means the actual restyling count,
Packit f0b94e
  // GetUndisplayedRestyleGeneration represents any possible DOM changes that
Packit f0b94e
  // can cause restyling. This is needed for getComputedStyle to work with
Packit f0b94e
  // non-styled (e.g. display: none) elements.
Packit f0b94e
  uint64_t GetUndisplayedRestyleGeneration() const {
Packit f0b94e
    return mUndisplayedRestyleGeneration;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Get an integer that increments every time there is a style change
Packit f0b94e
  // as a result of a change to the :hover content state.
Packit f0b94e
  uint32_t GetHoverGeneration() const { return mHoverGeneration; }
Packit f0b94e
Packit f0b94e
  void Disconnect() { mPresContext = nullptr; }
Packit f0b94e
Packit f0b94e
  static nsCString RestyleHintToString(nsRestyleHint aHint);
Packit f0b94e
Packit f0b94e
#ifdef DEBUG
Packit f0b94e
  static nsCString ChangeHintToString(nsChangeHint aHint);
Packit f0b94e
Packit f0b94e
  /**
Packit f0b94e
   * DEBUG ONLY method to verify integrity of style tree versus frame tree
Packit f0b94e
   */
Packit f0b94e
  void DebugVerifyStyleTree(nsIFrame* aFrame);
Packit f0b94e
#endif
Packit f0b94e
Packit f0b94e
  void FlushOverflowChangedTracker() { mOverflowChangedTracker.Flush(); }
Packit f0b94e
Packit f0b94e
  // Should be called when a frame is going to be destroyed and
Packit f0b94e
  // WillDestroyFrameTree hasn't been called yet.
Packit f0b94e
  void NotifyDestroyingFrame(nsIFrame* aFrame) {
Packit f0b94e
    mOverflowChangedTracker.RemoveFrame(aFrame);
Packit f0b94e
    // If ProcessRestyledFrames is tracking frames which have been
Packit f0b94e
    // destroyed (to avoid re-visiting them), add this one to its set.
Packit f0b94e
    if (mDestroyedFrames) {
Packit f0b94e
      mDestroyedFrames->PutEntry(aFrame);
Packit f0b94e
    }
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Note: It's the caller's responsibility to make sure to wrap a
Packit f0b94e
  // ProcessRestyledFrames call in a view update batch and a script blocker.
Packit f0b94e
  // This function does not call ProcessAttachedQueue() on the binding manager.
Packit f0b94e
  // If the caller wants that to happen synchronously, it needs to handle that
Packit f0b94e
  // itself.
Packit f0b94e
  void ProcessRestyledFrames(nsStyleChangeList& aChangeList);
Packit f0b94e
Packit f0b94e
  bool IsInStyleRefresh() const { return mInStyleRefresh; }
Packit f0b94e
Packit f0b94e
  // AnimationsWithDestroyedFrame is used to stop animations and transitions
Packit f0b94e
  // on elements that have no frame at the end of the restyling process.
Packit f0b94e
  // It only lives during the restyling process.
Packit f0b94e
  class MOZ_STACK_CLASS AnimationsWithDestroyedFrame final {
Packit f0b94e
   public:
Packit f0b94e
    // Construct a AnimationsWithDestroyedFrame object.  The caller must
Packit f0b94e
    // ensure that aRestyleManager lives at least as long as the
Packit f0b94e
    // object.  (This is generally easy since the caller is typically a
Packit f0b94e
    // method of RestyleManager.)
Packit f0b94e
    explicit AnimationsWithDestroyedFrame(RestyleManager* aRestyleManager);
Packit f0b94e
Packit f0b94e
    // This method takes the content node for the generated content for
Packit f0b94e
    // animation/transition on ::before and ::after, rather than the
Packit f0b94e
    // content node for the real element.
Packit f0b94e
    void Put(nsIContent* aContent, nsStyleContext* aStyleContext) {
Packit f0b94e
      MOZ_ASSERT(aContent);
Packit f0b94e
      CSSPseudoElementType pseudoType = aStyleContext->GetPseudoType();
Packit f0b94e
      if (pseudoType == CSSPseudoElementType::NotPseudo) {
Packit f0b94e
        mContents.AppendElement(aContent);
Packit f0b94e
      } else if (pseudoType == CSSPseudoElementType::before) {
Packit f0b94e
        MOZ_ASSERT(aContent->NodeInfo()->NameAtom() ==
Packit f0b94e
                   nsGkAtoms::mozgeneratedcontentbefore);
Packit f0b94e
        mBeforeContents.AppendElement(aContent->GetParent());
Packit f0b94e
      } else if (pseudoType == CSSPseudoElementType::after) {
Packit f0b94e
        MOZ_ASSERT(aContent->NodeInfo()->NameAtom() ==
Packit f0b94e
                   nsGkAtoms::mozgeneratedcontentafter);
Packit f0b94e
        mAfterContents.AppendElement(aContent->GetParent());
Packit f0b94e
      }
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    void StopAnimationsForElementsWithoutFrames();
Packit f0b94e
Packit f0b94e
   private:
Packit f0b94e
    void StopAnimationsWithoutFrame(nsTArray<RefPtr<nsIContent>>& aArray,
Packit f0b94e
                                    CSSPseudoElementType aPseudoType);
Packit f0b94e
Packit f0b94e
    RestyleManager* mRestyleManager;
Packit f0b94e
    AutoRestore<AnimationsWithDestroyedFrame*> mRestorePointer;
Packit f0b94e
Packit f0b94e
    // Below three arrays might include elements that have already had their
Packit f0b94e
    // animations or transitions stopped.
Packit f0b94e
    //
Packit f0b94e
    // mBeforeContents and mAfterContents hold the real element rather than
Packit f0b94e
    // the content node for the generated content (which might change during
Packit f0b94e
    // a reframe)
Packit f0b94e
    nsTArray<RefPtr<nsIContent>> mContents;
Packit f0b94e
    nsTArray<RefPtr<nsIContent>> mBeforeContents;
Packit f0b94e
    nsTArray<RefPtr<nsIContent>> mAfterContents;
Packit f0b94e
  };
Packit f0b94e
Packit f0b94e
  /**
Packit f0b94e
   * Return the current AnimationsWithDestroyedFrame struct, or null if we're
Packit f0b94e
   * not currently in a restyling operation.
Packit f0b94e
   */
Packit f0b94e
  AnimationsWithDestroyedFrame* GetAnimationsWithDestroyedFrame() {
Packit f0b94e
    return mAnimationsWithDestroyedFrame;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  void ContentInserted(nsINode* aContainer, nsIContent* aChild);
Packit f0b94e
  void ContentAppended(nsIContent* aContainer, nsIContent* aFirstNewContent);
Packit f0b94e
Packit f0b94e
  // This would be have the same logic as RestyleForInsertOrChange if we got the
Packit f0b94e
  // notification before the removal.  However, we get it after, so we need the
Packit f0b94e
  // following sibling in addition to the old child.  |aContainer| must be
Packit f0b94e
  // non-null; when the container is null, no work is needed.  aFollowingSibling
Packit f0b94e
  // is the sibling that used to come after aOldChild before the removal.
Packit f0b94e
  void ContentRemoved(nsINode* aContainer, nsIContent* aOldChild,
Packit f0b94e
                      nsIContent* aFollowingSibling);
Packit f0b94e
Packit f0b94e
  // Restyling for a ContentInserted (notification after insertion) or
Packit f0b94e
  // for some CharacterDataChanged.  |aContainer| must be non-null; when
Packit f0b94e
  // the container is null, no work is needed.
Packit f0b94e
  void RestyleForInsertOrChange(nsINode* aContainer, nsIContent* aChild);
Packit f0b94e
Packit f0b94e
  // Restyle for a CharacterDataChanged notification. In practice this can only
Packit f0b94e
  // affect :empty / :-moz-only-whitespace / :-moz-first-node / :-moz-last-node.
Packit f0b94e
  void CharacterDataChanged(nsIContent*, const CharacterDataChangeInfo&);
Packit f0b94e
Packit f0b94e
  MOZ_DECL_STYLO_METHODS(GeckoRestyleManager, ServoRestyleManager)
Packit f0b94e
Packit f0b94e
  inline void PostRestyleEvent(dom::Element* aElement,
Packit f0b94e
                               nsRestyleHint aRestyleHint,
Packit f0b94e
                               nsChangeHint aMinChangeHint);
Packit f0b94e
  inline void RebuildAllStyleData(nsChangeHint aExtraHint,
Packit f0b94e
                                  nsRestyleHint aRestyleHint);
Packit f0b94e
  inline void PostRebuildAllStyleDataEvent(nsChangeHint aExtraHint,
Packit f0b94e
                                           nsRestyleHint aRestyleHint);
Packit f0b94e
  inline void ProcessPendingRestyles();
Packit f0b94e
  inline void ContentStateChanged(nsIContent* aContent, EventStates aStateMask);
Packit f0b94e
  inline void AttributeWillChange(dom::Element* aElement, int32_t aNameSpaceID,
Packit f0b94e
                                  nsAtom* aAttribute, int32_t aModType,
Packit f0b94e
                                  const nsAttrValue* aNewValue);
Packit f0b94e
  inline void AttributeChanged(dom::Element* aElement, int32_t aNameSpaceID,
Packit f0b94e
                               nsAtom* aAttribute, int32_t aModType,
Packit f0b94e
                               const nsAttrValue* aOldValue);
Packit f0b94e
  inline nsresult ReparentStyleContext(nsIFrame* aFrame);
Packit f0b94e
Packit f0b94e
  inline void UpdateOnlyAnimationStyles();
Packit f0b94e
Packit f0b94e
  // Get a counter that increments on every style change, that we use to
Packit f0b94e
  // track whether off-main-thread animations are up-to-date.
Packit f0b94e
  uint64_t GetAnimationGeneration() const { return mAnimationGeneration; }
Packit f0b94e
Packit f0b94e
  static uint64_t GetAnimationGenerationForFrame(nsIFrame* aFrame);
Packit f0b94e
Packit f0b94e
  // Update the animation generation count to mark that animation state
Packit f0b94e
  // has changed.
Packit f0b94e
  //
Packit f0b94e
  // This is normally performed automatically by ProcessPendingRestyles
Packit f0b94e
  // but it is also called when we have out-of-band changes to animations
Packit f0b94e
  // such as changes made through the Web Animations API.
Packit f0b94e
  void IncrementAnimationGeneration();
Packit f0b94e
Packit f0b94e
  static void AddLayerChangesForAnimation(
Packit f0b94e
      nsIFrame* aFrame, nsIContent* aContent,
Packit f0b94e
      nsStyleChangeList& aChangeListToProcess);
Packit f0b94e
Packit f0b94e
 protected:
Packit f0b94e
  RestyleManager(StyleBackendType aType, nsPresContext* aPresContext);
Packit f0b94e
Packit f0b94e
  virtual ~RestyleManager() {
Packit f0b94e
    MOZ_ASSERT(!mAnimationsWithDestroyedFrame,
Packit f0b94e
               "leaving dangling pointers from AnimationsWithDestroyedFrame");
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  void RestyleForEmptyChange(Element* aContainer);
Packit f0b94e
  void MaybeRestyleForEdgeChildChange(Element* aContainer,
Packit f0b94e
                                      nsIContent* aChangedChild);
Packit f0b94e
Packit f0b94e
  void ContentStateChangedInternal(Element* aElement, EventStates aStateMask,
Packit f0b94e
                                   nsChangeHint* aOutChangeHint);
Packit f0b94e
Packit f0b94e
  bool IsDisconnected() { return mPresContext == nullptr; }
Packit f0b94e
Packit f0b94e
  void IncrementHoverGeneration() { ++mHoverGeneration; }
Packit f0b94e
Packit f0b94e
  void IncrementRestyleGeneration() {
Packit f0b94e
    if (++mRestyleGeneration == 0) {
Packit f0b94e
      // Keep mRestyleGeneration from being 0, since that's what
Packit f0b94e
      // nsPresContext::GetRestyleGeneration returns when it no
Packit f0b94e
      // longer has a RestyleManager.
Packit f0b94e
      ++mRestyleGeneration;
Packit f0b94e
    }
Packit f0b94e
    IncrementUndisplayedRestyleGeneration();
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  void IncrementUndisplayedRestyleGeneration() {
Packit f0b94e
    if (++mUndisplayedRestyleGeneration == 0) {
Packit f0b94e
      // Ensure mUndisplayedRestyleGeneration > 0, for the same reason as
Packit f0b94e
      // IncrementRestyleGeneration.
Packit f0b94e
      ++mUndisplayedRestyleGeneration;
Packit f0b94e
    }
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  nsPresContext* PresContext() const {
Packit f0b94e
    MOZ_ASSERT(mPresContext);
Packit f0b94e
    return mPresContext;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  nsCSSFrameConstructor* FrameConstructor() const {
Packit f0b94e
    return PresContext()->FrameConstructor();
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
 private:
Packit f0b94e
  nsPresContext* mPresContext;  // weak, can be null after Disconnect().
Packit f0b94e
  uint64_t mRestyleGeneration;
Packit f0b94e
  uint64_t mUndisplayedRestyleGeneration;
Packit f0b94e
  uint32_t mHoverGeneration;
Packit f0b94e
Packit f0b94e
  // Used to keep track of frames that have been destroyed during
Packit f0b94e
  // ProcessRestyledFrames, so we don't try to touch them again even if
Packit f0b94e
  // they're referenced again later in the changelist.
Packit f0b94e
  mozilla::UniquePtr<nsTHashtable<nsPtrHashKey<const nsIFrame>>>
Packit f0b94e
      mDestroyedFrames;
Packit f0b94e
Packit f0b94e
 protected:
Packit f0b94e
  const StyleBackendType mType;
Packit f0b94e
Packit f0b94e
  // True if we're in the middle of a nsRefreshDriver refresh
Packit f0b94e
  bool mInStyleRefresh;
Packit f0b94e
Packit f0b94e
  // The total number of animation flushes by this frame constructor.
Packit f0b94e
  // Used to keep the layer and animation manager in sync.
Packit f0b94e
  uint64_t mAnimationGeneration;
Packit f0b94e
Packit f0b94e
  OverflowChangedTracker mOverflowChangedTracker;
Packit f0b94e
Packit f0b94e
  AnimationsWithDestroyedFrame* mAnimationsWithDestroyedFrame = nullptr;
Packit f0b94e
Packit f0b94e
  friend class mozilla::GeckoRestyleManager;
Packit f0b94e
  friend class mozilla::ServoRestyleManager;
Packit f0b94e
};
Packit f0b94e
Packit f0b94e
}  // namespace mozilla
Packit f0b94e
Packit f0b94e
#endif