/* -*- 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/. */ /* the interface (to internal code) for retrieving computed style data */ #ifndef _ComputedStyle_h_ #define _ComputedStyle_h_ #include "nsIMemoryReporter.h" #include #include "mozilla/Assertions.h" #include "mozilla/CachedInheritingStyles.h" #include "mozilla/Maybe.h" #include "mozilla/ServoComputedData.h" #include "mozilla/ServoTypes.h" #include "mozilla/ServoUtils.h" #include "mozilla/StyleComplexColor.h" #include "nsCSSAnonBoxes.h" #include "nsCSSPseudoElements.h" #include "nsStyleStructFwd.h" class nsAtom; enum nsChangeHint : uint32_t; class nsIPresShell; class nsPresContext; class nsWindowSizes; #define STYLE_STRUCT(name_) struct nsStyle##name_; #include "nsStyleStructList.h" #undef STYLE_STRUCT extern "C" { void Servo_ComputedStyle_AddRef(const mozilla::ComputedStyle* aStyle); void Servo_ComputedStyle_Release(const mozilla::ComputedStyle* aStyle); void Gecko_ComputedStyle_Destroy(mozilla::ComputedStyle*); } namespace mozilla { namespace dom { class Document; } enum class CSSPseudoElementType : uint8_t; class ComputedStyle; /** * A ComputedStyle represents the computed style data for an element. * * The computed style data are stored in a set of reference counted structs * (see nsStyleStruct.h) that are stored directly on the ComputedStyle. * * Style structs are immutable once they have been produced, so when any change * is made that needs a restyle, we create a new ComputedStyle. * * ComputedStyles are reference counted. References are generally held by: * * 1. nsIFrame::mComputedStyle, for every frame * 2. Element::mServoData, for every element not inside a display:none subtree * 3. nsComputedDOMStyle, when created for elements in display:none subtrees * 4. media_queries::Device, which holds the initial value of every property */ enum class ComputedStyleBit : uint8_t { HasTextDecorationLines = 1 << 0, HasPseudoElementData = 1 << 1, SuppressLineBreak = 1 << 2, IsTextCombined = 1 << 3, RelevantLinkVisited = 1 << 4, }; MOZ_MAKE_ENUM_CLASS_BITWISE_OPERATORS(ComputedStyleBit) class ComputedStyle { using Bit = ComputedStyleBit; public: ComputedStyle(nsAtom* aPseudoTag, CSSPseudoElementType aPseudoType, ServoComputedDataForgotten aComputedValues); void AddRef() { Servo_ComputedStyle_AddRef(this); } void Release() { Servo_ComputedStyle_Release(this); } // Return the ComputedStyle whose style data should be used for the R, // G, and B components of color, background-color, and border-*-color // if RelevantLinkIsVisited(). // // GetPseudo() and GetPseudoType() on this ComputedStyle return the // same as on |this|, and its depth in the tree (number of GetParent() // calls until null is returned) is the same as |this|, since its // parent is either |this|'s parent or |this|'s parent's // style-if-visited. // // Structs on this context should never be examined without also // examining the corresponding struct on |this|. Doing so will likely // both (1) lead to a privacy leak and (2) lead to dynamic change bugs // related to the Peek code in ComputedStyle::CalcStyleDifference. ComputedStyle* GetStyleIfVisited() const { return mSource.visited_style.mPtr; } bool IsLazilyCascadedPseudoElement() const { return IsPseudoElement() && !nsCSSPseudoElements::IsEagerlyCascadedInServo(GetPseudoType()); } nsAtom* GetPseudo() const { return mPseudoTag; } mozilla::CSSPseudoElementType GetPseudoType() const { return mPseudoType; } bool IsInheritingAnonBox() const { return GetPseudoType() == mozilla::CSSPseudoElementType::InheritingAnonBox; } bool IsNonInheritingAnonBox() const { return GetPseudoType() == mozilla::CSSPseudoElementType::NonInheritingAnonBox; } // This function is rather slow; you probably don't want to use it outside // asserts unless you have to. We _could_ add a new CSSPseudoElementType for // wrapper anon boxes, but that adds a bunch of complexity everywhere we // resolve anonymous box styles... bool IsWrapperAnonBox() const { return nsCSSAnonBoxes::IsWrapperAnonBox(GetPseudo()); } bool IsAnonBox() const { return IsInheritingAnonBox() || IsNonInheritingAnonBox(); } bool IsPseudoElement() const { return mPseudoTag && !IsAnonBox(); } // Does this ComputedStyle or any of its ancestors have text // decoration lines? // Differs from nsStyleTextReset::HasTextDecorationLines, which tests // only the data for a single context. bool HasTextDecorationLines() const { return bool(mBits & Bit::HasTextDecorationLines); } // Whether any line break inside should be suppressed? If this returns // true, the line should not be broken inside, which means inlines act // as if nowrap is set,
is suppressed, and blocks are inlinized. // This bit is propogated to all children of line partitipants. It is // currently used by ruby to make its content frames unbreakable. // NOTE: for nsTextFrame, use nsTextFrame::ShouldSuppressLineBreak() // instead of this method. bool ShouldSuppressLineBreak() const { return bool(mBits & Bit::SuppressLineBreak); } // Is this horizontal-in-vertical (tate-chu-yoko) text? This flag is // only set on ComputedStyles whose pseudo is nsCSSAnonBoxes::mozText(). bool IsTextCombined() const { return bool(mBits & Bit::IsTextCombined); } // Does this ComputedStyle represent the style for a pseudo-element or // inherit data from such a ComputedStyle? Whether this returns true // is equivalent to whether it or any of its ancestors returns // non-null for IsPseudoElement(). bool HasPseudoElementData() const { return bool(mBits & Bit::HasPseudoElementData); } // Is the only link whose visitedness is allowed to influence the // style of the node this ComputedStyle is for (which is that element // or its nearest ancestor that is a link) visited? bool RelevantLinkVisited() const { return bool(mBits & Bit::RelevantLinkVisited); } ComputedStyle* GetCachedInheritingAnonBoxStyle(nsAtom* aAnonBox) const { MOZ_ASSERT(nsCSSAnonBoxes::IsInheritingAnonBox(aAnonBox)); return mCachedInheritingStyles.Lookup(aAnonBox); } void SetCachedInheritedAnonBoxStyle(nsAtom* aAnonBox, ComputedStyle* aStyle) { MOZ_ASSERT(!GetCachedInheritingAnonBoxStyle(aAnonBox)); mCachedInheritingStyles.Insert(aStyle); } ComputedStyle* GetCachedLazyPseudoStyle(CSSPseudoElementType aPseudo) const; void SetCachedLazyPseudoStyle(ComputedStyle* aStyle) { MOZ_ASSERT(aStyle->GetPseudo() && !aStyle->IsAnonBox()); MOZ_ASSERT(!GetCachedLazyPseudoStyle(aStyle->GetPseudoType())); MOZ_ASSERT(!IsLazilyCascadedPseudoElement(), "lazy pseudos can't inherit lazy pseudos"); MOZ_ASSERT(aStyle->IsLazilyCascadedPseudoElement()); // Since we're caching lazy pseudo styles on the ComputedValues of the // originating element, we can assume that we either have the same // originating element, or that they were at least similar enough to share // the same ComputedValues, which means that they would match the same // pseudo rules. This allows us to avoid matching selectors and checking // the rule node before deciding to share. // // The one place this optimization breaks is with pseudo-elements that // support state (like :hover). So we just avoid sharing in those cases. if (nsCSSPseudoElements::PseudoElementSupportsUserActionState( aStyle->GetPseudoType())) { return; } mCachedInheritingStyles.Insert(aStyle); } #define STYLE_STRUCT(name_) \ inline const nsStyle##name_* Style##name_() const MOZ_NONNULL_RETURN { \ return mSource.GetStyle##name_(); \ } #include "nsStyleStructList.h" #undef STYLE_STRUCT /** * Compute the style changes needed during restyling when this style * context is being replaced by aNewContext. (This is nonsymmetric since * we optimize by skipping comparison for styles that have never been * requested.) * * This method returns a change hint (see nsChangeHint.h). All change * hints apply to the frame and its later continuations or ib-split * siblings. Most (all of those except the "NotHandledForDescendants" * hints) also apply to all descendants. * * aEqualStructs must not be null. Into it will be stored a bitfield * representing which structs were compared to be non-equal. * * CSS Variables are not compared here. Instead, the caller is responsible for * that when needed (basically only for elements). */ nsChangeHint CalcStyleDifference(const ComputedStyle& aNewContext, uint32_t* aEqualStructs) const; public: /** * Get a color that depends on link-visitedness using this and * this->GetStyleIfVisited(). * * @param aField A pointer to a member variable in a style struct. * The member variable and its style struct must have * been listed in nsCSSVisitedDependentPropList.h. */ template nscolor GetVisitedDependentColor(T S::*aField); /** * aColors should be a two element array of nscolor in which the first * color is the unvisited color and the second is the visited color. * * Combine the R, G, and B components of whichever of aColors should * be used based on aLinkIsVisited with the A component of aColors[0]. */ static nscolor CombineVisitedColors(nscolor* aColors, bool aLinkIsVisited); /** * Start image loads for this style. * * The Document is used to get a hand on the image loader. The old style is a * hack for bug 1439285. */ inline void StartImageLoads(dom::Document&, const ComputedStyle* aOldStyle = nullptr); #ifdef DEBUG void List(FILE* out, int32_t aIndent); static const char* StructName(StyleStructID aSID); static Maybe LookupStruct(const nsACString& aName); #endif // The |aCVsSize| outparam on this function is where the actual CVs size // value is added. It's done that way because the callers know which value // the size should be added to. void AddSizeOfIncludingThis(nsWindowSizes& aSizes, size_t* aCVsSize) const; protected: // Needs to be friend so that it can call the destructor without making it // public. friend void ::Gecko_ComputedStyle_Destroy(ComputedStyle*); ~ComputedStyle() = default; ServoComputedData mSource; // A cache of anonymous box and lazy pseudo styles inheriting from this style. CachedInheritingStyles mCachedInheritingStyles; // If this ComputedStyle is for a pseudo-element or anonymous box, // the relevant atom. const RefPtr mPseudoTag; const Bit mBits; const CSSPseudoElementType mPseudoType; }; } // namespace mozilla #endif