|
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
|