summaryrefslogtreecommitdiff
path: root/layout/style/nsRuleNode.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'layout/style/nsRuleNode.cpp')
-rw-r--r--layout/style/nsRuleNode.cpp10736
1 files changed, 10736 insertions, 0 deletions
diff --git a/layout/style/nsRuleNode.cpp b/layout/style/nsRuleNode.cpp
new file mode 100644
index 0000000000..b22002d87d
--- /dev/null
+++ b/layout/style/nsRuleNode.cpp
@@ -0,0 +1,10736 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set ts=2 et sw=2 tw=78: */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+/*
+ * a node in the lexicographic tree of rules that match an element,
+ * responsible for converting the rules' information into computed style
+ */
+
+#include <algorithm>
+
+#include "mozilla/ArrayUtils.h"
+#include "mozilla/Assertions.h"
+#include "mozilla/DebugOnly.h"
+#include "mozilla/Function.h"
+#include "mozilla/dom/AnimationEffectReadOnlyBinding.h" // for PlaybackDirection
+#include "mozilla/Likely.h"
+#include "mozilla/LookAndFeel.h"
+#include "mozilla/OperatorNewExtensions.h"
+#include "mozilla/Unused.h"
+
+#include "mozilla/css/Declaration.h"
+
+#include "nsAlgorithm.h" // for clamped()
+#include "nsRuleNode.h"
+#include "nscore.h"
+#include "nsIWidget.h"
+#include "nsIPresShell.h"
+#include "nsFontMetrics.h"
+#include "gfxFont.h"
+#include "nsCSSAnonBoxes.h"
+#include "nsCSSPseudoElements.h"
+#include "nsThemeConstants.h"
+#include "PLDHashTable.h"
+#include "nsStyleContext.h"
+#include "nsStyleSet.h"
+#include "nsStyleStruct.h"
+#include "nsSize.h"
+#include "nsRuleData.h"
+#include "nsIStyleRule.h"
+#include "nsBidiUtils.h"
+#include "nsStyleStructInlines.h"
+#include "nsCSSProps.h"
+#include "nsTArray.h"
+#include "nsContentUtils.h"
+#include "CSSCalc.h"
+#include "nsPrintfCString.h"
+#include "nsRenderingContext.h"
+#include "nsStyleUtil.h"
+#include "nsIDocument.h"
+#include "prtime.h"
+#include "CSSVariableResolver.h"
+#include "nsCSSParser.h"
+#include "CounterStyleManager.h"
+#include "nsCSSPropertyIDSet.h"
+#include "mozilla/RuleNodeCacheConditions.h"
+#include "nsDeviceContext.h"
+#include "nsQueryObject.h"
+#include "nsUnicodeProperties.h"
+
+#if defined(_MSC_VER) || defined(__MINGW32__)
+#include <malloc.h>
+#ifdef _MSC_VER
+#define alloca _alloca
+#endif
+#endif
+#ifdef SOLARIS
+#include <alloca.h>
+#endif
+
+using std::max;
+using std::min;
+using namespace mozilla;
+using namespace mozilla::dom;
+
+namespace mozilla {
+
+enum UnsetAction
+{
+ eUnsetInitial,
+ eUnsetInherit
+};
+
+} // namespace mozilla
+
+void*
+nsConditionalResetStyleData::GetConditionalStyleData(nsStyleStructID aSID,
+ nsStyleContext* aStyleContext) const
+{
+ Entry* e = static_cast<Entry*>(mEntries[aSID]);
+ MOZ_ASSERT(e, "if mConditionalBits bit is set, we must have at least one "
+ "conditional style struct");
+ do {
+ if (e->mConditions.Matches(aStyleContext)) {
+ void* data = e->mStyleStruct;
+
+ // For reset structs with conditions, we cache the data on the
+ // style context.
+ // Tell the style context that it doesn't own the data
+ aStyleContext->AddStyleBit(GetBitForSID(aSID));
+ aStyleContext->SetStyle(aSID, data);
+
+ return data;
+ }
+ e = e->mNext;
+ } while (e);
+ return nullptr;
+}
+
+// Creates and returns an imgRequestProxy based on the specified
+// value in aValue.
+static imgRequestProxy*
+GetImageRequest(nsPresContext* aPresContext, const nsCSSValue& aValue)
+{
+ return aValue.GetImageValue(aPresContext->Document());
+}
+
+// Creates an imgRequestProxy based on the specified value in
+// aValue and calls aCallback with it. If the nsPresContext
+// is static (e.g. for printing), then a static request (i.e.
+// showing the first frame, without animation) will be created.
+// (The expectation is then that aCallback will set the resulting
+// imgRequestProxy in a style struct somewhere.)
+static void
+SetImageRequest(function<void(imgRequestProxy*)> aCallback,
+ nsPresContext* aPresContext,
+ const nsCSSValue& aValue)
+{
+ RefPtr<imgRequestProxy> req =
+ aValue.GetPossiblyStaticImageValue(aPresContext->Document(),
+ aPresContext);
+ aCallback(req);
+}
+
+static void
+SetStyleImageRequest(function<void(nsStyleImageRequest*)> aCallback,
+ nsPresContext* aPresContext,
+ const nsCSSValue& aValue,
+ nsStyleImageRequest::Mode aModeFlags =
+ nsStyleImageRequest::Mode::Track)
+{
+ SetImageRequest([&](imgRequestProxy* aProxy) {
+ RefPtr<nsStyleImageRequest> request;
+ if (aProxy) {
+ css::ImageValue* imageValue = aValue.GetImageStructValue();
+ ImageTracker* imageTracker =
+ (aModeFlags & nsStyleImageRequest::Mode::Track)
+ ? aPresContext->Document()->ImageTracker()
+ : nullptr;
+ request =
+ new nsStyleImageRequest(aModeFlags, aProxy, imageValue, imageTracker);
+ }
+ aCallback(request);
+ }, aPresContext, aValue);
+}
+
+template<typename ReferenceBox>
+static void
+SetStyleShapeSourceToCSSValue(StyleShapeSource<ReferenceBox>* aShapeSource,
+ const nsCSSValue* aValue,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions);
+
+/* Helper function to convert a CSS <position> specified value into its
+ * computed-style form. */
+static void
+ComputePositionValue(nsStyleContext* aStyleContext,
+ const nsCSSValue& aValue,
+ Position& aComputedValue,
+ RuleNodeCacheConditions& aConditions);
+
+/*
+ * For storage of an |nsRuleNode|'s children in a PLDHashTable.
+ */
+
+struct ChildrenHashEntry : public PLDHashEntryHdr {
+ // key is |mRuleNode->GetKey()|
+ nsRuleNode *mRuleNode;
+};
+
+/* static */ PLDHashNumber
+nsRuleNode::ChildrenHashHashKey(const void *aKey)
+{
+ const nsRuleNode::Key *key =
+ static_cast<const nsRuleNode::Key*>(aKey);
+ // Disagreement on importance and level for the same rule is extremely
+ // rare, so hash just on the rule.
+ return PLDHashTable::HashVoidPtrKeyStub(key->mRule);
+}
+
+/* static */ bool
+nsRuleNode::ChildrenHashMatchEntry(const PLDHashEntryHdr *aHdr,
+ const void *aKey)
+{
+ const ChildrenHashEntry *entry =
+ static_cast<const ChildrenHashEntry*>(aHdr);
+ const nsRuleNode::Key *key =
+ static_cast<const nsRuleNode::Key*>(aKey);
+ return entry->mRuleNode->GetKey() == *key;
+}
+
+/* static */ const PLDHashTableOps
+nsRuleNode::ChildrenHashOps = {
+ // It's probably better to allocate the table itself using malloc and
+ // free rather than the pres shell's arena because the table doesn't
+ // grow very often and the pres shell's arena doesn't recycle very
+ // large size allocations.
+ ChildrenHashHashKey,
+ ChildrenHashMatchEntry,
+ PLDHashTable::MoveEntryStub,
+ PLDHashTable::ClearEntryStub,
+ nullptr
+};
+
+
+// EnsureBlockDisplay:
+// Never change display:none or display:contents *ever*, otherwise:
+// - if the display value (argument) is not a block-type
+// then we set it to a valid block display value
+// - For enforcing the floated/positioned element CSS2 rules
+// - We allow the behavior of "list-item" to be customized.
+// CSS21 says that position/float do not convert 'list-item' to 'block',
+// but it explicitly does not define whether 'list-item' should be
+// converted to block *on the root node*. To allow for flexibility
+// (so that we don't have to support a list-item root node), this method
+// lets the caller pick either behavior, using the 'aConvertListItem' arg.
+// Reference: http://www.w3.org/TR/CSS21/visuren.html#dis-pos-flo
+/* static */
+void
+nsRuleNode::EnsureBlockDisplay(StyleDisplay& display,
+ bool aConvertListItem /* = false */)
+{
+ // see if the display value is already a block
+ switch (display) {
+ case StyleDisplay::ListItem:
+ if (aConvertListItem) {
+ display = StyleDisplay::Block;
+ break;
+ } // else, fall through to share the 'break' for non-changing display vals
+ MOZ_FALLTHROUGH;
+ case StyleDisplay::None:
+ case StyleDisplay::Contents:
+ // never change display:none or display:contents *ever*
+ case StyleDisplay::Table:
+ case StyleDisplay::Block:
+ case StyleDisplay::Flex:
+ case StyleDisplay::WebkitBox:
+ case StyleDisplay::Grid:
+ // do not muck with these at all - already blocks
+ // This is equivalent to nsStyleDisplay::IsBlockOutside. (XXX Maybe we
+ // should just call that?)
+ // This needs to match the check done in
+ // nsCSSFrameConstructor::FindMathMLData for <math>.
+ break;
+
+ case StyleDisplay::InlineTable:
+ // make inline tables into tables
+ display = StyleDisplay::Table;
+ break;
+
+ case StyleDisplay::InlineFlex:
+ // make inline flex containers into flex containers
+ display = StyleDisplay::Flex;
+ break;
+
+ case StyleDisplay::WebkitInlineBox:
+ // make -webkit-inline-box containers into -webkit-box containers
+ display = StyleDisplay::WebkitBox;
+ break;
+
+ case StyleDisplay::InlineGrid:
+ // make inline grid containers into grid containers
+ display = StyleDisplay::Grid;
+ break;
+
+ default:
+ // make it a block
+ display = StyleDisplay::Block;
+ }
+}
+
+// EnsureInlineDisplay:
+// - if the display value (argument) is not an inline type
+// then we set it to a valid inline display value
+/* static */
+void
+nsRuleNode::EnsureInlineDisplay(StyleDisplay& display)
+{
+ // see if the display value is already inline
+ switch (display) {
+ case StyleDisplay::Block:
+ display = StyleDisplay::InlineBlock;
+ break;
+ case StyleDisplay::Table:
+ display = StyleDisplay::InlineTable;
+ break;
+ case StyleDisplay::Flex:
+ display = StyleDisplay::InlineFlex;
+ break;
+ case StyleDisplay::WebkitBox:
+ display = StyleDisplay::WebkitInlineBox;
+ break;
+ case StyleDisplay::Grid:
+ display = StyleDisplay::InlineGrid;
+ break;
+ case StyleDisplay::Box:
+ display = StyleDisplay::InlineBox;
+ break;
+ case StyleDisplay::Stack:
+ display = StyleDisplay::InlineStack;
+ break;
+ default:
+ break; // Do nothing
+ }
+}
+
+static nscoord CalcLengthWith(const nsCSSValue& aValue,
+ nscoord aFontSize,
+ const nsStyleFont* aStyleFont,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ bool aUseProvidedRootEmSize,
+ bool aUseUserFontSet,
+ RuleNodeCacheConditions& aConditions);
+
+struct CalcLengthCalcOps : public css::BasicCoordCalcOps,
+ public css::NumbersAlreadyNormalizedOps
+{
+ // All of the parameters to CalcLengthWith except aValue.
+ const nscoord mFontSize;
+ const nsStyleFont* const mStyleFont;
+ nsStyleContext* const mStyleContext;
+ nsPresContext* const mPresContext;
+ const bool mUseProvidedRootEmSize;
+ const bool mUseUserFontSet;
+ RuleNodeCacheConditions& mConditions;
+
+ CalcLengthCalcOps(nscoord aFontSize, const nsStyleFont* aStyleFont,
+ nsStyleContext* aStyleContext, nsPresContext* aPresContext,
+ bool aUseProvidedRootEmSize, bool aUseUserFontSet,
+ RuleNodeCacheConditions& aConditions)
+ : mFontSize(aFontSize),
+ mStyleFont(aStyleFont),
+ mStyleContext(aStyleContext),
+ mPresContext(aPresContext),
+ mUseProvidedRootEmSize(aUseProvidedRootEmSize),
+ mUseUserFontSet(aUseUserFontSet),
+ mConditions(aConditions)
+ {
+ }
+
+ result_type ComputeLeafValue(const nsCSSValue& aValue)
+ {
+ return CalcLengthWith(aValue, mFontSize, mStyleFont,
+ mStyleContext, mPresContext, mUseProvidedRootEmSize,
+ mUseUserFontSet, mConditions);
+ }
+};
+
+static inline nscoord ScaleCoordRound(const nsCSSValue& aValue, float aFactor)
+{
+ return NSToCoordRoundWithClamp(aValue.GetFloatValue() * aFactor);
+}
+
+static inline nscoord ScaleViewportCoordTrunc(const nsCSSValue& aValue,
+ nscoord aViewportSize)
+{
+ // For units (like percentages and viewport units) where authors might
+ // repeatedly use a value and expect some multiple of the value to be
+ // smaller than a container, we need to use floor rather than round.
+ // We need to use division by 100.0 rather than multiplication by 0.1f
+ // to avoid introducing error.
+ return NSToCoordTruncClamped(aValue.GetFloatValue() *
+ aViewportSize / 100.0f);
+}
+
+already_AddRefed<nsFontMetrics>
+GetMetricsFor(nsPresContext* aPresContext,
+ nsStyleContext* aStyleContext,
+ const nsStyleFont* aStyleFont,
+ nscoord aFontSize, // overrides value from aStyleFont
+ bool aUseUserFontSet)
+{
+ nsFont font = aStyleFont->mFont;
+ font.size = aFontSize;
+ gfxFont::Orientation orientation = gfxFont::eHorizontal;
+ if (aStyleContext) {
+ WritingMode wm(aStyleContext);
+ if (wm.IsVertical() && !wm.IsSideways()) {
+ orientation = gfxFont::eVertical;
+ }
+ }
+ nsFontMetrics::Params params;
+ params.language = aStyleFont->mLanguage;
+ params.explicitLanguage = aStyleFont->mExplicitLanguage;
+ params.orientation = orientation;
+ params.userFontSet =
+ aUseUserFontSet ? aPresContext->GetUserFontSet() : nullptr;
+ params.textPerf = aPresContext->GetTextPerfMetrics();
+ return aPresContext->DeviceContext()->GetMetricsFor(font, params);
+}
+
+
+static nsSize CalcViewportUnitsScale(nsPresContext* aPresContext)
+{
+ // The caller is making use of viewport units, so notify the pres context
+ // that it will need to rebuild the rule tree if the size of the viewport
+ // changes.
+ aPresContext->SetUsesViewportUnits(true);
+
+ // The default (when we have 'overflow: auto' on the root element, or
+ // trivially for 'overflow: hidden' since we never have scrollbars in that
+ // case) is to define the scale of the viewport units without considering
+ // scrollbars.
+ nsSize viewportSize(aPresContext->GetVisibleArea().Size());
+
+ // Check for 'overflow: scroll' styles on the root scroll frame. If we find
+ // any, the standard requires us to take scrollbars into account.
+ nsIScrollableFrame* scrollFrame =
+ aPresContext->PresShell()->GetRootScrollFrameAsScrollable();
+ if (scrollFrame) {
+ ScrollbarStyles styles(scrollFrame->GetScrollbarStyles());
+
+ if (styles.mHorizontal == NS_STYLE_OVERFLOW_SCROLL ||
+ styles.mVertical == NS_STYLE_OVERFLOW_SCROLL) {
+ // Gather scrollbar size information.
+ nsRenderingContext context(
+ aPresContext->PresShell()->CreateReferenceRenderingContext());
+ nsMargin sizes(scrollFrame->GetDesiredScrollbarSizes(aPresContext, &context));
+
+ if (styles.mHorizontal == NS_STYLE_OVERFLOW_SCROLL) {
+ // 'overflow-x: scroll' means we must consider the horizontal scrollbar,
+ // which affects the scale of viewport height units.
+ viewportSize.height -= sizes.TopBottom();
+ }
+
+ if (styles.mVertical == NS_STYLE_OVERFLOW_SCROLL) {
+ // 'overflow-y: scroll' means we must consider the vertical scrollbar,
+ // which affects the scale of viewport width units.
+ viewportSize.width -= sizes.LeftRight();
+ }
+ }
+ }
+
+ return viewportSize;
+}
+
+// If |aStyleFont| is nullptr, aStyleContext->StyleFont() is used.
+//
+// In case that |aValue| is rem unit, if |aStyleContext| is null, callers must
+// specify a valid |aStyleFont| and |aUseProvidedRootEmSize| must be true so
+// that we can get the length from |aStyleFont|.
+static nscoord CalcLengthWith(const nsCSSValue& aValue,
+ nscoord aFontSize,
+ const nsStyleFont* aStyleFont,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ bool aUseProvidedRootEmSize,
+ // aUseUserFontSet should always be true
+ // except when called from
+ // CalcLengthWithInitialFont.
+ bool aUseUserFontSet,
+ RuleNodeCacheConditions& aConditions)
+{
+ NS_ASSERTION(aValue.IsLengthUnit() || aValue.IsCalcUnit(),
+ "not a length or calc unit");
+ NS_ASSERTION(aStyleFont || aStyleContext,
+ "Must have style data");
+ NS_ASSERTION(aStyleContext || aUseProvidedRootEmSize,
+ "Must have style context or specify aUseProvidedRootEmSize");
+ NS_ASSERTION(aPresContext, "Must have prescontext");
+
+ if (aValue.IsFixedLengthUnit()) {
+ return aValue.GetFixedLength(aPresContext);
+ }
+ if (aValue.IsPixelLengthUnit()) {
+ return aValue.GetPixelLength();
+ }
+ if (aValue.IsCalcUnit()) {
+ // For properties for which lengths are the *only* units accepted in
+ // calc(), we can handle calc() here and just compute a final
+ // result. We ensure that we don't get to this code for other
+ // properties by not calling CalcLength in those cases: SetCoord
+ // only calls CalcLength for a calc when it is appropriate to do so.
+ CalcLengthCalcOps ops(aFontSize, aStyleFont,
+ aStyleContext, aPresContext,
+ aUseProvidedRootEmSize, aUseUserFontSet,
+ aConditions);
+ return css::ComputeCalc(aValue, ops);
+ }
+ switch (aValue.GetUnit()) {
+ // nsPresContext::SetVisibleArea and
+ // nsPresContext::MediaFeatureValuesChanged handle dynamic changes
+ // of the basis for viewport units by rebuilding the rule tree and
+ // style context tree. Not caching them in the rule tree wouldn't
+ // be sufficient to handle these changes because we also need a way
+ // to get rid of cached values in the style context tree without any
+ // changes in specified style. We can either do this by not caching
+ // in the rule tree and then throwing away the style context tree
+ // for dynamic viewport size changes, or by allowing caching in the
+ // rule tree and using the existing rebuild style data path that
+ // throws away the style context and the rule tree.
+ // Thus we do cache viewport units in the rule tree. This allows us
+ // to benefit from the performance advantages of the rule tree
+ // (e.g., faster dynamic changes on other things, like transforms)
+ // and allows us not to need an additional code path, in exchange
+ // for an increased cost to dynamic changes to the viewport size
+ // when viewport units are in use.
+ case eCSSUnit_ViewportWidth: {
+ nscoord viewportWidth = CalcViewportUnitsScale(aPresContext).width;
+ return ScaleViewportCoordTrunc(aValue, viewportWidth);
+ }
+ case eCSSUnit_ViewportHeight: {
+ nscoord viewportHeight = CalcViewportUnitsScale(aPresContext).height;
+ return ScaleViewportCoordTrunc(aValue, viewportHeight);
+ }
+ case eCSSUnit_ViewportMin: {
+ nsSize vuScale(CalcViewportUnitsScale(aPresContext));
+ nscoord viewportMin = min(vuScale.width, vuScale.height);
+ return ScaleViewportCoordTrunc(aValue, viewportMin);
+ }
+ case eCSSUnit_ViewportMax: {
+ nsSize vuScale(CalcViewportUnitsScale(aPresContext));
+ nscoord viewportMax = max(vuScale.width, vuScale.height);
+ return ScaleViewportCoordTrunc(aValue, viewportMax);
+ }
+ // While we could deal with 'rem' units correctly by simply not
+ // caching any data that uses them in the rule tree, it's valuable
+ // to store them in the rule tree (for faster dynamic changes of
+ // other things). And since the font size of the root element
+ // changes rarely, we instead handle dynamic changes to the root
+ // element's font size by rebuilding all style data in
+ // nsCSSFrameConstructor::RestyleElement.
+ case eCSSUnit_RootEM: {
+ aPresContext->SetUsesRootEMUnits(true);
+ nscoord rootFontSize;
+
+ // NOTE: Be very careful with |styleFont|, since we haven't added any
+ // conditions to aConditions or set it to uncacheable yet, so we don't
+ // want to introduce any dependencies on aStyleContext's data here.
+ const nsStyleFont *styleFont =
+ aStyleFont ? aStyleFont : aStyleContext->StyleFont();
+
+ if (aUseProvidedRootEmSize) {
+ // We should use the provided aFontSize as the reference length to
+ // scale. This only happens when we are calculating font-size or
+ // an equivalent (scriptminsize or CalcLengthWithInitialFont) on
+ // the root element, in which case aFontSize is already the
+ // value we want.
+ if (aFontSize == -1) {
+ // XXX Should this be styleFont->mSize instead to avoid taking
+ // minfontsize prefs into account?
+ aFontSize = styleFont->mFont.size;
+ }
+ rootFontSize = aFontSize;
+ } else if (aStyleContext && !aStyleContext->GetParent()) {
+ // This is the root element (XXX we don't really know this, but
+ // nsRuleNode::SetFont makes the same assumption!), so we should
+ // use StyleFont on this context to get the root element's
+ // font size.
+ rootFontSize = styleFont->mFont.size;
+ } else {
+ // This is not the root element or we are calculating something other
+ // than font size, so rem is relative to the root element's font size.
+ // Find the root style context by walking up the style context tree.
+ nsStyleContext* rootStyle = aStyleContext;
+ while (rootStyle->GetParent()) {
+ rootStyle = rootStyle->GetParent();
+ }
+
+ const nsStyleFont *rootStyleFont = rootStyle->StyleFont();
+ rootFontSize = rootStyleFont->mFont.size;
+ }
+
+ return ScaleCoordRound(aValue, float(rootFontSize));
+ }
+ default:
+ // Fall through to the code for units that can't be stored in the
+ // rule tree because they depend on font data.
+ break;
+ }
+ // Common code for units that depend on the element's font data and
+ // thus can't be stored in the rule tree:
+ const nsStyleFont *styleFont =
+ aStyleFont ? aStyleFont : aStyleContext->StyleFont();
+ if (aFontSize == -1) {
+ // XXX Should this be styleFont->mSize instead to avoid taking minfontsize
+ // prefs into account?
+ aFontSize = styleFont->mFont.size;
+ }
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_EM: {
+ if (aValue.GetFloatValue() == 0.0f) {
+ // Don't call SetFontSizeDependency for '0em'.
+ return 0;
+ }
+ // CSS2.1 specifies that this unit scales to the computed font
+ // size, not the em-width in the font metrics, despite the name.
+ aConditions.SetFontSizeDependency(aFontSize);
+ return ScaleCoordRound(aValue, float(aFontSize));
+ }
+ case eCSSUnit_XHeight: {
+ aPresContext->SetUsesExChUnits(true);
+ RefPtr<nsFontMetrics> fm =
+ GetMetricsFor(aPresContext, aStyleContext, styleFont,
+ aFontSize, aUseUserFontSet);
+ aConditions.SetUncacheable();
+ return ScaleCoordRound(aValue, float(fm->XHeight()));
+ }
+ case eCSSUnit_Char: {
+ aPresContext->SetUsesExChUnits(true);
+ RefPtr<nsFontMetrics> fm =
+ GetMetricsFor(aPresContext, aStyleContext, styleFont,
+ aFontSize, aUseUserFontSet);
+ gfxFloat zeroWidth =
+ fm->GetThebesFontGroup()->GetFirstValidFont()->
+ GetMetrics(fm->Orientation()).zeroOrAveCharWidth;
+
+ aConditions.SetUncacheable();
+ return ScaleCoordRound(aValue, ceil(aPresContext->AppUnitsPerDevPixel() *
+ zeroWidth));
+ }
+ default:
+ NS_NOTREACHED("unexpected unit");
+ break;
+ }
+ return 0;
+}
+
+/* static */ nscoord
+nsRuleNode::CalcLength(const nsCSSValue& aValue,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+{
+ NS_ASSERTION(aStyleContext, "Must have style data");
+
+ return CalcLengthWith(aValue, -1, nullptr,
+ aStyleContext, aPresContext,
+ false, true, aConditions);
+}
+
+/* Inline helper function to redirect requests to CalcLength. */
+static inline nscoord CalcLength(const nsCSSValue& aValue,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+{
+ return nsRuleNode::CalcLength(aValue, aStyleContext,
+ aPresContext, aConditions);
+}
+
+/* static */ nscoord
+nsRuleNode::CalcLengthWithInitialFont(nsPresContext* aPresContext,
+ const nsCSSValue& aValue)
+{
+ nsStyleFont defaultFont(aPresContext); // FIXME: best language?
+ RuleNodeCacheConditions conditions;
+ return CalcLengthWith(aValue, -1, &defaultFont,
+ nullptr, aPresContext,
+ true, false, conditions);
+}
+
+struct LengthPercentPairCalcOps : public css::NumbersAlreadyNormalizedOps
+{
+ typedef nsRuleNode::ComputedCalc result_type;
+
+ LengthPercentPairCalcOps(nsStyleContext* aContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+ : mContext(aContext),
+ mPresContext(aPresContext),
+ mConditions(aConditions),
+ mHasPercent(false) {}
+
+ nsStyleContext* mContext;
+ nsPresContext* mPresContext;
+ RuleNodeCacheConditions& mConditions;
+ bool mHasPercent;
+
+ result_type ComputeLeafValue(const nsCSSValue& aValue)
+ {
+ if (aValue.GetUnit() == eCSSUnit_Percent) {
+ mHasPercent = true;
+ return result_type(0, aValue.GetPercentValue());
+ }
+ return result_type(CalcLength(aValue, mContext, mPresContext,
+ mConditions),
+ 0.0f);
+ }
+
+ result_type
+ MergeAdditive(nsCSSUnit aCalcFunction,
+ result_type aValue1, result_type aValue2)
+ {
+ if (aCalcFunction == eCSSUnit_Calc_Plus) {
+ return result_type(NSCoordSaturatingAdd(aValue1.mLength,
+ aValue2.mLength),
+ aValue1.mPercent + aValue2.mPercent);
+ }
+ MOZ_ASSERT(aCalcFunction == eCSSUnit_Calc_Minus,
+ "min() and max() are not allowed in calc() on transform");
+ return result_type(NSCoordSaturatingSubtract(aValue1.mLength,
+ aValue2.mLength, 0),
+ aValue1.mPercent - aValue2.mPercent);
+ }
+
+ result_type
+ MergeMultiplicativeL(nsCSSUnit aCalcFunction,
+ float aValue1, result_type aValue2)
+ {
+ MOZ_ASSERT(aCalcFunction == eCSSUnit_Calc_Times_L,
+ "unexpected unit");
+ return result_type(NSCoordSaturatingMultiply(aValue2.mLength, aValue1),
+ aValue1 * aValue2.mPercent);
+ }
+
+ result_type
+ MergeMultiplicativeR(nsCSSUnit aCalcFunction,
+ result_type aValue1, float aValue2)
+ {
+ MOZ_ASSERT(aCalcFunction == eCSSUnit_Calc_Times_R ||
+ aCalcFunction == eCSSUnit_Calc_Divided,
+ "unexpected unit");
+ if (aCalcFunction == eCSSUnit_Calc_Divided) {
+ aValue2 = 1.0f / aValue2;
+ }
+ return result_type(NSCoordSaturatingMultiply(aValue1.mLength, aValue2),
+ aValue1.mPercent * aValue2);
+ }
+
+};
+
+static void
+SpecifiedCalcToComputedCalc(const nsCSSValue& aValue, nsStyleCoord& aCoord,
+ nsStyleContext* aStyleContext,
+ RuleNodeCacheConditions& aConditions)
+{
+ LengthPercentPairCalcOps ops(aStyleContext, aStyleContext->PresContext(),
+ aConditions);
+ nsRuleNode::ComputedCalc vals = ComputeCalc(aValue, ops);
+
+ nsStyleCoord::Calc* calcObj = new nsStyleCoord::Calc;
+
+ calcObj->mLength = vals.mLength;
+ calcObj->mPercent = vals.mPercent;
+ calcObj->mHasPercent = ops.mHasPercent;
+
+ aCoord.SetCalcValue(calcObj);
+}
+
+/* static */ nsRuleNode::ComputedCalc
+nsRuleNode::SpecifiedCalcToComputedCalc(const nsCSSValue& aValue,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+{
+ LengthPercentPairCalcOps ops(aStyleContext, aPresContext,
+ aConditions);
+ return ComputeCalc(aValue, ops);
+}
+
+// This is our public API for handling calc() expressions that involve
+// percentages.
+/* static */ nscoord
+nsRuleNode::ComputeComputedCalc(const nsStyleCoord& aValue,
+ nscoord aPercentageBasis)
+{
+ nsStyleCoord::Calc* calc = aValue.GetCalcValue();
+ return calc->mLength +
+ NSToCoordFloorClamped(aPercentageBasis * calc->mPercent);
+}
+
+/* static */ nscoord
+nsRuleNode::ComputeCoordPercentCalc(const nsStyleCoord& aCoord,
+ nscoord aPercentageBasis)
+{
+ switch (aCoord.GetUnit()) {
+ case eStyleUnit_Coord:
+ return aCoord.GetCoordValue();
+ case eStyleUnit_Percent:
+ return NSToCoordFloorClamped(aPercentageBasis * aCoord.GetPercentValue());
+ case eStyleUnit_Calc:
+ return ComputeComputedCalc(aCoord, aPercentageBasis);
+ default:
+ MOZ_ASSERT(false, "unexpected unit");
+ return 0;
+ }
+}
+
+/* Given an enumerated value that represents a box position, converts it to
+ * a float representing the percentage of the box it corresponds to. For
+ * example, "center" becomes 0.5f.
+ *
+ * @param aEnumValue The enumerated value.
+ * @return The float percent it corresponds to.
+ */
+static float
+GetFloatFromBoxPosition(int32_t aEnumValue)
+{
+ switch (aEnumValue) {
+ case NS_STYLE_IMAGELAYER_POSITION_LEFT:
+ case NS_STYLE_IMAGELAYER_POSITION_TOP:
+ return 0.0f;
+ case NS_STYLE_IMAGELAYER_POSITION_RIGHT:
+ case NS_STYLE_IMAGELAYER_POSITION_BOTTOM:
+ return 1.0f;
+ default:
+ MOZ_FALLTHROUGH_ASSERT("unexpected box position value");
+ case NS_STYLE_IMAGELAYER_POSITION_CENTER:
+ return 0.5f;
+ }
+}
+
+#define SETCOORD_NORMAL 0x01 // N
+#define SETCOORD_AUTO 0x02 // A
+#define SETCOORD_INHERIT 0x04 // H
+#define SETCOORD_PERCENT 0x08 // P
+#define SETCOORD_FACTOR 0x10 // F
+#define SETCOORD_LENGTH 0x20 // L
+#define SETCOORD_INTEGER 0x40 // I
+#define SETCOORD_ENUMERATED 0x80 // E
+#define SETCOORD_NONE 0x100 // O
+#define SETCOORD_INITIAL_ZERO 0x200
+#define SETCOORD_INITIAL_AUTO 0x400
+#define SETCOORD_INITIAL_NONE 0x800
+#define SETCOORD_INITIAL_NORMAL 0x1000
+#define SETCOORD_INITIAL_HALF 0x2000
+#define SETCOORD_INITIAL_HUNDRED_PCT 0x00004000
+#define SETCOORD_INITIAL_FACTOR_ONE 0x00008000
+#define SETCOORD_INITIAL_FACTOR_ZERO 0x00010000
+#define SETCOORD_CALC_LENGTH_ONLY 0x00020000
+#define SETCOORD_CALC_CLAMP_NONNEGATIVE 0x00040000 // modifier for CALC_LENGTH_ONLY
+#define SETCOORD_STORE_CALC 0x00080000
+#define SETCOORD_BOX_POSITION 0x00100000 // exclusive with _ENUMERATED
+#define SETCOORD_ANGLE 0x00200000
+#define SETCOORD_UNSET_INHERIT 0x00400000
+#define SETCOORD_UNSET_INITIAL 0x00800000
+
+#define SETCOORD_LP (SETCOORD_LENGTH | SETCOORD_PERCENT)
+#define SETCOORD_LH (SETCOORD_LENGTH | SETCOORD_INHERIT)
+#define SETCOORD_AH (SETCOORD_AUTO | SETCOORD_INHERIT)
+#define SETCOORD_LAH (SETCOORD_AUTO | SETCOORD_LENGTH | SETCOORD_INHERIT)
+#define SETCOORD_LPH (SETCOORD_LP | SETCOORD_INHERIT)
+#define SETCOORD_LPAH (SETCOORD_LP | SETCOORD_AH)
+#define SETCOORD_LPE (SETCOORD_LP | SETCOORD_ENUMERATED)
+#define SETCOORD_LPEH (SETCOORD_LPE | SETCOORD_INHERIT)
+#define SETCOORD_LPAEH (SETCOORD_LPAH | SETCOORD_ENUMERATED)
+#define SETCOORD_LPO (SETCOORD_LP | SETCOORD_NONE)
+#define SETCOORD_LPOH (SETCOORD_LPH | SETCOORD_NONE)
+#define SETCOORD_LPOEH (SETCOORD_LPOH | SETCOORD_ENUMERATED)
+#define SETCOORD_LE (SETCOORD_LENGTH | SETCOORD_ENUMERATED)
+#define SETCOORD_LEH (SETCOORD_LE | SETCOORD_INHERIT)
+#define SETCOORD_IA (SETCOORD_INTEGER | SETCOORD_AUTO)
+#define SETCOORD_LAE (SETCOORD_LENGTH | SETCOORD_AUTO | SETCOORD_ENUMERATED)
+
+// changes aCoord iff it returns true
+static bool SetCoord(const nsCSSValue& aValue, nsStyleCoord& aCoord,
+ const nsStyleCoord& aParentCoord,
+ int32_t aMask, nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+{
+ bool result = true;
+ if (aValue.GetUnit() == eCSSUnit_Null) {
+ result = false;
+ }
+ else if ((((aMask & SETCOORD_LENGTH) != 0) &&
+ aValue.IsLengthUnit()) ||
+ (((aMask & SETCOORD_CALC_LENGTH_ONLY) != 0) &&
+ aValue.IsCalcUnit())) {
+ nscoord len = CalcLength(aValue, aStyleContext, aPresContext,
+ aConditions);
+ if ((aMask & SETCOORD_CALC_CLAMP_NONNEGATIVE) && len < 0) {
+ NS_ASSERTION(aValue.IsCalcUnit(),
+ "parser should have ensured no nonnegative lengths");
+ len = 0;
+ }
+ aCoord.SetCoordValue(len);
+ }
+ else if (((aMask & SETCOORD_PERCENT) != 0) &&
+ (aValue.GetUnit() == eCSSUnit_Percent)) {
+ aCoord.SetPercentValue(aValue.GetPercentValue());
+ }
+ else if (((aMask & SETCOORD_INTEGER) != 0) &&
+ (aValue.GetUnit() == eCSSUnit_Integer)) {
+ aCoord.SetIntValue(aValue.GetIntValue(), eStyleUnit_Integer);
+ }
+ else if (((aMask & SETCOORD_ENUMERATED) != 0) &&
+ (aValue.GetUnit() == eCSSUnit_Enumerated)) {
+ aCoord.SetIntValue(aValue.GetIntValue(), eStyleUnit_Enumerated);
+ }
+ else if (((aMask & SETCOORD_BOX_POSITION) != 0) &&
+ (aValue.GetUnit() == eCSSUnit_Enumerated)) {
+ aCoord.SetPercentValue(GetFloatFromBoxPosition(aValue.GetIntValue()));
+ }
+ else if (((aMask & SETCOORD_AUTO) != 0) &&
+ (aValue.GetUnit() == eCSSUnit_Auto)) {
+ aCoord.SetAutoValue();
+ }
+ else if ((((aMask & SETCOORD_INHERIT) != 0) &&
+ aValue.GetUnit() == eCSSUnit_Inherit) ||
+ (((aMask & SETCOORD_UNSET_INHERIT) != 0) &&
+ aValue.GetUnit() == eCSSUnit_Unset)) {
+ aCoord = aParentCoord; // just inherit value from parent
+ aConditions.SetUncacheable();
+ }
+ else if (((aMask & SETCOORD_NORMAL) != 0) &&
+ (aValue.GetUnit() == eCSSUnit_Normal)) {
+ aCoord.SetNormalValue();
+ }
+ else if (((aMask & SETCOORD_NONE) != 0) &&
+ (aValue.GetUnit() == eCSSUnit_None)) {
+ aCoord.SetNoneValue();
+ }
+ else if (((aMask & SETCOORD_FACTOR) != 0) &&
+ (aValue.GetUnit() == eCSSUnit_Number)) {
+ aCoord.SetFactorValue(aValue.GetFloatValue());
+ }
+ else if (((aMask & SETCOORD_STORE_CALC) != 0) &&
+ (aValue.IsCalcUnit())) {
+ SpecifiedCalcToComputedCalc(aValue, aCoord, aStyleContext,
+ aConditions);
+ }
+ else if (aValue.GetUnit() == eCSSUnit_Initial ||
+ (aValue.GetUnit() == eCSSUnit_Unset &&
+ ((aMask & SETCOORD_UNSET_INITIAL) != 0))) {
+ if ((aMask & SETCOORD_INITIAL_AUTO) != 0) {
+ aCoord.SetAutoValue();
+ }
+ else if ((aMask & SETCOORD_INITIAL_ZERO) != 0) {
+ aCoord.SetCoordValue(0);
+ }
+ else if ((aMask & SETCOORD_INITIAL_FACTOR_ZERO) != 0) {
+ aCoord.SetFactorValue(0.0f);
+ }
+ else if ((aMask & SETCOORD_INITIAL_NONE) != 0) {
+ aCoord.SetNoneValue();
+ }
+ else if ((aMask & SETCOORD_INITIAL_NORMAL) != 0) {
+ aCoord.SetNormalValue();
+ }
+ else if ((aMask & SETCOORD_INITIAL_HALF) != 0) {
+ aCoord.SetPercentValue(0.5f);
+ }
+ else if ((aMask & SETCOORD_INITIAL_HUNDRED_PCT) != 0) {
+ aCoord.SetPercentValue(1.0f);
+ }
+ else if ((aMask & SETCOORD_INITIAL_FACTOR_ONE) != 0) {
+ aCoord.SetFactorValue(1.0f);
+ }
+ else {
+ result = false; // didn't set anything
+ }
+ }
+ else if ((aMask & SETCOORD_ANGLE) != 0 &&
+ (aValue.IsAngularUnit())) {
+ nsStyleUnit unit;
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Degree: unit = eStyleUnit_Degree; break;
+ case eCSSUnit_Grad: unit = eStyleUnit_Grad; break;
+ case eCSSUnit_Radian: unit = eStyleUnit_Radian; break;
+ case eCSSUnit_Turn: unit = eStyleUnit_Turn; break;
+ default: NS_NOTREACHED("unrecognized angular unit");
+ unit = eStyleUnit_Degree;
+ }
+ aCoord.SetAngleValue(aValue.GetAngleValue(), unit);
+ }
+ else {
+ result = false; // didn't set anything
+ }
+ return result;
+}
+
+// This inline function offers a shortcut for SetCoord() by refusing to accept
+// SETCOORD_LENGTH, SETCOORD_INHERIT and SETCOORD_UNSET_* masks.
+static inline bool SetAbsCoord(const nsCSSValue& aValue,
+ nsStyleCoord& aCoord,
+ int32_t aMask)
+{
+ MOZ_ASSERT((aMask & (SETCOORD_LH | SETCOORD_UNSET_INHERIT |
+ SETCOORD_UNSET_INITIAL)) == 0,
+ "does not handle SETCOORD_LENGTH, SETCOORD_INHERIT and "
+ "SETCOORD_UNSET_*");
+
+ // The values of the following variables will never be used; so it does not
+ // matter what to set.
+ const nsStyleCoord dummyParentCoord;
+ nsStyleContext* dummyStyleContext = nullptr;
+ nsPresContext* dummyPresContext = nullptr;
+ RuleNodeCacheConditions dummyCacheKey;
+
+ bool rv = SetCoord(aValue, aCoord, dummyParentCoord, aMask,
+ dummyStyleContext, dummyPresContext,
+ dummyCacheKey);
+ MOZ_ASSERT(dummyCacheKey.CacheableWithoutDependencies(),
+ "SetCoord() should not modify dummyCacheKey.");
+
+ return rv;
+}
+
+/* Given a specified value that might be a pair value, call SetCoord twice,
+ * either using each member of the pair, or using the unpaired value twice.
+ */
+static bool
+SetPairCoords(const nsCSSValue& aValue,
+ nsStyleCoord& aCoordX, nsStyleCoord& aCoordY,
+ const nsStyleCoord& aParentX, const nsStyleCoord& aParentY,
+ int32_t aMask, nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext, RuleNodeCacheConditions& aConditions)
+{
+ const nsCSSValue& valX =
+ aValue.GetUnit() == eCSSUnit_Pair ? aValue.GetPairValue().mXValue : aValue;
+ const nsCSSValue& valY =
+ aValue.GetUnit() == eCSSUnit_Pair ? aValue.GetPairValue().mYValue : aValue;
+
+ bool cX = SetCoord(valX, aCoordX, aParentX, aMask, aStyleContext,
+ aPresContext, aConditions);
+ mozilla::DebugOnly<bool> cY = SetCoord(valY, aCoordY, aParentY, aMask,
+ aStyleContext, aPresContext, aConditions);
+ MOZ_ASSERT(cX == cY, "changed one but not the other");
+ return cX;
+}
+
+static bool SetColor(const nsCSSValue& aValue, const nscolor aParentColor,
+ nsPresContext* aPresContext, nsStyleContext *aContext,
+ nscolor& aResult, RuleNodeCacheConditions& aConditions)
+{
+ bool result = false;
+ nsCSSUnit unit = aValue.GetUnit();
+
+ if (aValue.IsNumericColorUnit()) {
+ aResult = aValue.GetColorValue();
+ result = true;
+ }
+ else if (eCSSUnit_Ident == unit) {
+ nsAutoString value;
+ aValue.GetStringValue(value);
+ nscolor rgba;
+ if (NS_ColorNameToRGB(value, &rgba)) {
+ aResult = rgba;
+ result = true;
+ }
+ }
+ else if (eCSSUnit_EnumColor == unit) {
+ int32_t intValue = aValue.GetIntValue();
+ if (0 <= intValue) {
+ LookAndFeel::ColorID colorID = (LookAndFeel::ColorID) intValue;
+ bool useStandinsForNativeColors = aPresContext &&
+ !aPresContext->IsChrome();
+ if (NS_SUCCEEDED(LookAndFeel::GetColor(colorID,
+ useStandinsForNativeColors, &aResult))) {
+ result = true;
+ }
+ }
+ else {
+ aResult = NS_RGB(0, 0, 0);
+ result = false;
+ switch (intValue) {
+ case NS_COLOR_MOZ_HYPERLINKTEXT:
+ if (aPresContext) {
+ aResult = aPresContext->DefaultLinkColor();
+ result = true;
+ }
+ break;
+ case NS_COLOR_MOZ_VISITEDHYPERLINKTEXT:
+ if (aPresContext) {
+ aResult = aPresContext->DefaultVisitedLinkColor();
+ result = true;
+ }
+ break;
+ case NS_COLOR_MOZ_ACTIVEHYPERLINKTEXT:
+ if (aPresContext) {
+ aResult = aPresContext->DefaultActiveLinkColor();
+ result = true;
+ }
+ break;
+ case NS_COLOR_CURRENTCOLOR:
+ // The data computed from this can't be shared in the rule tree
+ // because they could be used on a node with a different color
+ aConditions.SetUncacheable();
+ if (aContext) {
+ aResult = aContext->StyleColor()->mColor;
+ result = true;
+ }
+ break;
+ case NS_COLOR_MOZ_DEFAULT_COLOR:
+ if (aPresContext) {
+ aResult = aPresContext->DefaultColor();
+ result = true;
+ }
+ break;
+ case NS_COLOR_MOZ_DEFAULT_BACKGROUND_COLOR:
+ if (aPresContext) {
+ aResult = aPresContext->DefaultBackgroundColor();
+ result = true;
+ }
+ break;
+ default:
+ NS_NOTREACHED("Should never have an unknown negative colorID.");
+ break;
+ }
+ }
+ }
+ else if (eCSSUnit_Inherit == unit) {
+ aResult = aParentColor;
+ result = true;
+ aConditions.SetUncacheable();
+ }
+ else if (eCSSUnit_Enumerated == unit &&
+ aValue.GetIntValue() == NS_STYLE_COLOR_INHERIT_FROM_BODY) {
+ NS_ASSERTION(aPresContext->CompatibilityMode() == eCompatibility_NavQuirks,
+ "Should only get this value in quirks mode");
+ // We just grab the color from the prescontext, and rely on the fact that
+ // if the body color ever changes all its descendants will get new style
+ // contexts (but NOT necessarily new rulenodes).
+ aResult = aPresContext->BodyTextColor();
+ result = true;
+ aConditions.SetUncacheable();
+ }
+ return result;
+}
+
+template<UnsetAction UnsetTo>
+static void
+SetComplexColor(const nsCSSValue& aValue,
+ const StyleComplexColor& aParentColor,
+ const StyleComplexColor& aInitialColor,
+ nsPresContext* aPresContext,
+ StyleComplexColor& aResult,
+ RuleNodeCacheConditions& aConditions)
+{
+ nsCSSUnit unit = aValue.GetUnit();
+ if (unit == eCSSUnit_Null) {
+ return;
+ }
+ if (unit == eCSSUnit_Initial ||
+ (UnsetTo == eUnsetInitial && unit == eCSSUnit_Unset)) {
+ aResult = aInitialColor;
+ } else if (unit == eCSSUnit_Inherit ||
+ (UnsetTo == eUnsetInherit && unit == eCSSUnit_Unset)) {
+ aConditions.SetUncacheable();
+ aResult = aParentColor;
+ } else if (unit == eCSSUnit_EnumColor &&
+ aValue.GetIntValue() == NS_COLOR_CURRENTCOLOR) {
+ aResult = StyleComplexColor::CurrentColor();
+ } else if (unit == eCSSUnit_ComplexColor) {
+ aResult = aValue.GetStyleComplexColorValue();
+ } else {
+ if (!SetColor(aValue, aParentColor.mColor, aPresContext,
+ nullptr, aResult.mColor, aConditions)) {
+ MOZ_ASSERT_UNREACHABLE("Unknown color value");
+ return;
+ }
+ aResult.mForegroundRatio = 0;
+ }
+}
+
+static void SetGradientCoord(const nsCSSValue& aValue, nsPresContext* aPresContext,
+ nsStyleContext* aContext, nsStyleCoord& aResult,
+ RuleNodeCacheConditions& aConditions)
+{
+ // OK to pass bad aParentCoord since we're not passing SETCOORD_INHERIT
+ if (!SetCoord(aValue, aResult, nsStyleCoord(),
+ SETCOORD_LPO | SETCOORD_BOX_POSITION | SETCOORD_STORE_CALC,
+ aContext, aPresContext, aConditions)) {
+ NS_NOTREACHED("unexpected unit for gradient anchor point");
+ aResult.SetNoneValue();
+ }
+}
+
+static void SetGradient(const nsCSSValue& aValue, nsPresContext* aPresContext,
+ nsStyleContext* aContext, nsStyleGradient& aResult,
+ RuleNodeCacheConditions& aConditions)
+{
+ MOZ_ASSERT(aValue.GetUnit() == eCSSUnit_Gradient,
+ "The given data is not a gradient");
+
+ const nsCSSValueGradient* gradient = aValue.GetGradientValue();
+
+ if (gradient->mIsExplicitSize) {
+ SetCoord(gradient->GetRadiusX(), aResult.mRadiusX, nsStyleCoord(),
+ SETCOORD_LP | SETCOORD_STORE_CALC,
+ aContext, aPresContext, aConditions);
+ if (gradient->GetRadiusY().GetUnit() != eCSSUnit_None) {
+ SetCoord(gradient->GetRadiusY(), aResult.mRadiusY, nsStyleCoord(),
+ SETCOORD_LP | SETCOORD_STORE_CALC,
+ aContext, aPresContext, aConditions);
+ aResult.mShape = NS_STYLE_GRADIENT_SHAPE_ELLIPTICAL;
+ } else {
+ aResult.mRadiusY = aResult.mRadiusX;
+ aResult.mShape = NS_STYLE_GRADIENT_SHAPE_CIRCULAR;
+ }
+ aResult.mSize = NS_STYLE_GRADIENT_SIZE_EXPLICIT_SIZE;
+ } else if (gradient->mIsRadial) {
+ if (gradient->GetRadialShape().GetUnit() == eCSSUnit_Enumerated) {
+ aResult.mShape = gradient->GetRadialShape().GetIntValue();
+ } else {
+ NS_ASSERTION(gradient->GetRadialShape().GetUnit() == eCSSUnit_None,
+ "bad unit for radial shape");
+ aResult.mShape = NS_STYLE_GRADIENT_SHAPE_ELLIPTICAL;
+ }
+ if (gradient->GetRadialSize().GetUnit() == eCSSUnit_Enumerated) {
+ aResult.mSize = gradient->GetRadialSize().GetIntValue();
+ } else {
+ NS_ASSERTION(gradient->GetRadialSize().GetUnit() == eCSSUnit_None,
+ "bad unit for radial shape");
+ aResult.mSize = NS_STYLE_GRADIENT_SIZE_FARTHEST_CORNER;
+ }
+ } else {
+ NS_ASSERTION(gradient->GetRadialShape().GetUnit() == eCSSUnit_None,
+ "bad unit for linear shape");
+ NS_ASSERTION(gradient->GetRadialSize().GetUnit() == eCSSUnit_None,
+ "bad unit for linear size");
+ aResult.mShape = NS_STYLE_GRADIENT_SHAPE_LINEAR;
+ aResult.mSize = NS_STYLE_GRADIENT_SIZE_FARTHEST_CORNER;
+ }
+
+ aResult.mLegacySyntax = gradient->mIsLegacySyntax;
+
+ // bg-position
+ SetGradientCoord(gradient->mBgPos.mXValue, aPresContext, aContext,
+ aResult.mBgPosX, aConditions);
+
+ SetGradientCoord(gradient->mBgPos.mYValue, aPresContext, aContext,
+ aResult.mBgPosY, aConditions);
+
+ aResult.mRepeating = gradient->mIsRepeating;
+
+ // angle
+ const nsStyleCoord dummyParentCoord;
+ if (!SetCoord(gradient->mAngle, aResult.mAngle, dummyParentCoord, SETCOORD_ANGLE,
+ aContext, aPresContext, aConditions)) {
+ NS_ASSERTION(gradient->mAngle.GetUnit() == eCSSUnit_None,
+ "bad unit for gradient angle");
+ aResult.mAngle.SetNoneValue();
+ }
+
+ // stops
+ for (uint32_t i = 0; i < gradient->mStops.Length(); i++) {
+ nsStyleGradientStop stop;
+ const nsCSSValueGradientStop &valueStop = gradient->mStops[i];
+
+ if (!SetCoord(valueStop.mLocation, stop.mLocation,
+ nsStyleCoord(), SETCOORD_LPO | SETCOORD_STORE_CALC,
+ aContext, aPresContext, aConditions)) {
+ NS_NOTREACHED("unexpected unit for gradient stop location");
+ }
+
+ stop.mIsInterpolationHint = valueStop.mIsInterpolationHint;
+
+ // inherit is not a valid color for stops, so we pass in a dummy
+ // parent color
+ NS_ASSERTION(valueStop.mColor.GetUnit() != eCSSUnit_Inherit,
+ "inherit is not a valid color for gradient stops");
+ if (!valueStop.mIsInterpolationHint) {
+ SetColor(valueStop.mColor, NS_RGB(0, 0, 0), aPresContext,
+ aContext, stop.mColor, aConditions);
+ } else {
+ // Always initialize to the same color so we don't need to worry
+ // about comparisons.
+ stop.mColor = NS_RGB(0, 0, 0);
+ }
+
+ aResult.mStops.AppendElement(stop);
+ }
+}
+
+// -moz-image-rect(<uri>, <top>, <right>, <bottom>, <left>)
+static void SetStyleImageToImageRect(nsStyleContext* aStyleContext,
+ const nsCSSValue& aValue,
+ nsStyleImage& aResult)
+{
+ MOZ_ASSERT(aValue.GetUnit() == eCSSUnit_Function &&
+ aValue.EqualsFunction(eCSSKeyword__moz_image_rect),
+ "the value is not valid -moz-image-rect()");
+
+ nsCSSValue::Array* arr = aValue.GetArrayValue();
+ MOZ_ASSERT(arr && arr->Count() == 6, "invalid number of arguments");
+
+ // <uri>
+ if (arr->Item(1).GetUnit() == eCSSUnit_Image) {
+ SetStyleImageRequest([&](nsStyleImageRequest* req) {
+ aResult.SetImageRequest(do_AddRef(req));
+ }, aStyleContext->PresContext(), arr->Item(1));
+ } else {
+ NS_WARNING("nsCSSValue::Image::Image() failed?");
+ }
+
+ // <top>, <right>, <bottom>, <left>
+ nsStyleSides cropRect;
+ NS_FOR_CSS_SIDES(side) {
+ nsStyleCoord coord;
+ const nsCSSValue& val = arr->Item(2 + side);
+
+#ifdef DEBUG
+ bool unitOk =
+#endif
+ SetAbsCoord(val, coord, SETCOORD_FACTOR | SETCOORD_PERCENT);
+ MOZ_ASSERT(unitOk, "Incorrect data structure created by CSS parser");
+ cropRect.Set(side, coord);
+ }
+ aResult.SetCropRect(MakeUnique<nsStyleSides>(cropRect));
+}
+
+static void SetStyleImage(nsStyleContext* aStyleContext,
+ const nsCSSValue& aValue,
+ nsStyleImage& aResult,
+ RuleNodeCacheConditions& aConditions)
+{
+ if (aValue.GetUnit() == eCSSUnit_Null) {
+ return;
+ }
+
+ aResult.SetNull();
+
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Image:
+ SetStyleImageRequest([&](nsStyleImageRequest* req) {
+ aResult.SetImageRequest(do_AddRef(req));
+ }, aStyleContext->PresContext(), aValue);
+ break;
+ case eCSSUnit_Function:
+ if (aValue.EqualsFunction(eCSSKeyword__moz_image_rect)) {
+ SetStyleImageToImageRect(aStyleContext, aValue, aResult);
+ } else {
+ NS_NOTREACHED("-moz-image-rect() is the only expected function");
+ }
+ break;
+ case eCSSUnit_Gradient:
+ {
+ nsStyleGradient* gradient = new nsStyleGradient();
+ SetGradient(aValue, aStyleContext->PresContext(), aStyleContext,
+ *gradient, aConditions);
+ aResult.SetGradientData(gradient);
+ break;
+ }
+ case eCSSUnit_Element:
+ aResult.SetElementId(aValue.GetStringBufferValue());
+ break;
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_None:
+ break;
+ case eCSSUnit_URL:
+ {
+#ifdef DEBUG
+ // eCSSUnit_URL is expected only if
+ // 1. we have eCSSUnit_URL values for if-visited style contexts, which
+ // we can safely treat like 'none'.
+ // 2. aValue is a local-ref URL, e.g. url(#foo).
+ // 3. aValue is a not a local-ref URL, but it refers to an element in
+ // the current document. For example, the url of the current document
+ // is "http://foo.html" and aValue is url(http://foo.html#foo).
+ //
+ // We skip image download in TryToStartImageLoadOnValue under #2 and #3,
+ // and that's part of reasons we get eCSSUnit_URL instead of
+ // eCSSUnit_Image here.
+
+ // Check #2.
+ bool isLocalRef = aValue.GetURLStructValue()->IsLocalRef();
+
+ // Check #3.
+ bool isEqualExceptRef = false;
+ if (!isLocalRef) {
+ nsIDocument* currentDoc = aStyleContext->PresContext()->Document();
+ nsIURI* docURI = currentDoc->GetDocumentURI();
+ nsIURI* imageURI = aValue.GetURLValue();
+ imageURI->EqualsExceptRef(docURI, &isEqualExceptRef);
+ }
+
+ MOZ_ASSERT(aStyleContext->IsStyleIfVisited() || isEqualExceptRef ||
+ isLocalRef,
+ "unexpected unit; maybe nsCSSValue::Image::Image() failed?");
+#endif
+
+ break;
+ }
+ default:
+ MOZ_ASSERT_UNREACHABLE("Unexpected Unit type.");
+ break;
+ }
+}
+
+struct SetEnumValueHelper
+{
+ template<typename FieldT>
+ static void SetIntegerValue(FieldT&, const nsCSSValue&)
+ {
+ // FIXME Is it possible to turn this assertion into a compilation error?
+ MOZ_ASSERT_UNREACHABLE("inappropriate unit");
+ }
+
+#define DEFINE_ENUM_CLASS_SETTER(type_, min_, max_) \
+ static void SetEnumeratedValue(type_& aField, const nsCSSValue& aValue) \
+ { \
+ auto value = aValue.GetIntValue(); \
+ MOZ_ASSERT(value >= static_cast<decltype(value)>(type_::min_) && \
+ value <= static_cast<decltype(value)>(type_::max_), \
+ "inappropriate value"); \
+ aField = static_cast<type_>(value); \
+ }
+
+ DEFINE_ENUM_CLASS_SETTER(StyleBoxAlign, Stretch, End)
+ DEFINE_ENUM_CLASS_SETTER(StyleBoxDecorationBreak, Slice, Clone)
+ DEFINE_ENUM_CLASS_SETTER(StyleBoxDirection, Normal, Reverse)
+ DEFINE_ENUM_CLASS_SETTER(StyleBoxOrient, Horizontal, Vertical)
+ DEFINE_ENUM_CLASS_SETTER(StyleBoxPack, Start, Justify)
+ DEFINE_ENUM_CLASS_SETTER(StyleBoxSizing, Content, Border)
+ DEFINE_ENUM_CLASS_SETTER(StyleClear, None, Both)
+ DEFINE_ENUM_CLASS_SETTER(StyleFillRule, Nonzero, Evenodd)
+ DEFINE_ENUM_CLASS_SETTER(StyleFloat, None, InlineEnd)
+ DEFINE_ENUM_CLASS_SETTER(StyleFloatEdge, ContentBox, MarginBox)
+ DEFINE_ENUM_CLASS_SETTER(StyleUserFocus, None, SelectMenu)
+ DEFINE_ENUM_CLASS_SETTER(StyleUserSelect, None, MozText)
+ DEFINE_ENUM_CLASS_SETTER(StyleUserInput, None, Auto)
+ DEFINE_ENUM_CLASS_SETTER(StyleUserModify, ReadOnly, WriteOnly)
+ DEFINE_ENUM_CLASS_SETTER(StyleWindowDragging, Default, NoDrag)
+ DEFINE_ENUM_CLASS_SETTER(StyleOrient, Inline, Vertical)
+#ifdef MOZ_XUL
+ DEFINE_ENUM_CLASS_SETTER(StyleDisplay, None, Popup)
+#else
+ DEFINE_ENUM_CLASS_SETTER(StyleDisplay, None, InlineBox)
+#endif
+
+#undef DEF_SET_ENUMERATED_VALUE
+};
+
+template<typename FieldT>
+struct SetIntegerValueHelper
+{
+ static void SetIntegerValue(FieldT& aField, const nsCSSValue& aValue)
+ {
+ aField = aValue.GetIntValue();
+ }
+ static void SetEnumeratedValue(FieldT& aField, const nsCSSValue& aValue)
+ {
+ aField = aValue.GetIntValue();
+ }
+};
+
+template<typename FieldT>
+struct SetValueHelper : Conditional<IsEnum<FieldT>::value,
+ SetEnumValueHelper,
+ SetIntegerValueHelper<FieldT>>::Type
+{
+ template<typename ValueT>
+ static void SetValue(FieldT& aField, const ValueT& aValue)
+ {
+ aField = aValue;
+ }
+ static void SetValue(FieldT&, unused_t)
+ {
+ // FIXME Is it possible to turn this assertion into a compilation error?
+ MOZ_ASSERT_UNREACHABLE("inappropriate unit");
+ }
+};
+
+
+// flags for SetValue - align values with SETCOORD_* constants
+// where possible
+
+#define SETVAL_INTEGER 0x40 // I
+#define SETVAL_ENUMERATED 0x80 // E
+#define SETVAL_UNSET_INHERIT 0x00400000
+#define SETVAL_UNSET_INITIAL 0x00800000
+
+// no caller cares whether aField was changed or not
+template<typename FieldT, typename InitialT,
+ typename AutoT, typename NoneT, typename NormalT, typename SysFontT>
+static void
+SetValue(const nsCSSValue& aValue, FieldT& aField,
+ RuleNodeCacheConditions& aConditions, uint32_t aMask,
+ FieldT aParentValue,
+ InitialT aInitialValue,
+ AutoT aAutoValue,
+ NoneT aNoneValue,
+ NormalT aNormalValue,
+ SysFontT aSystemFontValue)
+{
+ typedef SetValueHelper<FieldT> Helper;
+
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Null:
+ return;
+
+ // every caller of SetValue provides inherit and initial
+ // alternatives, so we don't require them to say so in the mask
+ case eCSSUnit_Inherit:
+ aConditions.SetUncacheable();
+ aField = aParentValue;
+ return;
+
+ case eCSSUnit_Initial:
+ Helper::SetValue(aField, aInitialValue);
+ return;
+
+ // every caller provides one or other of these alternatives,
+ // but they have to say which
+ case eCSSUnit_Enumerated:
+ if (aMask & SETVAL_ENUMERATED) {
+ Helper::SetEnumeratedValue(aField, aValue);
+ return;
+ }
+ break;
+
+ case eCSSUnit_Integer:
+ if (aMask & SETVAL_INTEGER) {
+ Helper::SetIntegerValue(aField, aValue);
+ return;
+ }
+ break;
+
+ // remaining possibilities in descending order of frequency of use
+ case eCSSUnit_Auto:
+ Helper::SetValue(aField, aAutoValue);
+ return;
+
+ case eCSSUnit_None:
+ Helper::SetValue(aField, aNoneValue);
+ return;
+
+ case eCSSUnit_Normal:
+ Helper::SetValue(aField, aNormalValue);
+ return;
+
+ case eCSSUnit_System_Font:
+ Helper::SetValue(aField, aSystemFontValue);
+ return;
+
+ case eCSSUnit_Unset:
+ if (aMask & SETVAL_UNSET_INHERIT) {
+ aConditions.SetUncacheable();
+ aField = aParentValue;
+ return;
+ }
+ if (aMask & SETVAL_UNSET_INITIAL) {
+ Helper::SetValue(aField, aInitialValue);
+ return;
+ }
+ break;
+
+ default:
+ break;
+ }
+
+ NS_NOTREACHED("SetValue: inappropriate unit");
+}
+
+template <typename FieldT, typename T1>
+static void
+SetValue(const nsCSSValue& aValue, FieldT& aField,
+ RuleNodeCacheConditions& aConditions, uint32_t aMask,
+ FieldT aParentValue, T1 aInitialValue)
+{
+ SetValue(aValue, aField, aConditions, aMask, aParentValue,
+ aInitialValue, Unused, Unused, Unused, Unused);
+}
+
+// flags for SetFactor
+#define SETFCT_POSITIVE 0x01 // assert value is >= 0.0f
+#define SETFCT_OPACITY 0x02 // clamp value to [0.0f .. 1.0f]
+#define SETFCT_NONE 0x04 // allow _None (uses aInitialValue).
+#define SETFCT_UNSET_INHERIT 0x00400000
+#define SETFCT_UNSET_INITIAL 0x00800000
+
+static void
+SetFactor(const nsCSSValue& aValue, float& aField, RuleNodeCacheConditions& aConditions,
+ float aParentValue, float aInitialValue, uint32_t aFlags = 0)
+{
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Null:
+ return;
+
+ case eCSSUnit_Number:
+ aField = aValue.GetFloatValue();
+ if (aFlags & SETFCT_POSITIVE) {
+ NS_ASSERTION(aField >= 0.0f, "negative value for positive-only property");
+ if (aField < 0.0f)
+ aField = 0.0f;
+ }
+ if (aFlags & SETFCT_OPACITY) {
+ if (aField < 0.0f)
+ aField = 0.0f;
+ if (aField > 1.0f)
+ aField = 1.0f;
+ }
+ return;
+
+ case eCSSUnit_Inherit:
+ aConditions.SetUncacheable();
+ aField = aParentValue;
+ return;
+
+ case eCSSUnit_Initial:
+ aField = aInitialValue;
+ return;
+
+ case eCSSUnit_None:
+ if (aFlags & SETFCT_NONE) {
+ aField = aInitialValue;
+ return;
+ }
+ break;
+
+ case eCSSUnit_Unset:
+ if (aFlags & SETFCT_UNSET_INHERIT) {
+ aConditions.SetUncacheable();
+ aField = aParentValue;
+ return;
+ }
+ if (aFlags & SETFCT_UNSET_INITIAL) {
+ aField = aInitialValue;
+ return;
+ }
+ break;
+
+ default:
+ break;
+ }
+
+ NS_NOTREACHED("SetFactor: inappropriate unit");
+}
+
+void*
+nsRuleNode::operator new(size_t sz, nsPresContext* aPresContext)
+{
+ // Check the recycle list first.
+ return aPresContext->PresShell()->AllocateByObjectID(eArenaObjectID_nsRuleNode, sz);
+}
+
+// Overridden to prevent the global delete from being called, since the memory
+// came out of an nsIArena instead of the global delete operator's heap.
+void
+nsRuleNode::Destroy()
+{
+ // Destroy ourselves.
+ this->~nsRuleNode();
+
+ // Don't let the memory be freed, since it will be recycled
+ // instead. Don't call the global operator delete.
+ mPresContext->PresShell()->FreeByObjectID(eArenaObjectID_nsRuleNode, this);
+}
+
+already_AddRefed<nsRuleNode>
+nsRuleNode::CreateRootNode(nsPresContext* aPresContext)
+{
+ return do_AddRef(new (aPresContext)
+ nsRuleNode(aPresContext, nullptr, nullptr, SheetType::Unknown, false));
+}
+
+nsRuleNode::nsRuleNode(nsPresContext* aContext, nsRuleNode* aParent,
+ nsIStyleRule* aRule, SheetType aLevel,
+ bool aIsImportant)
+ : mPresContext(aContext),
+ mParent(aParent),
+ mRule(aRule),
+ mNextSibling(nullptr),
+ mDependentBits((uint32_t(aLevel) << NS_RULE_NODE_LEVEL_SHIFT) |
+ (aIsImportant ? NS_RULE_NODE_IS_IMPORTANT : 0)),
+ mNoneBits(aParent ? aParent->mNoneBits & NS_RULE_NODE_HAS_ANIMATION_DATA :
+ 0),
+ mRefCnt(0)
+{
+ MOZ_ASSERT(aContext);
+ MOZ_ASSERT(IsRoot() == !aRule,
+ "non-root rule nodes must have a rule");
+
+ mChildren.asVoid = nullptr;
+ MOZ_COUNT_CTOR(nsRuleNode);
+
+ NS_ASSERTION(IsRoot() || GetLevel() == aLevel, "not enough bits");
+ NS_ASSERTION(IsRoot() || IsImportantRule() == aIsImportant, "yikes");
+ MOZ_ASSERT(aContext->StyleSet()->IsGecko(),
+ "ServoStyleSets should not have rule nodes");
+ aContext->StyleSet()->AsGecko()->RuleNodeUnused(this, /* aMayGC = */ false);
+
+ // nsStyleSet::GetContext depends on there being only one animation
+ // rule.
+ MOZ_ASSERT(IsRoot() || GetLevel() != SheetType::Animation ||
+ mParent->IsRoot() ||
+ mParent->GetLevel() != SheetType::Animation,
+ "must be only one rule at animation level");
+}
+
+nsRuleNode::~nsRuleNode()
+{
+ MOZ_ASSERT(!HaveChildren());
+ MOZ_COUNT_DTOR(nsRuleNode);
+ if (mParent) {
+ mParent->RemoveChild(this);
+ }
+
+ if (mStyleData.mResetData || mStyleData.mInheritedData)
+ mStyleData.Destroy(mDependentBits, mPresContext);
+}
+
+nsRuleNode*
+nsRuleNode::Transition(nsIStyleRule* aRule, SheetType aLevel,
+ bool aIsImportantRule)
+{
+#ifdef DEBUG
+ {
+ RefPtr<css::Declaration> declaration(do_QueryObject(aRule));
+ MOZ_ASSERT(!declaration || !declaration->IsMutable(),
+ "caller must call Declaration::SetImmutable first");
+ }
+#endif
+
+ nsRuleNode* next = nullptr;
+ nsRuleNode::Key key(aRule, aLevel, aIsImportantRule);
+
+ if (HaveChildren() && !ChildrenAreHashed()) {
+ int32_t numKids = 0;
+ nsRuleNode* curr = ChildrenList();
+ while (curr && curr->GetKey() != key) {
+ curr = curr->mNextSibling;
+ ++numKids;
+ }
+ if (curr)
+ next = curr;
+ else if (numKids >= kMaxChildrenInList)
+ ConvertChildrenToHash(numKids);
+ }
+
+ if (ChildrenAreHashed()) {
+ auto entry =
+ static_cast<ChildrenHashEntry*>(ChildrenHash()->Add(&key, fallible));
+ if (!entry) {
+ NS_WARNING("out of memory");
+ return this;
+ }
+ if (entry->mRuleNode)
+ next = entry->mRuleNode;
+ else {
+ next = entry->mRuleNode = new (mPresContext)
+ nsRuleNode(mPresContext, this, aRule, aLevel, aIsImportantRule);
+ }
+ } else if (!next) {
+ // Create the new entry in our list.
+ next = new (mPresContext)
+ nsRuleNode(mPresContext, this, aRule, aLevel, aIsImportantRule);
+ next->mNextSibling = ChildrenList();
+ SetChildrenList(next);
+ }
+
+ return next;
+}
+
+nsRuleNode*
+nsRuleNode::RuleTree()
+{
+ nsRuleNode* n = this;
+ while (n->mParent) {
+ n = n->mParent;
+ }
+ return n;
+}
+
+void nsRuleNode::SetUsedDirectly()
+{
+ mDependentBits |= NS_RULE_NODE_USED_DIRECTLY;
+
+ // Maintain the invariant that any rule node that is used directly has
+ // all structs that live in the rule tree cached (which
+ // nsRuleNode::GetStyleData depends on for speed).
+ if (mDependentBits & NS_STYLE_INHERIT_MASK) {
+ for (nsStyleStructID sid = nsStyleStructID(0); sid < nsStyleStructID_Length;
+ sid = nsStyleStructID(sid + 1)) {
+ uint32_t bit = nsCachedStyleData::GetBitForSID(sid);
+ if (mDependentBits & bit) {
+ nsRuleNode *source = mParent;
+ while ((source->mDependentBits & bit) && !source->IsUsedDirectly()) {
+ source = source->mParent;
+ }
+ void *data = source->mStyleData.GetStyleData(sid);
+ NS_ASSERTION(data, "unexpected null struct");
+ mStyleData.SetStyleData(sid, mPresContext, data);
+ }
+ }
+ }
+}
+
+void
+nsRuleNode::ConvertChildrenToHash(int32_t aNumKids)
+{
+ NS_ASSERTION(!ChildrenAreHashed() && HaveChildren(),
+ "must have a non-empty list of children");
+ PLDHashTable *hash = new PLDHashTable(&ChildrenHashOps,
+ sizeof(ChildrenHashEntry),
+ aNumKids);
+ for (nsRuleNode* curr = ChildrenList(); curr; curr = curr->mNextSibling) {
+ Key key = curr->GetKey();
+ // This will never fail because of the initial size we gave the table.
+ auto entry =
+ static_cast<ChildrenHashEntry*>(hash->Add(&key));
+ NS_ASSERTION(!entry->mRuleNode, "duplicate entries in list");
+ entry->mRuleNode = curr;
+ }
+ SetChildrenHash(hash);
+}
+
+void
+nsRuleNode::RemoveChild(nsRuleNode* aNode)
+{
+ MOZ_ASSERT(HaveChildren());
+ if (ChildrenAreHashed()) {
+ PLDHashTable* children = ChildrenHash();
+ Key key = aNode->GetKey();
+ MOZ_ASSERT(children->Search(&key));
+ children->Remove(&key);
+ if (children->EntryCount() == 0) {
+ delete children;
+ mChildren.asVoid = nullptr;
+ }
+ } else {
+ // This linear traversal is unfortunate, but we do the same thing when
+ // adding nodes. The traversal is bounded by kMaxChildrenInList.
+ nsRuleNode** curr = &mChildren.asList;
+ while (*curr != aNode) {
+ curr = &((*curr)->mNextSibling);
+ MOZ_ASSERT(*curr);
+ }
+ *curr = (*curr)->mNextSibling;
+
+ // If there was one element in the list, this sets mChildren.asList
+ // to 0, and HaveChildren() will return false.
+ }
+}
+
+inline void
+nsRuleNode::PropagateNoneBit(uint32_t aBit, nsRuleNode* aHighestNode)
+{
+ nsRuleNode* curr = this;
+ for (;;) {
+ NS_ASSERTION(!(curr->mNoneBits & aBit), "propagating too far");
+ curr->mNoneBits |= aBit;
+ if (curr == aHighestNode)
+ break;
+ curr = curr->mParent;
+ }
+}
+
+inline void
+nsRuleNode::PropagateDependentBit(nsStyleStructID aSID, nsRuleNode* aHighestNode,
+ void* aStruct)
+{
+ NS_ASSERTION(aStruct, "expected struct");
+
+ uint32_t bit = nsCachedStyleData::GetBitForSID(aSID);
+ for (nsRuleNode* curr = this; curr != aHighestNode; curr = curr->mParent) {
+ if (curr->mDependentBits & bit) {
+#ifdef DEBUG
+ while (curr != aHighestNode) {
+ NS_ASSERTION(curr->mDependentBits & bit, "bit not set");
+ curr = curr->mParent;
+ }
+#endif
+ break;
+ }
+
+ curr->mDependentBits |= bit;
+
+ if (curr->IsUsedDirectly()) {
+ curr->mStyleData.SetStyleData(aSID, mPresContext, aStruct);
+ }
+ }
+}
+
+/* static */ void
+nsRuleNode::PropagateGrandancestorBit(nsStyleContext* aContext,
+ nsStyleContext* aContextInheritedFrom)
+{
+ MOZ_ASSERT(aContext);
+ MOZ_ASSERT(aContextInheritedFrom &&
+ aContextInheritedFrom != aContext,
+ "aContextInheritedFrom must be an ancestor of aContext");
+
+ for (nsStyleContext* context = aContext->GetParent();
+ context != aContextInheritedFrom;
+ context = context->GetParent()) {
+ if (!context) {
+ MOZ_ASSERT(false, "aContextInheritedFrom must be an ancestor of "
+ "aContext's parent");
+ break;
+ }
+ context->AddStyleBit(NS_STYLE_CHILD_USES_GRANDANCESTOR_STYLE);
+ }
+}
+
+/*
+ * The following "Check" functions are used for determining what type of
+ * sharing can be used for the data on this rule node. MORE HERE...
+ */
+
+/*
+ * a callback function that that can revise the result of
+ * CheckSpecifiedProperties before finishing; aResult is the current
+ * result, and it returns the revised one.
+ */
+typedef nsRuleNode::RuleDetail
+ (* CheckCallbackFn)(const nsRuleData* aRuleData,
+ nsRuleNode::RuleDetail aResult);
+
+/**
+ * @param aValue the value being examined
+ * @param aSpecifiedCount to be incremented by one if the value is specified
+ * @param aInheritedCount to be incremented by one if the value is set to inherit
+ * @param aUnsetCount to be incremented by one if the value is set to unset
+ */
+inline void
+ExamineCSSValue(const nsCSSValue& aValue,
+ uint32_t& aSpecifiedCount,
+ uint32_t& aInheritedCount,
+ uint32_t& aUnsetCount)
+{
+ if (aValue.GetUnit() != eCSSUnit_Null) {
+ ++aSpecifiedCount;
+ if (aValue.GetUnit() == eCSSUnit_Inherit) {
+ ++aInheritedCount;
+ } else if (aValue.GetUnit() == eCSSUnit_Unset) {
+ ++aUnsetCount;
+ }
+ }
+}
+
+static nsRuleNode::RuleDetail
+CheckFontCallback(const nsRuleData* aRuleData,
+ nsRuleNode::RuleDetail aResult)
+{
+ // em, ex, percent, 'larger', and 'smaller' values on font-size depend
+ // on the parent context's font-size
+ // Likewise, 'lighter' and 'bolder' values of 'font-weight', and 'wider'
+ // and 'narrower' values of 'font-stretch' depend on the parent.
+ const nsCSSValue& size = *aRuleData->ValueForFontSize();
+ const nsCSSValue& weight = *aRuleData->ValueForFontWeight();
+ if ((size.IsRelativeLengthUnit() && size.GetUnit() != eCSSUnit_RootEM) ||
+ size.GetUnit() == eCSSUnit_Percent ||
+ (size.GetUnit() == eCSSUnit_Enumerated &&
+ (size.GetIntValue() == NS_STYLE_FONT_SIZE_SMALLER ||
+ size.GetIntValue() == NS_STYLE_FONT_SIZE_LARGER)) ||
+ aRuleData->ValueForScriptLevel()->GetUnit() == eCSSUnit_Integer ||
+ (weight.GetUnit() == eCSSUnit_Enumerated &&
+ (weight.GetIntValue() == NS_STYLE_FONT_WEIGHT_BOLDER ||
+ weight.GetIntValue() == NS_STYLE_FONT_WEIGHT_LIGHTER))) {
+ NS_ASSERTION(aResult == nsRuleNode::eRulePartialReset ||
+ aResult == nsRuleNode::eRuleFullReset ||
+ aResult == nsRuleNode::eRulePartialMixed ||
+ aResult == nsRuleNode::eRuleFullMixed,
+ "we know we already have a reset-counted property");
+ // Promote reset to mixed since we have something that depends on
+ // the parent. But never promote to inherited since that could
+ // cause inheritance of the exact value.
+ if (aResult == nsRuleNode::eRulePartialReset)
+ aResult = nsRuleNode::eRulePartialMixed;
+ else if (aResult == nsRuleNode::eRuleFullReset)
+ aResult = nsRuleNode::eRuleFullMixed;
+ }
+
+ return aResult;
+}
+
+static nsRuleNode::RuleDetail
+CheckColorCallback(const nsRuleData* aRuleData,
+ nsRuleNode::RuleDetail aResult)
+{
+ // currentColor values for color require inheritance
+ const nsCSSValue* colorValue = aRuleData->ValueForColor();
+ if (colorValue->GetUnit() == eCSSUnit_EnumColor &&
+ colorValue->GetIntValue() == NS_COLOR_CURRENTCOLOR) {
+ NS_ASSERTION(aResult == nsRuleNode::eRuleFullReset,
+ "we should already be counted as full-reset");
+ aResult = nsRuleNode::eRuleFullInherited;
+ }
+
+ return aResult;
+}
+
+static nsRuleNode::RuleDetail
+CheckTextCallback(const nsRuleData* aRuleData,
+ nsRuleNode::RuleDetail aResult)
+{
+ const nsCSSValue* textAlignValue = aRuleData->ValueForTextAlign();
+ if (textAlignValue->GetUnit() == eCSSUnit_Enumerated &&
+ (textAlignValue->GetIntValue() ==
+ NS_STYLE_TEXT_ALIGN_MOZ_CENTER_OR_INHERIT ||
+ textAlignValue->GetIntValue() == NS_STYLE_TEXT_ALIGN_MATCH_PARENT)) {
+ // Promote reset to mixed since we have something that depends on
+ // the parent.
+ if (aResult == nsRuleNode::eRulePartialReset)
+ aResult = nsRuleNode::eRulePartialMixed;
+ else if (aResult == nsRuleNode::eRuleFullReset)
+ aResult = nsRuleNode::eRuleFullMixed;
+ }
+
+ return aResult;
+}
+
+static nsRuleNode::RuleDetail
+CheckVariablesCallback(const nsRuleData* aRuleData,
+ nsRuleNode::RuleDetail aResult)
+{
+ // We don't actually have any properties on nsStyleVariables, so we do
+ // all of the RuleDetail calculation in here.
+ if (aRuleData->mVariables) {
+ return nsRuleNode::eRulePartialMixed;
+ }
+ return nsRuleNode::eRuleNone;
+}
+
+#define FLAG_DATA_FOR_PROPERTY(name_, id_, method_, flags_, pref_, \
+ parsevariant_, kwtable_, stylestructoffset_, \
+ animtype_) \
+ flags_,
+
+// The order here must match the enums in *CheckCounter in nsCSSProps.cpp.
+
+static const uint32_t gFontFlags[] = {
+#define CSS_PROP_FONT FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_FONT
+};
+
+static const uint32_t gDisplayFlags[] = {
+#define CSS_PROP_DISPLAY FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_DISPLAY
+};
+
+static const uint32_t gVisibilityFlags[] = {
+#define CSS_PROP_VISIBILITY FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_VISIBILITY
+};
+
+static const uint32_t gMarginFlags[] = {
+#define CSS_PROP_MARGIN FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_MARGIN
+};
+
+static const uint32_t gBorderFlags[] = {
+#define CSS_PROP_BORDER FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_BORDER
+};
+
+static const uint32_t gPaddingFlags[] = {
+#define CSS_PROP_PADDING FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_PADDING
+};
+
+static const uint32_t gOutlineFlags[] = {
+#define CSS_PROP_OUTLINE FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_OUTLINE
+};
+
+static const uint32_t gListFlags[] = {
+#define CSS_PROP_LIST FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_LIST
+};
+
+static const uint32_t gColorFlags[] = {
+#define CSS_PROP_COLOR FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_COLOR
+};
+
+static const uint32_t gBackgroundFlags[] = {
+#define CSS_PROP_BACKGROUND FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_BACKGROUND
+};
+
+static const uint32_t gPositionFlags[] = {
+#define CSS_PROP_POSITION FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_POSITION
+};
+
+static const uint32_t gTableFlags[] = {
+#define CSS_PROP_TABLE FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_TABLE
+};
+
+static const uint32_t gTableBorderFlags[] = {
+#define CSS_PROP_TABLEBORDER FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_TABLEBORDER
+};
+
+static const uint32_t gContentFlags[] = {
+#define CSS_PROP_CONTENT FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_CONTENT
+};
+
+static const uint32_t gTextFlags[] = {
+#define CSS_PROP_TEXT FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_TEXT
+};
+
+static const uint32_t gTextResetFlags[] = {
+#define CSS_PROP_TEXTRESET FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_TEXTRESET
+};
+
+static const uint32_t gUserInterfaceFlags[] = {
+#define CSS_PROP_USERINTERFACE FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_USERINTERFACE
+};
+
+static const uint32_t gUIResetFlags[] = {
+#define CSS_PROP_UIRESET FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_UIRESET
+};
+
+static const uint32_t gXULFlags[] = {
+#define CSS_PROP_XUL FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_XUL
+};
+
+static const uint32_t gSVGFlags[] = {
+#define CSS_PROP_SVG FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_SVG
+};
+
+static const uint32_t gSVGResetFlags[] = {
+#define CSS_PROP_SVGRESET FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_SVGRESET
+};
+
+static const uint32_t gColumnFlags[] = {
+#define CSS_PROP_COLUMN FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_COLUMN
+};
+
+// There are no properties in nsStyleVariables, but we can't have a
+// zero length array.
+static const uint32_t gVariablesFlags[] = {
+ 0,
+#define CSS_PROP_VARIABLES FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_VARIABLES
+};
+static_assert(sizeof(gVariablesFlags) == sizeof(uint32_t),
+ "if nsStyleVariables has properties now you can remove the dummy "
+ "gVariablesFlags entry");
+
+static const uint32_t gEffectsFlags[] = {
+#define CSS_PROP_EFFECTS FLAG_DATA_FOR_PROPERTY
+#include "nsCSSPropList.h"
+#undef CSS_PROP_EFFECTS
+};
+
+#undef FLAG_DATA_FOR_PROPERTY
+
+static const uint32_t* gFlagsByStruct[] = {
+
+#define STYLE_STRUCT(name, checkdata_cb) \
+ g##name##Flags,
+#include "nsStyleStructList.h"
+#undef STYLE_STRUCT
+
+};
+
+static const CheckCallbackFn gCheckCallbacks[] = {
+
+#define STYLE_STRUCT(name, checkdata_cb) \
+ checkdata_cb,
+#include "nsStyleStructList.h"
+#undef STYLE_STRUCT
+
+};
+
+#ifdef DEBUG
+static bool
+AreAllMathMLPropertiesUndefined(const nsRuleData* aRuleData)
+{
+ return
+ aRuleData->ValueForScriptLevel()->GetUnit() == eCSSUnit_Null &&
+ aRuleData->ValueForScriptSizeMultiplier()->GetUnit() == eCSSUnit_Null &&
+ aRuleData->ValueForScriptMinSize()->GetUnit() == eCSSUnit_Null &&
+ aRuleData->ValueForMathVariant()->GetUnit() == eCSSUnit_Null &&
+ aRuleData->ValueForMathDisplay()->GetUnit() == eCSSUnit_Null;
+}
+#endif
+
+inline nsRuleNode::RuleDetail
+nsRuleNode::CheckSpecifiedProperties(const nsStyleStructID aSID,
+ const nsRuleData* aRuleData)
+{
+ // Build a count of the:
+ uint32_t total = 0, // total number of props in the struct
+ specified = 0, // number that were specified for this node
+ inherited = 0, // number that were 'inherit' (and not
+ // eCSSUnit_Inherit) for this node
+ unset = 0; // number that were 'unset'
+
+ // See comment in nsRuleData.h above mValueOffsets.
+ MOZ_ASSERT(aRuleData->mValueOffsets[aSID] == 0,
+ "we assume the value offset is zero instead of adding it");
+ for (nsCSSValue *values = aRuleData->mValueStorage,
+ *values_end = values + nsCSSProps::PropertyCountInStruct(aSID);
+ values != values_end; ++values) {
+ ++total;
+ ExamineCSSValue(*values, specified, inherited, unset);
+ }
+
+ if (!nsCachedStyleData::IsReset(aSID)) {
+ // For inherited properties, 'unset' means the same as 'inherit'.
+ inherited += unset;
+ unset = 0;
+ }
+
+#if 0
+ printf("CheckSpecifiedProperties: SID=%d total=%d spec=%d inh=%d.\n",
+ aSID, total, specified, inherited);
+#endif
+
+ NS_ASSERTION(aSID != eStyleStruct_Font ||
+ mPresContext->Document()->GetMathMLEnabled() ||
+ AreAllMathMLPropertiesUndefined(aRuleData),
+ "MathML style property was defined even though MathML is disabled");
+
+ /*
+ * Return the most specific information we can: prefer None or Full
+ * over Partial, and Reset or Inherited over Mixed, since we can
+ * optimize based on the edge cases and not the in-between cases.
+ */
+ nsRuleNode::RuleDetail result;
+ if (inherited == total)
+ result = eRuleFullInherited;
+ else if (specified == total
+ // MathML defines 5 properties in Font that will never be set when
+ // MathML is not in use. Therefore if all but five
+ // properties have been set, and MathML is not enabled, we can treat
+ // this as fully specified. Code in nsMathMLElementFactory will
+ // rebuild the rule tree and style data when MathML is first enabled
+ // (see nsMathMLElement::BindToTree).
+ || (aSID == eStyleStruct_Font && specified + 5 == total &&
+ !mPresContext->Document()->GetMathMLEnabled())
+ ) {
+ if (inherited == 0)
+ result = eRuleFullReset;
+ else
+ result = eRuleFullMixed;
+ } else if (specified == 0)
+ result = eRuleNone;
+ else if (specified == inherited)
+ result = eRulePartialInherited;
+ else if (inherited == 0)
+ result = eRulePartialReset;
+ else
+ result = eRulePartialMixed;
+
+ CheckCallbackFn cb = gCheckCallbacks[aSID];
+ if (cb) {
+ result = (*cb)(aRuleData, result);
+ }
+
+ return result;
+}
+
+// If we need to restrict which properties apply to the style context,
+// return the bit to check in nsCSSProp's flags table. Otherwise,
+// return 0.
+inline uint32_t
+GetPseudoRestriction(nsStyleContext *aContext)
+{
+ // This needs to match nsStyleSet::WalkRestrictionRule.
+ uint32_t pseudoRestriction = 0;
+ nsIAtom *pseudoType = aContext->GetPseudo();
+ if (pseudoType) {
+ if (pseudoType == nsCSSPseudoElements::firstLetter) {
+ pseudoRestriction = CSS_PROPERTY_APPLIES_TO_FIRST_LETTER;
+ } else if (pseudoType == nsCSSPseudoElements::firstLine) {
+ pseudoRestriction = CSS_PROPERTY_APPLIES_TO_FIRST_LINE;
+ } else if (pseudoType == nsCSSPseudoElements::placeholder) {
+ pseudoRestriction = CSS_PROPERTY_APPLIES_TO_PLACEHOLDER;
+ }
+ }
+ return pseudoRestriction;
+}
+
+static void
+UnsetPropertiesWithoutFlags(const nsStyleStructID aSID,
+ nsRuleData* aRuleData,
+ uint32_t aFlags)
+{
+ NS_ASSERTION(aFlags != 0, "aFlags must be nonzero");
+
+ const uint32_t *flagData = gFlagsByStruct[aSID];
+
+ // See comment in nsRuleData.h above mValueOffsets.
+ MOZ_ASSERT(aRuleData->mValueOffsets[aSID] == 0,
+ "we assume the value offset is zero instead of adding it");
+ nsCSSValue *values = aRuleData->mValueStorage;
+
+ for (size_t i = 0, i_end = nsCSSProps::PropertyCountInStruct(aSID);
+ i != i_end; ++i) {
+ if ((flagData[i] & aFlags) != aFlags)
+ values[i].Reset();
+ }
+}
+
+/**
+ * We allocate arrays of CSS values with alloca. (These arrays are a
+ * fixed size per style struct, but we don't want to waste the
+ * allocation and construction/destruction costs of the big structs when
+ * we're handling much smaller ones.) Since the lifetime of an alloca
+ * allocation is the life of the calling function, the caller must call
+ * alloca. However, to ensure that constructors and destructors are
+ * balanced, we do the constructor and destructor calling from this RAII
+ * class, AutoCSSValueArray.
+ */
+struct AutoCSSValueArray {
+ /**
+ * aStorage must be the result of alloca(aCount * sizeof(nsCSSValue))
+ */
+ AutoCSSValueArray(void* aStorage, size_t aCount) {
+ MOZ_ASSERT(size_t(aStorage) % NS_ALIGNMENT_OF(nsCSSValue) == 0,
+ "bad alignment from alloca");
+ mCount = aCount;
+ // Don't use placement new[], since it might store extra data
+ // for the count (on Windows!).
+ mArray = static_cast<nsCSSValue*>(aStorage);
+ for (size_t i = 0; i < mCount; ++i) {
+ new (KnownNotNull, mArray + i) nsCSSValue();
+ }
+ }
+
+ ~AutoCSSValueArray() {
+ for (size_t i = 0; i < mCount; ++i) {
+ mArray[i].~nsCSSValue();
+ }
+ }
+
+ nsCSSValue* get() { return mArray; }
+
+private:
+ nsCSSValue *mArray;
+ size_t mCount;
+};
+
+/* static */ bool
+nsRuleNode::ResolveVariableReferences(const nsStyleStructID aSID,
+ nsRuleData* aRuleData,
+ nsStyleContext* aContext)
+{
+ MOZ_ASSERT(aSID != eStyleStruct_Variables);
+ MOZ_ASSERT(aRuleData->mSIDs & nsCachedStyleData::GetBitForSID(aSID));
+ MOZ_ASSERT(aRuleData->mValueOffsets[aSID] == 0);
+
+ nsCSSParser parser;
+ bool anyTokenStreams = false;
+
+ // Look at each property in the nsRuleData for the given style struct.
+ size_t nprops = nsCSSProps::PropertyCountInStruct(aSID);
+ for (nsCSSValue* value = aRuleData->mValueStorage,
+ *values_end = aRuleData->mValueStorage + nprops;
+ value != values_end; value++) {
+ if (value->GetUnit() != eCSSUnit_TokenStream) {
+ continue;
+ }
+
+ const CSSVariableValues* variables =
+ &aContext->StyleVariables()->mVariables;
+ nsCSSValueTokenStream* tokenStream = value->GetTokenStreamValue();
+
+ MOZ_ASSERT(tokenStream->mLevel != SheetType::Count,
+ "Token stream should have a defined level");
+
+ AutoRestore<SheetType> saveLevel(aRuleData->mLevel);
+ aRuleData->mLevel = tokenStream->mLevel;
+
+ // Note that ParsePropertyWithVariableReferences relies on the fact
+ // that the nsCSSValue in aRuleData for the property we are re-parsing
+ // is still the token stream value. When
+ // ParsePropertyWithVariableReferences calls
+ // nsCSSExpandedDataBlock::MapRuleInfoInto, that function will add
+ // the ImageValue that is created into the token stream object's
+ // mImageValues table; see the comment above mImageValues for why.
+
+ // XXX Should pass in sheet here (see bug 952338).
+ parser.ParsePropertyWithVariableReferences(
+ tokenStream->mPropertyID, tokenStream->mShorthandPropertyID,
+ tokenStream->mTokenStream, variables, aRuleData,
+ tokenStream->mSheetURI, tokenStream->mBaseURI,
+ tokenStream->mSheetPrincipal, nullptr,
+ tokenStream->mLineNumber, tokenStream->mLineOffset);
+ aRuleData->mConditions.SetUncacheable();
+ anyTokenStreams = true;
+ }
+
+ return anyTokenStreams;
+}
+
+const void*
+nsRuleNode::WalkRuleTree(const nsStyleStructID aSID,
+ nsStyleContext* aContext)
+{
+ // use placement new[] on the result of alloca() to allocate a
+ // variable-sized stack array, including execution of constructors,
+ // and use an RAII class to run the destructors too.
+ size_t nprops = nsCSSProps::PropertyCountInStruct(aSID);
+ void* dataStorage = alloca(nprops * sizeof(nsCSSValue));
+ AutoCSSValueArray dataArray(dataStorage, nprops);
+
+ nsRuleData ruleData(nsCachedStyleData::GetBitForSID(aSID),
+ dataArray.get(), mPresContext, aContext);
+ ruleData.mValueOffsets[aSID] = 0;
+
+ // We start at the most specific rule in the tree.
+ void* startStruct = nullptr;
+
+ nsRuleNode* ruleNode = this;
+ nsRuleNode* highestNode = nullptr; // The highest node in the rule tree
+ // that has the same properties
+ // specified for struct |aSID| as
+ // |this| does.
+ nsRuleNode* rootNode = this; // After the loop below, this will be the
+ // highest node that we've walked without
+ // finding cached data on the rule tree.
+ // If we don't find any cached data, it
+ // will be the root. (XXX misnamed)
+ RuleDetail detail = eRuleNone;
+ uint32_t bit = nsCachedStyleData::GetBitForSID(aSID);
+
+ while (ruleNode) {
+ // See if this rule node has cached the fact that the remaining
+ // nodes along this path specify no data whatsoever.
+ if (ruleNode->mNoneBits & bit)
+ break;
+
+ // If the dependent bit is set on a rule node for this struct, that
+ // means its rule won't have any information to add, so skip it.
+ // NOTE: If we exit the loop because of the !IsUsedDirectly() check,
+ // then we're guaranteed to break immediately afterwards due to a
+ // non-null startStruct.
+ while ((ruleNode->mDependentBits & bit) && !ruleNode->IsUsedDirectly()) {
+ NS_ASSERTION(ruleNode->mStyleData.GetStyleData(aSID) == nullptr,
+ "dependent bit with cached data makes no sense");
+ // Climb up to the next rule in the tree (a less specific rule).
+ rootNode = ruleNode;
+ ruleNode = ruleNode->mParent;
+ NS_ASSERTION(!(ruleNode->mNoneBits & bit), "can't have both bits set");
+ }
+
+ // Check for cached data after the inner loop above -- otherwise
+ // we'll miss it.
+ startStruct = ruleNode->mStyleData.GetStyleData(aSID);
+ if (startStruct)
+ break; // We found a rule with fully specified data. We don't
+ // need to go up the tree any further, since the remainder
+ // of this branch has already been computed.
+
+ // Ask the rule to fill in the properties that it specifies.
+ nsIStyleRule *rule = ruleNode->mRule;
+ if (rule) {
+ ruleData.mLevel = ruleNode->GetLevel();
+ ruleData.mIsImportantRule = ruleNode->IsImportantRule();
+ rule->MapRuleInfoInto(&ruleData);
+ }
+
+ // Now we check to see how many properties have been specified by
+ // the rules we've examined so far.
+ RuleDetail oldDetail = detail;
+ detail = CheckSpecifiedProperties(aSID, &ruleData);
+
+ if (oldDetail == eRuleNone && detail != eRuleNone)
+ highestNode = ruleNode;
+
+ if (detail == eRuleFullReset ||
+ detail == eRuleFullMixed ||
+ detail == eRuleFullInherited)
+ break; // We don't need to examine any more rules. All properties
+ // have been fully specified.
+
+ // Climb up to the next rule in the tree (a less specific rule).
+ rootNode = ruleNode;
+ ruleNode = ruleNode->mParent;
+ }
+
+ bool recomputeDetail = false;
+
+ // If we are computing a style struct other than nsStyleVariables, and
+ // ruleData has any properties with variable references (nsCSSValues of
+ // type eCSSUnit_TokenStream), then we need to resolve these.
+ if (aSID != eStyleStruct_Variables) {
+ // A property's value might have became 'inherit' after resolving
+ // variable references. (This happens when an inherited property
+ // fails to parse its resolved value.) We need to recompute
+ // |detail| in case this happened.
+ recomputeDetail = ResolveVariableReferences(aSID, &ruleData, aContext);
+ }
+
+ // If needed, unset the properties that don't have a flag that allows
+ // them to be set for this style context. (For example, only some
+ // properties apply to :first-line and :first-letter.)
+ uint32_t pseudoRestriction = GetPseudoRestriction(aContext);
+ if (pseudoRestriction) {
+ UnsetPropertiesWithoutFlags(aSID, &ruleData, pseudoRestriction);
+
+ // We need to recompute |detail| based on the restrictions we just applied.
+ // We can adjust |detail| arbitrarily because of the restriction
+ // rule added in nsStyleSet::WalkRestrictionRule.
+ recomputeDetail = true;
+ }
+
+ if (recomputeDetail) {
+ detail = CheckSpecifiedProperties(aSID, &ruleData);
+ }
+
+ NS_ASSERTION(!startStruct || (detail != eRuleFullReset &&
+ detail != eRuleFullMixed &&
+ detail != eRuleFullInherited),
+ "can't have start struct and be fully specified");
+
+ bool isReset = nsCachedStyleData::IsReset(aSID);
+ if (!highestNode)
+ highestNode = rootNode;
+
+ MOZ_ASSERT(!(aSID == eStyleStruct_Variables && startStruct),
+ "if we start caching Variables structs in the rule tree, then "
+ "not forcing detail to eRulePartialMixed just below is no "
+ "longer valid");
+
+ if (detail == eRuleNone && isReset) {
+ // We specified absolutely no rule information for a reset struct, and we
+ // may or may not have found a parent rule in the tree that specified all
+ // the rule information. Regardless, we don't need to use any cache
+ // conditions if we cache this struct in the rule tree.
+ //
+ // Normally ruleData.mConditions would already indicate that the struct
+ // is cacheable without conditions if detail is eRuleNone, but because
+ // of the UnsetPropertiesWithoutFlags call above, we may have encountered
+ // some rules with dependencies, which we then cleared out of ruleData.
+ //
+ // ruleData.mConditions could also indicate we are not cacheable at all,
+ // such as when AnimValuesStyleRule prevents us from caching structs
+ // when attempting to apply animations to pseudos.
+ //
+ // So if we we are uncacheable, we leave it, but if we are cacheable
+ // with dependencies, we convert that to cacheable without dependencies.
+ if (ruleData.mConditions.CacheableWithDependencies()) {
+ MOZ_ASSERT(pseudoRestriction,
+ "should only be cacheable with dependencies if we had a "
+ "pseudo restriction");
+ ruleData.mConditions.Clear();
+ } else {
+ // XXXheycam We shouldn't have `|| GetLevel() == SheetType::Transition`
+ // in the assertion condition, but rule nodes created by
+ // ResolveStyleByAddingRules don't call SetIsAnimationRule().
+ MOZ_ASSERT(ruleData.mConditions.CacheableWithoutDependencies() ||
+ ((HasAnimationData() ||
+ GetLevel() == SheetType::Transition) &&
+ aContext->GetParent() &&
+ aContext->GetParent()->HasPseudoElementData()),
+ "should only be uncacheable if we had an animation rule "
+ "and we're inside a pseudo");
+ }
+ }
+
+ if (!ruleData.mConditions.CacheableWithoutDependencies() &&
+ aSID != eStyleStruct_Variables) {
+ // Treat as though some data is specified to avoid the optimizations and
+ // force data computation.
+ //
+ // We don't need to do this for Variables structs since we know those are
+ // never cached in the rule tree, and it avoids wasteful computation of a
+ // new Variables struct when we have no additional variable declarations,
+ // which otherwise could happen when there is an AnimValuesStyleRule
+ // (which calls SetUncacheable for style contexts with pseudo data).
+ detail = eRulePartialMixed;
+ }
+
+ if (detail == eRuleNone && startStruct) {
+ // We specified absolutely no rule information, but a parent rule in the tree
+ // specified all the rule information. We set a bit along the branch from our
+ // node in the tree to the node that specified the data that tells nodes on that
+ // branch that they never need to examine their rules for this particular struct type
+ // ever again.
+ PropagateDependentBit(aSID, ruleNode, startStruct);
+ // For inherited structs, mark the struct (which will be set on
+ // the context by our caller) as not being owned by the context.
+ if (!isReset) {
+ aContext->AddStyleBit(nsCachedStyleData::GetBitForSID(aSID));
+ } else if (HasAnimationData()) {
+ // If we have animation data, the struct should be cached on the style
+ // context so that we can peek the struct.
+ // See comment in AnimValuesStyleRule::MapRuleInfoInto.
+ StoreStyleOnContext(aContext, aSID, startStruct);
+ }
+
+ return startStruct;
+ }
+ if ((!startStruct && !isReset &&
+ (detail == eRuleNone || detail == eRulePartialInherited)) ||
+ detail == eRuleFullInherited) {
+ // We specified no non-inherited information and neither did any of
+ // our parent rules.
+
+ // We set a bit along the branch from the highest node (ruleNode)
+ // down to our node (this) indicating that no non-inherited data was
+ // specified. This bit is guaranteed to be set already on the path
+ // from the highest node to the root node in the case where
+ // (detail == eRuleNone), which is the most common case here.
+ // We must check |!isReset| because the Compute*Data functions for
+ // reset structs wouldn't handle none bits correctly.
+ if (highestNode != this && !isReset)
+ PropagateNoneBit(bit, highestNode);
+
+ // All information must necessarily be inherited from our parent style context.
+ // In the absence of any computed data in the rule tree and with
+ // no rules specified that didn't have values of 'inherit', we should check our parent.
+ nsStyleContext* parentContext = aContext->GetParent();
+ if (isReset) {
+ /* Reset structs don't inherit from first-line. */
+ /* See similar code in COMPUTE_START_RESET */
+ while (parentContext &&
+ parentContext->GetPseudo() == nsCSSPseudoElements::firstLine) {
+ parentContext = parentContext->GetParent();
+ }
+ if (parentContext && parentContext != aContext->GetParent()) {
+ PropagateGrandancestorBit(aContext, parentContext);
+ }
+ }
+ if (parentContext) {
+ // We have a parent, and so we should just inherit from the parent.
+ // Set the inherit bits on our context. These bits tell the style context that
+ // it never has to go back to the rule tree for data. Instead the style context tree
+ // should be walked to find the data.
+ const void* parentStruct = parentContext->StyleData(aSID);
+ aContext->AddStyleBit(bit); // makes const_cast OK.
+ aContext->SetStyle(aSID, const_cast<void*>(parentStruct));
+ if (isReset) {
+ parentContext->AddStyleBit(NS_STYLE_HAS_CHILD_THAT_USES_RESET_STYLE);
+ }
+ return parentStruct;
+ }
+ else
+ // We are the root. In the case of fonts, the default values just
+ // come from the pres context.
+ return SetDefaultOnRoot(aSID, aContext);
+ }
+
+ typedef const void* (nsRuleNode::*ComputeFunc)(void*, const nsRuleData*,
+ nsStyleContext*, nsRuleNode*,
+ RuleDetail,
+ const RuleNodeCacheConditions);
+ static const ComputeFunc sComputeFuncs[] = {
+#define STYLE_STRUCT(name, checkdata_cb) &nsRuleNode::Compute##name##Data,
+#include "nsStyleStructList.h"
+#undef STYLE_STRUCT
+ };
+
+ // We need to compute the data from the information that the rules specified.
+ return (this->*sComputeFuncs[aSID])(startStruct, &ruleData, aContext,
+ highestNode, detail,
+ ruleData.mConditions);
+}
+
+const void*
+nsRuleNode::SetDefaultOnRoot(const nsStyleStructID aSID, nsStyleContext* aContext)
+{
+ switch (aSID) {
+ case eStyleStruct_Font:
+ {
+ nsStyleFont* fontData = new (mPresContext) nsStyleFont(mPresContext);
+ nscoord minimumFontSize = mPresContext->MinFontSize(fontData->mLanguage);
+
+ if (minimumFontSize > 0 && !mPresContext->IsChrome()) {
+ fontData->mFont.size = std::max(fontData->mSize, minimumFontSize);
+ }
+ else {
+ fontData->mFont.size = fontData->mSize;
+ }
+ aContext->SetStyle(eStyleStruct_Font, fontData);
+ return fontData;
+ }
+ case eStyleStruct_Display:
+ {
+ nsStyleDisplay* disp = new (mPresContext) nsStyleDisplay(mPresContext);
+ aContext->SetStyle(eStyleStruct_Display, disp);
+ return disp;
+ }
+ case eStyleStruct_Visibility:
+ {
+ nsStyleVisibility* vis = new (mPresContext) nsStyleVisibility(mPresContext);
+ aContext->SetStyle(eStyleStruct_Visibility, vis);
+ return vis;
+ }
+ case eStyleStruct_Text:
+ {
+ nsStyleText* text = new (mPresContext) nsStyleText(mPresContext);
+ aContext->SetStyle(eStyleStruct_Text, text);
+ return text;
+ }
+ case eStyleStruct_TextReset:
+ {
+ nsStyleTextReset* text = new (mPresContext) nsStyleTextReset(mPresContext);
+ aContext->SetStyle(eStyleStruct_TextReset, text);
+ return text;
+ }
+ case eStyleStruct_Color:
+ {
+ nsStyleColor* color = new (mPresContext) nsStyleColor(mPresContext);
+ aContext->SetStyle(eStyleStruct_Color, color);
+ return color;
+ }
+ case eStyleStruct_Background:
+ {
+ nsStyleBackground* bg = new (mPresContext) nsStyleBackground(mPresContext);
+ aContext->SetStyle(eStyleStruct_Background, bg);
+ return bg;
+ }
+ case eStyleStruct_Margin:
+ {
+ nsStyleMargin* margin = new (mPresContext) nsStyleMargin(mPresContext);
+ aContext->SetStyle(eStyleStruct_Margin, margin);
+ return margin;
+ }
+ case eStyleStruct_Border:
+ {
+ nsStyleBorder* border = new (mPresContext) nsStyleBorder(mPresContext);
+ aContext->SetStyle(eStyleStruct_Border, border);
+ return border;
+ }
+ case eStyleStruct_Padding:
+ {
+ nsStylePadding* padding = new (mPresContext) nsStylePadding(mPresContext);
+ aContext->SetStyle(eStyleStruct_Padding, padding);
+ return padding;
+ }
+ case eStyleStruct_Outline:
+ {
+ nsStyleOutline* outline = new (mPresContext) nsStyleOutline(mPresContext);
+ aContext->SetStyle(eStyleStruct_Outline, outline);
+ return outline;
+ }
+ case eStyleStruct_List:
+ {
+ nsStyleList* list = new (mPresContext) nsStyleList(mPresContext);
+ aContext->SetStyle(eStyleStruct_List, list);
+ return list;
+ }
+ case eStyleStruct_Position:
+ {
+ nsStylePosition* pos = new (mPresContext) nsStylePosition(mPresContext);
+ aContext->SetStyle(eStyleStruct_Position, pos);
+ return pos;
+ }
+ case eStyleStruct_Table:
+ {
+ nsStyleTable* table = new (mPresContext) nsStyleTable(mPresContext);
+ aContext->SetStyle(eStyleStruct_Table, table);
+ return table;
+ }
+ case eStyleStruct_TableBorder:
+ {
+ nsStyleTableBorder* table = new (mPresContext) nsStyleTableBorder(mPresContext);
+ aContext->SetStyle(eStyleStruct_TableBorder, table);
+ return table;
+ }
+ case eStyleStruct_Content:
+ {
+ nsStyleContent* content = new (mPresContext) nsStyleContent(mPresContext);
+ aContext->SetStyle(eStyleStruct_Content, content);
+ return content;
+ }
+ case eStyleStruct_UserInterface:
+ {
+ nsStyleUserInterface* ui = new (mPresContext) nsStyleUserInterface(mPresContext);
+ aContext->SetStyle(eStyleStruct_UserInterface, ui);
+ return ui;
+ }
+ case eStyleStruct_UIReset:
+ {
+ nsStyleUIReset* ui = new (mPresContext) nsStyleUIReset(mPresContext);
+ aContext->SetStyle(eStyleStruct_UIReset, ui);
+ return ui;
+ }
+ case eStyleStruct_XUL:
+ {
+ nsStyleXUL* xul = new (mPresContext) nsStyleXUL(mPresContext);
+ aContext->SetStyle(eStyleStruct_XUL, xul);
+ return xul;
+ }
+ case eStyleStruct_Column:
+ {
+ nsStyleColumn* column = new (mPresContext) nsStyleColumn(mPresContext);
+ aContext->SetStyle(eStyleStruct_Column, column);
+ return column;
+ }
+ case eStyleStruct_SVG:
+ {
+ nsStyleSVG* svg = new (mPresContext) nsStyleSVG(mPresContext);
+ aContext->SetStyle(eStyleStruct_SVG, svg);
+ return svg;
+ }
+ case eStyleStruct_SVGReset:
+ {
+ nsStyleSVGReset* svgReset = new (mPresContext) nsStyleSVGReset(mPresContext);
+ aContext->SetStyle(eStyleStruct_SVGReset, svgReset);
+ return svgReset;
+ }
+ case eStyleStruct_Variables:
+ {
+ nsStyleVariables* vars = new (mPresContext) nsStyleVariables(mPresContext);
+ aContext->SetStyle(eStyleStruct_Variables, vars);
+ return vars;
+ }
+ case eStyleStruct_Effects:
+ {
+ nsStyleEffects* effects = new (mPresContext) nsStyleEffects(mPresContext);
+ aContext->SetStyle(eStyleStruct_Effects, effects);
+ return effects;
+ }
+ default:
+ /*
+ * unhandled case: nsStyleStructID_Length.
+ * last item of nsStyleStructID, to know its length.
+ */
+ MOZ_ASSERT(false, "unexpected SID");
+ return nullptr;
+ }
+ return nullptr;
+}
+
+/**
+ * Begin an nsRuleNode::Compute*Data function for an inherited struct.
+ *
+ * @param type_ The nsStyle* type this function computes.
+ * @param data_ Variable (declared here) holding the result of this
+ * function.
+ * @param parentdata_ Variable (declared here) holding the parent style
+ * context's data for this struct.
+ */
+#define COMPUTE_START_INHERITED(type_, data_, parentdata_) \
+ NS_ASSERTION(aRuleDetail != eRuleFullInherited, \
+ "should not have bothered calling Compute*Data"); \
+ \
+ nsStyleContext* parentContext = aContext->GetParent(); \
+ \
+ nsStyle##type_* data_ = nullptr; \
+ mozilla::Maybe<nsStyle##type_> maybeFakeParentData; \
+ const nsStyle##type_* parentdata_ = nullptr; \
+ RuleNodeCacheConditions conditions = aConditions; \
+ \
+ /* If |conditions.Cacheable()| might be true by the time we're done, we */ \
+ /* can't call parentContext->Style##type_() since it could recur into */ \
+ /* setting the same struct on the same rule node, causing a leak. */ \
+ if (aRuleDetail != eRuleFullReset && \
+ (!aStartStruct || (aRuleDetail != eRulePartialReset && \
+ aRuleDetail != eRuleNone))) { \
+ if (parentContext) { \
+ parentdata_ = parentContext->Style##type_(); \
+ } else { \
+ maybeFakeParentData.emplace(mPresContext); \
+ parentdata_ = maybeFakeParentData.ptr(); \
+ } \
+ } \
+ if (eStyleStruct_##type_ == eStyleStruct_Variables) \
+ /* no need to copy construct an nsStyleVariables, as we will copy */ \
+ /* inherited variables (and call SetUncacheable()) in */ \
+ /* ComputeVariablesData */ \
+ data_ = new (mPresContext) nsStyle##type_(mPresContext); \
+ else if (aStartStruct) \
+ /* We only need to compute the delta between this computed data and */ \
+ /* our computed data. */ \
+ data_ = new (mPresContext) \
+ nsStyle##type_(*static_cast<nsStyle##type_*>(aStartStruct)); \
+ else { \
+ if (aRuleDetail != eRuleFullMixed && aRuleDetail != eRuleFullReset) { \
+ /* No question. We will have to inherit. Go ahead and init */ \
+ /* with inherited vals from parent. */ \
+ conditions.SetUncacheable(); \
+ if (parentdata_) \
+ data_ = new (mPresContext) nsStyle##type_(*parentdata_); \
+ else \
+ data_ = new (mPresContext) nsStyle##type_(mPresContext); \
+ } \
+ else \
+ data_ = new (mPresContext) nsStyle##type_(mPresContext); \
+ } \
+ \
+ if (!parentdata_) \
+ parentdata_ = data_;
+
+/**
+ * Begin an nsRuleNode::Compute*Data function for a reset struct.
+ *
+ * @param type_ The nsStyle* type this function computes.
+ * @param data_ Variable (declared here) holding the result of this
+ * function.
+ * @param parentdata_ Variable (declared here) holding the parent style
+ * context's data for this struct.
+ */
+#define COMPUTE_START_RESET(type_, data_, parentdata_) \
+ NS_ASSERTION(aRuleDetail != eRuleFullInherited, \
+ "should not have bothered calling Compute*Data"); \
+ \
+ nsStyleContext* parentContext = aContext->GetParent(); \
+ /* Reset structs don't inherit from first-line */ \
+ /* See similar code in WalkRuleTree */ \
+ while (parentContext && \
+ parentContext->GetPseudo() == nsCSSPseudoElements::firstLine) { \
+ parentContext = parentContext->GetParent(); \
+ } \
+ \
+ nsStyle##type_* data_; \
+ if (aStartStruct) \
+ /* We only need to compute the delta between this computed data and */ \
+ /* our computed data. */ \
+ data_ = new (mPresContext) \
+ nsStyle##type_(*static_cast<nsStyle##type_*>(aStartStruct)); \
+ else \
+ data_ = new (mPresContext) nsStyle##type_(mPresContext); \
+ \
+ /* If |conditions.Cacheable()| might be true by the time we're done, we */ \
+ /* can't call parentContext->Style##type_() since it could recur into */ \
+ /* setting the same struct on the same rule node, causing a leak. */ \
+ mozilla::Maybe<nsStyle##type_> maybeFakeParentData; \
+ const nsStyle##type_* parentdata_ = data_; \
+ if (aRuleDetail != eRuleFullReset && \
+ aRuleDetail != eRulePartialReset && \
+ aRuleDetail != eRuleNone) { \
+ if (parentContext) { \
+ parentdata_ = parentContext->Style##type_(); \
+ } else { \
+ maybeFakeParentData.emplace(mPresContext); \
+ parentdata_ = maybeFakeParentData.ptr(); \
+ } \
+ } \
+ RuleNodeCacheConditions conditions = aConditions;
+
+/**
+ * End an nsRuleNode::Compute*Data function for an inherited struct.
+ *
+ * @param type_ The nsStyle* type this function computes.
+ * @param data_ Variable holding the result of this function.
+ */
+#define COMPUTE_END_INHERITED(type_, data_) \
+ NS_POSTCONDITION(!conditions.CacheableWithoutDependencies() || \
+ aRuleDetail == eRuleFullReset || \
+ (aStartStruct && aRuleDetail == eRulePartialReset), \
+ "conditions.CacheableWithoutDependencies() must be false " \
+ "for inherited structs unless all properties have been " \
+ "specified with values other than inherit"); \
+ if (conditions.CacheableWithoutDependencies()) { \
+ /* We were fully specified and can therefore be cached right on the */ \
+ /* rule node. */ \
+ if (!aHighestNode->mStyleData.mInheritedData) { \
+ aHighestNode->mStyleData.mInheritedData = \
+ new (mPresContext) nsInheritedStyleData; \
+ } \
+ NS_ASSERTION(!aHighestNode->mStyleData.mInheritedData-> \
+ mStyleStructs[eStyleStruct_##type_], \
+ "Going to leak style data"); \
+ aHighestNode->mStyleData.mInheritedData-> \
+ mStyleStructs[eStyleStruct_##type_] = data_; \
+ /* Propagate the bit down. */ \
+ PropagateDependentBit(eStyleStruct_##type_, aHighestNode, data_); \
+ /* Tell the style context that it doesn't own the data */ \
+ aContext->AddStyleBit(NS_STYLE_INHERIT_BIT(type_)); \
+ } \
+ /* For inherited structs, our caller will cache the data on the */ \
+ /* style context */ \
+ \
+ return data_;
+
+/**
+ * End an nsRuleNode::Compute*Data function for a reset struct.
+ *
+ * @param type_ The nsStyle* type this function computes.
+ * @param data_ Variable holding the result of this function.
+ */
+#define COMPUTE_END_RESET(type_, data_) \
+ NS_POSTCONDITION(!conditions.CacheableWithoutDependencies() || \
+ aRuleDetail == eRuleNone || \
+ aRuleDetail == eRulePartialReset || \
+ aRuleDetail == eRuleFullReset, \
+ "conditions.CacheableWithoutDependencies() must be false " \
+ "for reset structs if any properties were specified as " \
+ "inherit"); \
+ if (conditions.CacheableWithoutDependencies()) { \
+ /* We were fully specified and can therefore be cached right on the */ \
+ /* rule node. */ \
+ if (!aHighestNode->mStyleData.mResetData) { \
+ aHighestNode->mStyleData.mResetData = \
+ new (mPresContext) nsConditionalResetStyleData; \
+ } \
+ NS_ASSERTION(!aHighestNode->mStyleData.mResetData-> \
+ GetStyleData(eStyleStruct_##type_), \
+ "Going to leak style data"); \
+ aHighestNode->mStyleData.mResetData-> \
+ SetStyleData(eStyleStruct_##type_, data_); \
+ /* Propagate the bit down. */ \
+ PropagateDependentBit(eStyleStruct_##type_, aHighestNode, data_); \
+ if (HasAnimationData()) { \
+ /* If we have animation data, the struct should be cached on the */ \
+ /* style context so that we can peek the struct. */ \
+ /* See comment in AnimValuesStyleRule::MapRuleInfoInto. */ \
+ StoreStyleOnContext(aContext, eStyleStruct_##type_, data_); \
+ } \
+ } else if (conditions.Cacheable()) { \
+ if (!mStyleData.mResetData) { \
+ mStyleData.mResetData = new (mPresContext) nsConditionalResetStyleData; \
+ } \
+ mStyleData.mResetData-> \
+ SetStyleData(eStyleStruct_##type_, mPresContext, data_, conditions); \
+ /* Tell the style context that it doesn't own the data */ \
+ aContext->AddStyleBit(NS_STYLE_INHERIT_BIT(type_)); \
+ aContext->SetStyle(eStyleStruct_##type_, data_); \
+ } else { \
+ /* We can't be cached in the rule node. We have to be put right */ \
+ /* on the style context. */ \
+ aContext->SetStyle(eStyleStruct_##type_, data_); \
+ if (aContext->GetParent()) { \
+ /* This is pessimistic; we could be uncacheable because we had a */ \
+ /* relative font-weight, for example, which does not need to defeat */ \
+ /* the restyle optimizations in RestyleManager.cpp that look at */ \
+ /* NS_STYLE_HAS_CHILD_THAT_USES_RESET_STYLE. */ \
+ aContext->GetParent()-> \
+ AddStyleBit(NS_STYLE_HAS_CHILD_THAT_USES_RESET_STYLE); \
+ } \
+ } \
+ \
+ return data_;
+
+// This function figures out how much scaling should be suppressed to
+// satisfy scriptminsize. This is our attempt to implement
+// http://www.w3.org/TR/MathML2/chapter3.html#id.3.3.4.2.2
+// This is called after mScriptLevel, mScriptMinSize and mScriptSizeMultiplier
+// have been set in aFont.
+//
+// Here are the invariants we enforce:
+// 1) A decrease in size must not reduce the size below minscriptsize.
+// 2) An increase in size must not increase the size above the size we would
+// have if minscriptsize had not been applied anywhere.
+// 3) The scriptlevel-induced size change must between 1.0 and the parent's
+// scriptsizemultiplier^(new script level - old script level), as close to the
+// latter as possible subject to constraints 1 and 2.
+static nscoord
+ComputeScriptLevelSize(const nsStyleFont* aFont, const nsStyleFont* aParentFont,
+ nsPresContext* aPresContext, nscoord* aUnconstrainedSize)
+{
+ int32_t scriptLevelChange =
+ aFont->mScriptLevel - aParentFont->mScriptLevel;
+ if (scriptLevelChange == 0) {
+ *aUnconstrainedSize = aParentFont->mScriptUnconstrainedSize;
+ // Constraint #3 says that we cannot change size, and #1 and #2 are always
+ // satisfied with no change. It's important this be fast because it covers
+ // all non-MathML content.
+ return aParentFont->mSize;
+ }
+
+ // Compute actual value of minScriptSize
+ nscoord minScriptSize = aParentFont->mScriptMinSize;
+ if (aFont->mAllowZoom) {
+ minScriptSize = nsStyleFont::ZoomText(aPresContext, minScriptSize);
+ }
+
+ double scriptLevelScale =
+ pow(aParentFont->mScriptSizeMultiplier, scriptLevelChange);
+ // Compute the size we would have had if minscriptsize had never been
+ // applied, also prevent overflow (bug 413274)
+ *aUnconstrainedSize =
+ NSToCoordRoundWithClamp(aParentFont->mScriptUnconstrainedSize*scriptLevelScale);
+ // Compute the size we could get via scriptlevel change
+ nscoord scriptLevelSize =
+ NSToCoordRoundWithClamp(aParentFont->mSize*scriptLevelScale);
+ if (scriptLevelScale <= 1.0) {
+ if (aParentFont->mSize <= minScriptSize) {
+ // We can't decrease the font size at all, so just stick to no change
+ // (authors are allowed to explicitly set the font size smaller than
+ // minscriptsize)
+ return aParentFont->mSize;
+ }
+ // We can decrease, so apply constraint #1
+ return std::max(minScriptSize, scriptLevelSize);
+ } else {
+ // scriptminsize can only make sizes larger than the unconstrained size
+ NS_ASSERTION(*aUnconstrainedSize <= scriptLevelSize, "How can this ever happen?");
+ // Apply constraint #2
+ return std::min(scriptLevelSize, std::max(*aUnconstrainedSize, minScriptSize));
+ }
+}
+
+
+/* static */ nscoord
+nsRuleNode::CalcFontPointSize(int32_t aHTMLSize, int32_t aBasePointSize,
+ nsPresContext* aPresContext,
+ nsFontSizeType aFontSizeType)
+{
+#define sFontSizeTableMin 9
+#define sFontSizeTableMax 16
+
+// This table seems to be the one used by MacIE5. We hope its adoption in Mozilla
+// and eventually in WinIE5.5 will help to establish a standard rendering across
+// platforms and browsers. For now, it is used only in Strict mode. More can be read
+// in the document written by Todd Farhner at:
+// http://style.verso.com/font_size_intervals/altintervals.html
+//
+ static int32_t sStrictFontSizeTable[sFontSizeTableMax - sFontSizeTableMin + 1][8] =
+ {
+ { 9, 9, 9, 9, 11, 14, 18, 27},
+ { 9, 9, 9, 10, 12, 15, 20, 30},
+ { 9, 9, 10, 11, 13, 17, 22, 33},
+ { 9, 9, 10, 12, 14, 18, 24, 36},
+ { 9, 10, 12, 13, 16, 20, 26, 39},
+ { 9, 10, 12, 14, 17, 21, 28, 42},
+ { 9, 10, 13, 15, 18, 23, 30, 45},
+ { 9, 10, 13, 16, 18, 24, 32, 48}
+ };
+// HTML 1 2 3 4 5 6 7
+// CSS xxs xs s m l xl xxl
+// |
+// user pref
+//
+//------------------------------------------------------------
+//
+// This table gives us compatibility with WinNav4 for the default fonts only.
+// In WinNav4, the default fonts were:
+//
+// Times/12pt == Times/16px at 96ppi
+// Courier/10pt == Courier/13px at 96ppi
+//
+// The 2 lines below marked "anchored" have the exact pixel sizes used by
+// WinNav4 for Times/12pt and Courier/10pt at 96ppi. As you can see, the
+// HTML size 3 (user pref) for those 2 anchored lines is 13px and 16px.
+//
+// All values other than the anchored values were filled in by hand, never
+// going below 9px, and maintaining a "diagonal" relationship. See for
+// example the 13s -- they follow a diagonal line through the table.
+//
+ static int32_t sQuirksFontSizeTable[sFontSizeTableMax - sFontSizeTableMin + 1][8] =
+ {
+ { 9, 9, 9, 9, 11, 14, 18, 28 },
+ { 9, 9, 9, 10, 12, 15, 20, 31 },
+ { 9, 9, 9, 11, 13, 17, 22, 34 },
+ { 9, 9, 10, 12, 14, 18, 24, 37 },
+ { 9, 9, 10, 13, 16, 20, 26, 40 }, // anchored (13)
+ { 9, 9, 11, 14, 17, 21, 28, 42 },
+ { 9, 10, 12, 15, 17, 23, 30, 45 },
+ { 9, 10, 13, 16, 18, 24, 32, 48 } // anchored (16)
+ };
+// HTML 1 2 3 4 5 6 7
+// CSS xxs xs s m l xl xxl
+// |
+// user pref
+
+#if 0
+//
+// These are the exact pixel values used by WinIE5 at 96ppi.
+//
+ { ?, 8, 11, 12, 13, 16, 21, 32 }, // smallest
+ { ?, 9, 12, 13, 16, 21, 27, 40 }, // smaller
+ { ?, 10, 13, 16, 18, 24, 32, 48 }, // medium
+ { ?, 13, 16, 19, 21, 27, 37, ?? }, // larger
+ { ?, 16, 19, 21, 24, 32, 43, ?? } // largest
+//
+// HTML 1 2 3 4 5 6 7
+// CSS ? ? ? ? ? ? ? ?
+//
+// (CSS not tested yet.)
+//
+#endif
+
+ static int32_t sFontSizeFactors[8] = { 60,75,89,100,120,150,200,300 };
+
+ static int32_t sCSSColumns[7] = {0, 1, 2, 3, 4, 5, 6}; // xxs...xxl
+ static int32_t sHTMLColumns[7] = {1, 2, 3, 4, 5, 6, 7}; // 1...7
+
+ double dFontSize;
+
+ if (aFontSizeType == eFontSize_HTML) {
+ aHTMLSize--; // input as 1-7
+ }
+
+ if (aHTMLSize < 0)
+ aHTMLSize = 0;
+ else if (aHTMLSize > 6)
+ aHTMLSize = 6;
+
+ int32_t* column;
+ switch (aFontSizeType)
+ {
+ case eFontSize_HTML: column = sHTMLColumns; break;
+ case eFontSize_CSS: column = sCSSColumns; break;
+ }
+
+ // Make special call specifically for fonts (needed PrintPreview)
+ int32_t fontSize = nsPresContext::AppUnitsToIntCSSPixels(aBasePointSize);
+
+ if ((fontSize >= sFontSizeTableMin) && (fontSize <= sFontSizeTableMax))
+ {
+ int32_t row = fontSize - sFontSizeTableMin;
+
+ if (aPresContext->CompatibilityMode() == eCompatibility_NavQuirks) {
+ dFontSize = nsPresContext::CSSPixelsToAppUnits(sQuirksFontSizeTable[row][column[aHTMLSize]]);
+ } else {
+ dFontSize = nsPresContext::CSSPixelsToAppUnits(sStrictFontSizeTable[row][column[aHTMLSize]]);
+ }
+ }
+ else
+ {
+ int32_t factor = sFontSizeFactors[column[aHTMLSize]];
+ dFontSize = (factor * aBasePointSize) / 100;
+ }
+
+
+ if (1.0 < dFontSize) {
+ return (nscoord)dFontSize;
+ }
+ return (nscoord)1;
+}
+
+
+//------------------------------------------------------------------------------
+//
+//------------------------------------------------------------------------------
+
+/* static */ nscoord
+nsRuleNode::FindNextSmallerFontSize(nscoord aFontSize, int32_t aBasePointSize,
+ nsPresContext* aPresContext,
+ nsFontSizeType aFontSizeType)
+{
+ int32_t index;
+ int32_t indexMin;
+ int32_t indexMax;
+ float relativePosition;
+ nscoord smallerSize;
+ nscoord indexFontSize = aFontSize; // XXX initialize to quell a spurious gcc3.2 warning
+ nscoord smallestIndexFontSize;
+ nscoord largestIndexFontSize;
+ nscoord smallerIndexFontSize;
+ nscoord largerIndexFontSize;
+
+ nscoord onePx = nsPresContext::CSSPixelsToAppUnits(1);
+
+ if (aFontSizeType == eFontSize_HTML) {
+ indexMin = 1;
+ indexMax = 7;
+ } else {
+ indexMin = 0;
+ indexMax = 6;
+ }
+
+ smallestIndexFontSize = CalcFontPointSize(indexMin, aBasePointSize, aPresContext, aFontSizeType);
+ largestIndexFontSize = CalcFontPointSize(indexMax, aBasePointSize, aPresContext, aFontSizeType);
+ if (aFontSize > smallestIndexFontSize) {
+ if (aFontSize < NSToCoordRound(float(largestIndexFontSize) * 1.5)) { // smaller will be in HTML table
+ // find largest index smaller than current
+ for (index = indexMax; index >= indexMin; index--) {
+ indexFontSize = CalcFontPointSize(index, aBasePointSize, aPresContext, aFontSizeType);
+ if (indexFontSize < aFontSize)
+ break;
+ }
+ // set up points beyond table for interpolation purposes
+ if (indexFontSize == smallestIndexFontSize) {
+ smallerIndexFontSize = indexFontSize - onePx;
+ largerIndexFontSize = CalcFontPointSize(index+1, aBasePointSize, aPresContext, aFontSizeType);
+ } else if (indexFontSize == largestIndexFontSize) {
+ smallerIndexFontSize = CalcFontPointSize(index-1, aBasePointSize, aPresContext, aFontSizeType);
+ largerIndexFontSize = NSToCoordRound(float(largestIndexFontSize) * 1.5);
+ } else {
+ smallerIndexFontSize = CalcFontPointSize(index-1, aBasePointSize, aPresContext, aFontSizeType);
+ largerIndexFontSize = CalcFontPointSize(index+1, aBasePointSize, aPresContext, aFontSizeType);
+ }
+ // compute the relative position of the parent size between the two closest indexed sizes
+ relativePosition = float(aFontSize - indexFontSize) / float(largerIndexFontSize - indexFontSize);
+ // set the new size to have the same relative position between the next smallest two indexed sizes
+ smallerSize = smallerIndexFontSize + NSToCoordRound(relativePosition * (indexFontSize - smallerIndexFontSize));
+ }
+ else { // larger than HTML table, drop by 33%
+ smallerSize = NSToCoordRound(float(aFontSize) / 1.5);
+ }
+ }
+ else { // smaller than HTML table, drop by 1px
+ smallerSize = std::max(aFontSize - onePx, onePx);
+ }
+ return smallerSize;
+}
+
+//------------------------------------------------------------------------------
+//
+//------------------------------------------------------------------------------
+
+/* static */ nscoord
+nsRuleNode::FindNextLargerFontSize(nscoord aFontSize, int32_t aBasePointSize,
+ nsPresContext* aPresContext,
+ nsFontSizeType aFontSizeType)
+{
+ int32_t index;
+ int32_t indexMin;
+ int32_t indexMax;
+ float relativePosition;
+ nscoord adjustment;
+ nscoord largerSize;
+ nscoord indexFontSize = aFontSize; // XXX initialize to quell a spurious gcc3.2 warning
+ nscoord smallestIndexFontSize;
+ nscoord largestIndexFontSize;
+ nscoord smallerIndexFontSize;
+ nscoord largerIndexFontSize;
+
+ nscoord onePx = nsPresContext::CSSPixelsToAppUnits(1);
+
+ if (aFontSizeType == eFontSize_HTML) {
+ indexMin = 1;
+ indexMax = 7;
+ } else {
+ indexMin = 0;
+ indexMax = 6;
+ }
+
+ smallestIndexFontSize = CalcFontPointSize(indexMin, aBasePointSize, aPresContext, aFontSizeType);
+ largestIndexFontSize = CalcFontPointSize(indexMax, aBasePointSize, aPresContext, aFontSizeType);
+ if (aFontSize > (smallestIndexFontSize - onePx)) {
+ if (aFontSize < largestIndexFontSize) { // larger will be in HTML table
+ // find smallest index larger than current
+ for (index = indexMin; index <= indexMax; index++) {
+ indexFontSize = CalcFontPointSize(index, aBasePointSize, aPresContext, aFontSizeType);
+ if (indexFontSize > aFontSize)
+ break;
+ }
+ // set up points beyond table for interpolation purposes
+ if (indexFontSize == smallestIndexFontSize) {
+ smallerIndexFontSize = indexFontSize - onePx;
+ largerIndexFontSize = CalcFontPointSize(index+1, aBasePointSize, aPresContext, aFontSizeType);
+ } else if (indexFontSize == largestIndexFontSize) {
+ smallerIndexFontSize = CalcFontPointSize(index-1, aBasePointSize, aPresContext, aFontSizeType);
+ largerIndexFontSize = NSCoordSaturatingMultiply(largestIndexFontSize, 1.5);
+ } else {
+ smallerIndexFontSize = CalcFontPointSize(index-1, aBasePointSize, aPresContext, aFontSizeType);
+ largerIndexFontSize = CalcFontPointSize(index+1, aBasePointSize, aPresContext, aFontSizeType);
+ }
+ // compute the relative position of the parent size between the two closest indexed sizes
+ relativePosition = float(aFontSize - smallerIndexFontSize) / float(indexFontSize - smallerIndexFontSize);
+ // set the new size to have the same relative position between the next largest two indexed sizes
+ adjustment = NSCoordSaturatingNonnegativeMultiply(largerIndexFontSize - indexFontSize, relativePosition);
+ largerSize = NSCoordSaturatingAdd(indexFontSize, adjustment);
+ }
+ else { // larger than HTML table, increase by 50%
+ largerSize = NSCoordSaturatingMultiply(aFontSize, 1.5);
+ }
+ }
+ else { // smaller than HTML table, increase by 1px
+ largerSize = NSCoordSaturatingAdd(aFontSize, onePx);
+ }
+ return largerSize;
+}
+
+struct SetFontSizeCalcOps : public css::BasicCoordCalcOps,
+ public css::NumbersAlreadyNormalizedOps
+{
+ // The parameters beyond aValue that we need for CalcLengthWith.
+ const nscoord mParentSize;
+ const nsStyleFont* const mParentFont;
+ nsPresContext* const mPresContext;
+ nsStyleContext* const mStyleContext;
+ const bool mAtRoot;
+ RuleNodeCacheConditions& mConditions;
+
+ SetFontSizeCalcOps(nscoord aParentSize, const nsStyleFont* aParentFont,
+ nsPresContext* aPresContext,
+ nsStyleContext* aStyleContext,
+ bool aAtRoot,
+ RuleNodeCacheConditions& aConditions)
+ : mParentSize(aParentSize),
+ mParentFont(aParentFont),
+ mPresContext(aPresContext),
+ mStyleContext(aStyleContext),
+ mAtRoot(aAtRoot),
+ mConditions(aConditions)
+ {
+ }
+
+ result_type ComputeLeafValue(const nsCSSValue& aValue)
+ {
+ nscoord size;
+ if (aValue.IsLengthUnit()) {
+ // Note that font-based length units use the parent's size
+ // unadjusted for scriptlevel changes. A scriptlevel change
+ // between us and the parent is simply ignored.
+ size = CalcLengthWith(aValue, mParentSize,
+ mParentFont,
+ mStyleContext, mPresContext, mAtRoot,
+ true, mConditions);
+ if (!aValue.IsRelativeLengthUnit() && mParentFont->mAllowZoom) {
+ size = nsStyleFont::ZoomText(mPresContext, size);
+ }
+ }
+ else if (eCSSUnit_Percent == aValue.GetUnit()) {
+ mConditions.SetUncacheable();
+ // Note that % units use the parent's size unadjusted for scriptlevel
+ // changes. A scriptlevel change between us and the parent is simply
+ // ignored.
+ // aValue.GetPercentValue() may be negative for, e.g., calc(-50%)
+ size = NSCoordSaturatingMultiply(mParentSize, aValue.GetPercentValue());
+ } else {
+ MOZ_ASSERT(false, "unexpected value");
+ size = mParentSize;
+ }
+
+ return size;
+ }
+};
+
+/* static */ void
+nsRuleNode::SetFontSize(nsPresContext* aPresContext,
+ nsStyleContext* aContext,
+ const nsRuleData* aRuleData,
+ const nsStyleFont* aFont,
+ const nsStyleFont* aParentFont,
+ nscoord* aSize,
+ const nsFont& aSystemFont,
+ nscoord aParentSize,
+ nscoord aScriptLevelAdjustedParentSize,
+ bool aUsedStartStruct,
+ bool aAtRoot,
+ RuleNodeCacheConditions& aConditions)
+{
+ // If false, means that *aSize has not been zoomed. If true, means that
+ // *aSize has been zoomed iff aParentFont->mAllowZoom is true.
+ bool sizeIsZoomedAccordingToParent = false;
+
+ int32_t baseSize = (int32_t) aPresContext->
+ GetDefaultFont(aFont->mGenericID, aFont->mLanguage)->size;
+ const nsCSSValue* sizeValue = aRuleData->ValueForFontSize();
+ if (eCSSUnit_Enumerated == sizeValue->GetUnit()) {
+ int32_t value = sizeValue->GetIntValue();
+
+ if ((NS_STYLE_FONT_SIZE_XXSMALL <= value) &&
+ (value <= NS_STYLE_FONT_SIZE_XXLARGE)) {
+ *aSize = CalcFontPointSize(value, baseSize,
+ aPresContext, eFontSize_CSS);
+ }
+ else if (NS_STYLE_FONT_SIZE_XXXLARGE == value) {
+ // <font size="7"> is not specified in CSS, so we don't use eFontSize_CSS.
+ *aSize = CalcFontPointSize(value, baseSize, aPresContext);
+ }
+ else if (NS_STYLE_FONT_SIZE_LARGER == value ||
+ NS_STYLE_FONT_SIZE_SMALLER == value) {
+ aConditions.SetUncacheable();
+
+ // Un-zoom so we use the tables correctly. We'll then rezoom due
+ // to the |zoom = true| above.
+ // Note that relative units here use the parent's size unadjusted
+ // for scriptlevel changes. A scriptlevel change between us and the parent
+ // is simply ignored.
+ nscoord parentSize = aParentSize;
+ if (aParentFont->mAllowZoom) {
+ parentSize = nsStyleFont::UnZoomText(aPresContext, parentSize);
+ }
+
+ if (NS_STYLE_FONT_SIZE_LARGER == value) {
+ *aSize = FindNextLargerFontSize(parentSize,
+ baseSize, aPresContext, eFontSize_CSS);
+
+ NS_ASSERTION(*aSize >= parentSize,
+ "FindNextLargerFontSize failed");
+ }
+ else {
+ *aSize = FindNextSmallerFontSize(parentSize,
+ baseSize, aPresContext, eFontSize_CSS);
+ NS_ASSERTION(*aSize < parentSize ||
+ parentSize <= nsPresContext::CSSPixelsToAppUnits(1),
+ "FindNextSmallerFontSize failed");
+ }
+ } else {
+ NS_NOTREACHED("unexpected value");
+ }
+ }
+ else if (sizeValue->IsLengthUnit() ||
+ sizeValue->GetUnit() == eCSSUnit_Percent ||
+ sizeValue->IsCalcUnit()) {
+ SetFontSizeCalcOps ops(aParentSize, aParentFont,
+ aPresContext, aContext,
+ aAtRoot,
+ aConditions);
+ *aSize = css::ComputeCalc(*sizeValue, ops);
+ if (*aSize < 0) {
+ MOZ_ASSERT(sizeValue->IsCalcUnit(),
+ "negative lengths and percents should be rejected by parser");
+ *aSize = 0;
+ }
+ // The calc ops will always zoom its result according to the value
+ // of aParentFont->mAllowZoom.
+ sizeIsZoomedAccordingToParent = true;
+ }
+ else if (eCSSUnit_System_Font == sizeValue->GetUnit()) {
+ // this becomes our cascading size
+ *aSize = aSystemFont.size;
+ }
+ else if (eCSSUnit_Inherit == sizeValue->GetUnit() ||
+ eCSSUnit_Unset == sizeValue->GetUnit()) {
+ aConditions.SetUncacheable();
+ // We apply scriptlevel change for this case, because the default is
+ // to inherit and we don't want explicit "inherit" to differ from the
+ // default.
+ *aSize = aScriptLevelAdjustedParentSize;
+ sizeIsZoomedAccordingToParent = true;
+ }
+ else if (eCSSUnit_Initial == sizeValue->GetUnit()) {
+ // The initial value is 'medium', which has magical sizing based on
+ // the generic font family, so do that here too.
+ *aSize = baseSize;
+ } else {
+ NS_ASSERTION(eCSSUnit_Null == sizeValue->GetUnit(),
+ "What kind of font-size value is this?");
+ // if aUsedStartStruct is true, then every single property in the
+ // font struct is being set all at once. This means scriptlevel is not
+ // going to have any influence on the font size; there is no need to
+ // do anything here.
+ if (!aUsedStartStruct && aParentSize != aScriptLevelAdjustedParentSize) {
+ // There was no rule affecting the size but the size has been
+ // affected by the parent's size via scriptlevel change. So we cannot
+ // store the data in the rule tree.
+ aConditions.SetUncacheable();
+ *aSize = aScriptLevelAdjustedParentSize;
+ sizeIsZoomedAccordingToParent = true;
+ } else {
+ return;
+ }
+ }
+
+ // We want to zoom the cascaded size so that em-based measurements,
+ // line-heights, etc., work.
+ bool currentlyZoomed = sizeIsZoomedAccordingToParent &&
+ aParentFont->mAllowZoom;
+ if (!currentlyZoomed && aFont->mAllowZoom) {
+ *aSize = nsStyleFont::ZoomText(aPresContext, *aSize);
+ } else if (currentlyZoomed && !aFont->mAllowZoom) {
+ *aSize = nsStyleFont::UnZoomText(aPresContext, *aSize);
+ }
+}
+
+static int8_t ClampTo8Bit(int32_t aValue) {
+ if (aValue < -128)
+ return -128;
+ if (aValue > 127)
+ return 127;
+ return int8_t(aValue);
+}
+
+/* static */ void
+nsRuleNode::SetFont(nsPresContext* aPresContext, nsStyleContext* aContext,
+ uint8_t aGenericFontID, const nsRuleData* aRuleData,
+ const nsStyleFont* aParentFont,
+ nsStyleFont* aFont, bool aUsedStartStruct,
+ RuleNodeCacheConditions& aConditions)
+{
+ bool atRoot = !aContext->GetParent();
+
+ // -x-text-zoom: none, inherit, initial
+ bool allowZoom;
+ const nsCSSValue* textZoomValue = aRuleData->ValueForTextZoom();
+ if (eCSSUnit_Null != textZoomValue->GetUnit()) {
+ if (eCSSUnit_Inherit == textZoomValue->GetUnit()) {
+ allowZoom = aParentFont->mAllowZoom;
+ } else if (eCSSUnit_None == textZoomValue->GetUnit()) {
+ allowZoom = false;
+ } else {
+ MOZ_ASSERT(eCSSUnit_Initial == textZoomValue->GetUnit(),
+ "unexpected unit");
+ allowZoom = true;
+ }
+ aFont->EnableZoom(aPresContext, allowZoom);
+ }
+
+ // mLanguage must be set before before any of the CalcLengthWith calls
+ // (direct calls or calls via SetFontSize) for the cases where |aParentFont|
+ // is the same as |aFont|.
+ //
+ // -x-lang: string, inherit
+ // This is not a real CSS property, it is an HTML attribute mapped to CSS.
+ const nsCSSValue* langValue = aRuleData->ValueForLang();
+ if (eCSSUnit_Ident == langValue->GetUnit()) {
+ nsAutoString lang;
+ langValue->GetStringValue(lang);
+
+ nsContentUtils::ASCIIToLower(lang);
+ aFont->mLanguage = NS_Atomize(lang);
+ aFont->mExplicitLanguage = true;
+ }
+
+ const nsFont* defaultVariableFont =
+ aPresContext->GetDefaultFont(kPresContext_DefaultVariableFont_ID,
+ aFont->mLanguage);
+
+ // -moz-system-font: enum (never inherit!)
+ static_assert(
+ NS_STYLE_FONT_CAPTION == LookAndFeel::eFont_Caption &&
+ NS_STYLE_FONT_ICON == LookAndFeel::eFont_Icon &&
+ NS_STYLE_FONT_MENU == LookAndFeel::eFont_Menu &&
+ NS_STYLE_FONT_MESSAGE_BOX == LookAndFeel::eFont_MessageBox &&
+ NS_STYLE_FONT_SMALL_CAPTION == LookAndFeel::eFont_SmallCaption &&
+ NS_STYLE_FONT_STATUS_BAR == LookAndFeel::eFont_StatusBar &&
+ NS_STYLE_FONT_WINDOW == LookAndFeel::eFont_Window &&
+ NS_STYLE_FONT_DOCUMENT == LookAndFeel::eFont_Document &&
+ NS_STYLE_FONT_WORKSPACE == LookAndFeel::eFont_Workspace &&
+ NS_STYLE_FONT_DESKTOP == LookAndFeel::eFont_Desktop &&
+ NS_STYLE_FONT_INFO == LookAndFeel::eFont_Info &&
+ NS_STYLE_FONT_DIALOG == LookAndFeel::eFont_Dialog &&
+ NS_STYLE_FONT_BUTTON == LookAndFeel::eFont_Button &&
+ NS_STYLE_FONT_PULL_DOWN_MENU == LookAndFeel::eFont_PullDownMenu &&
+ NS_STYLE_FONT_LIST == LookAndFeel::eFont_List &&
+ NS_STYLE_FONT_FIELD == LookAndFeel::eFont_Field,
+ "LookAndFeel.h system-font constants out of sync with nsStyleConsts.h");
+
+ // Fall back to defaultVariableFont.
+ nsFont systemFont = *defaultVariableFont;
+ const nsCSSValue* systemFontValue = aRuleData->ValueForSystemFont();
+ if (eCSSUnit_Enumerated == systemFontValue->GetUnit()) {
+ gfxFontStyle fontStyle;
+ LookAndFeel::FontID fontID =
+ (LookAndFeel::FontID)systemFontValue->GetIntValue();
+ float devPerCSS =
+ (float)nsPresContext::AppUnitsPerCSSPixel() /
+ aPresContext->DeviceContext()->AppUnitsPerDevPixelAtUnitFullZoom();
+ nsAutoString systemFontName;
+ if (LookAndFeel::GetFont(fontID, systemFontName, fontStyle, devPerCSS)) {
+ systemFontName.Trim("\"'");
+ systemFont.fontlist = FontFamilyList(systemFontName, eUnquotedName);
+ systemFont.fontlist.SetDefaultFontType(eFamily_none);
+ systemFont.style = fontStyle.style;
+ systemFont.systemFont = fontStyle.systemFont;
+ systemFont.weight = fontStyle.weight;
+ systemFont.stretch = fontStyle.stretch;
+ systemFont.size =
+ NSFloatPixelsToAppUnits(fontStyle.size,
+ aPresContext->DeviceContext()->
+ AppUnitsPerDevPixelAtUnitFullZoom());
+ //systemFont.langGroup = fontStyle.langGroup;
+ systemFont.sizeAdjust = fontStyle.sizeAdjust;
+
+#ifdef XP_WIN
+ // XXXldb This platform-specific stuff should be in the
+ // LookAndFeel implementation, not here.
+ // XXXzw Should we even still *have* this code? It looks to be making
+ // old, probably obsolete assumptions.
+
+ if (fontID == LookAndFeel::eFont_Field ||
+ fontID == LookAndFeel::eFont_Button ||
+ fontID == LookAndFeel::eFont_List) {
+ // As far as I can tell the system default fonts and sizes
+ // on MS-Windows for Buttons, Listboxes/Comboxes and Text Fields are
+ // all pre-determined and cannot be changed by either the control panel
+ // or programmatically.
+ // Fields (text fields)
+ // Button and Selects (listboxes/comboboxes)
+ // We use whatever font is defined by the system. Which it appears
+ // (and the assumption is) it is always a proportional font. Then we
+ // always use 2 points smaller than what the browser has defined as
+ // the default proportional font.
+ // Assumption: system defined font is proportional
+ systemFont.size =
+ std::max(defaultVariableFont->size -
+ nsPresContext::CSSPointsToAppUnits(2), 0);
+ }
+#endif
+ }
+ }
+
+ // font-family: font family list, enum, inherit
+ const nsCSSValue* familyValue = aRuleData->ValueForFontFamily();
+ NS_ASSERTION(eCSSUnit_Enumerated != familyValue->GetUnit(),
+ "system fonts should not be in mFamily anymore");
+ if (eCSSUnit_FontFamilyList == familyValue->GetUnit()) {
+ // set the correct font if we are using DocumentFonts OR we are overriding for XUL
+ // MJA: bug 31816
+ bool useDocumentFonts =
+ aPresContext->GetCachedBoolPref(kPresContext_UseDocumentFonts);
+ if (aGenericFontID == kGenericFont_NONE ||
+ (!useDocumentFonts && (aGenericFontID == kGenericFont_cursive ||
+ aGenericFontID == kGenericFont_fantasy))) {
+ FontFamilyType defaultGeneric =
+ defaultVariableFont->fontlist.FirstGeneric();
+ MOZ_ASSERT(defaultVariableFont->fontlist.Length() == 1 &&
+ (defaultGeneric == eFamily_serif ||
+ defaultGeneric == eFamily_sans_serif));
+ if (defaultGeneric != eFamily_none) {
+ if (useDocumentFonts) {
+ aFont->mFont.fontlist.SetDefaultFontType(defaultGeneric);
+ } else {
+ // Either prioritize the first generic in the list,
+ // or (if there isn't one) prepend the default variable font.
+ if (!aFont->mFont.fontlist.PrioritizeFirstGeneric()) {
+ aFont->mFont.fontlist.PrependGeneric(defaultGeneric);
+ }
+ }
+ }
+ } else {
+ aFont->mFont.fontlist.SetDefaultFontType(eFamily_none);
+ }
+ aFont->mFont.systemFont = false;
+ // Technically this is redundant with the code below, but it's good
+ // to have since we'll still want it once we get rid of
+ // SetGenericFont (bug 380915).
+ aFont->mGenericID = aGenericFontID;
+ }
+ else if (eCSSUnit_System_Font == familyValue->GetUnit()) {
+ aFont->mFont.fontlist = systemFont.fontlist;
+ aFont->mFont.systemFont = true;
+ aFont->mGenericID = kGenericFont_NONE;
+ }
+ else if (eCSSUnit_Inherit == familyValue->GetUnit() ||
+ eCSSUnit_Unset == familyValue->GetUnit()) {
+ aConditions.SetUncacheable();
+ aFont->mFont.fontlist = aParentFont->mFont.fontlist;
+ aFont->mFont.systemFont = aParentFont->mFont.systemFont;
+ aFont->mGenericID = aParentFont->mGenericID;
+ }
+ else if (eCSSUnit_Initial == familyValue->GetUnit()) {
+ aFont->mFont.fontlist = defaultVariableFont->fontlist;
+ aFont->mFont.systemFont = defaultVariableFont->systemFont;
+ aFont->mGenericID = kGenericFont_NONE;
+ }
+
+ // When we're in the loop in SetGenericFont, we must ensure that we
+ // always keep aFont->mFlags set to the correct generic. But we have
+ // to be careful not to touch it when we're called directly from
+ // ComputeFontData, because we could have a start struct.
+ if (aGenericFontID != kGenericFont_NONE) {
+ aFont->mGenericID = aGenericFontID;
+ }
+
+ // -moz-math-variant: enum, inherit, initial
+ SetValue(*aRuleData->ValueForMathVariant(), aFont->mMathVariant,
+ aConditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ aParentFont->mMathVariant, NS_MATHML_MATHVARIANT_NONE);
+
+ // -moz-math-display: enum, inherit, initial
+ SetValue(*aRuleData->ValueForMathDisplay(), aFont->mMathDisplay,
+ aConditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ aParentFont->mMathDisplay, NS_MATHML_DISPLAYSTYLE_INLINE);
+
+ // font-smoothing: enum, inherit, initial
+ SetValue(*aRuleData->ValueForOsxFontSmoothing(),
+ aFont->mFont.smoothing, aConditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ aParentFont->mFont.smoothing,
+ defaultVariableFont->smoothing);
+
+ // font-style: enum, inherit, initial, -moz-system-font
+ if (aFont->mMathVariant != NS_MATHML_MATHVARIANT_NONE) {
+ // -moz-math-variant overrides font-style
+ aFont->mFont.style = NS_FONT_STYLE_NORMAL;
+ } else {
+ SetValue(*aRuleData->ValueForFontStyle(),
+ aFont->mFont.style, aConditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ aParentFont->mFont.style,
+ defaultVariableFont->style,
+ Unused, Unused, Unused, systemFont.style);
+ }
+
+ // font-weight: int, enum, inherit, initial, -moz-system-font
+ // special handling for enum
+ const nsCSSValue* weightValue = aRuleData->ValueForFontWeight();
+ if (aFont->mMathVariant != NS_MATHML_MATHVARIANT_NONE) {
+ // -moz-math-variant overrides font-weight
+ aFont->mFont.weight = NS_FONT_WEIGHT_NORMAL;
+ } else if (eCSSUnit_Enumerated == weightValue->GetUnit()) {
+ int32_t value = weightValue->GetIntValue();
+ switch (value) {
+ case NS_STYLE_FONT_WEIGHT_NORMAL:
+ case NS_STYLE_FONT_WEIGHT_BOLD:
+ aFont->mFont.weight = value;
+ break;
+ case NS_STYLE_FONT_WEIGHT_BOLDER: {
+ aConditions.SetUncacheable();
+ int32_t inheritedValue = aParentFont->mFont.weight;
+ if (inheritedValue <= 300) {
+ aFont->mFont.weight = 400;
+ } else if (inheritedValue <= 500) {
+ aFont->mFont.weight = 700;
+ } else {
+ aFont->mFont.weight = 900;
+ }
+ break;
+ }
+ case NS_STYLE_FONT_WEIGHT_LIGHTER: {
+ aConditions.SetUncacheable();
+ int32_t inheritedValue = aParentFont->mFont.weight;
+ if (inheritedValue < 600) {
+ aFont->mFont.weight = 100;
+ } else if (inheritedValue < 800) {
+ aFont->mFont.weight = 400;
+ } else {
+ aFont->mFont.weight = 700;
+ }
+ break;
+ }
+ }
+ } else
+ SetValue(*weightValue, aFont->mFont.weight, aConditions,
+ SETVAL_INTEGER | SETVAL_UNSET_INHERIT,
+ aParentFont->mFont.weight,
+ defaultVariableFont->weight,
+ Unused, Unused, Unused, systemFont.weight);
+
+ // font-stretch: enum, inherit, initial, -moz-system-font
+ SetValue(*aRuleData->ValueForFontStretch(),
+ aFont->mFont.stretch, aConditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ aParentFont->mFont.stretch,
+ defaultVariableFont->stretch,
+ Unused, Unused, Unused, systemFont.stretch);
+
+ // Compute scriptlevel, scriptminsize and scriptsizemultiplier now so
+ // they're available for font-size computation.
+
+ // -moz-script-min-size: length
+ const nsCSSValue* scriptMinSizeValue = aRuleData->ValueForScriptMinSize();
+ if (scriptMinSizeValue->IsLengthUnit()) {
+ // scriptminsize in font units (em, ex) has to be interpreted relative
+ // to the parent font, or the size definitions are circular and we
+ //
+ aFont->mScriptMinSize =
+ CalcLengthWith(*scriptMinSizeValue, aParentFont->mSize,
+ aParentFont,
+ aContext, aPresContext, atRoot, true /* aUseUserFontSet */,
+ aConditions);
+ }
+
+ // -moz-script-size-multiplier: factor, inherit, initial
+ SetFactor(*aRuleData->ValueForScriptSizeMultiplier(),
+ aFont->mScriptSizeMultiplier,
+ aConditions, aParentFont->mScriptSizeMultiplier,
+ NS_MATHML_DEFAULT_SCRIPT_SIZE_MULTIPLIER,
+ SETFCT_POSITIVE | SETFCT_UNSET_INHERIT);
+
+ // -moz-script-level: integer, number, inherit
+ const nsCSSValue* scriptLevelValue = aRuleData->ValueForScriptLevel();
+ if (eCSSUnit_Integer == scriptLevelValue->GetUnit()) {
+ // "relative"
+ aConditions.SetUncacheable();
+ aFont->mScriptLevel = ClampTo8Bit(aParentFont->mScriptLevel + scriptLevelValue->GetIntValue());
+ }
+ else if (eCSSUnit_Number == scriptLevelValue->GetUnit()) {
+ // "absolute"
+ aFont->mScriptLevel = ClampTo8Bit(int32_t(scriptLevelValue->GetFloatValue()));
+ }
+ else if (eCSSUnit_Auto == scriptLevelValue->GetUnit()) {
+ // auto
+ aConditions.SetUncacheable();
+ aFont->mScriptLevel = ClampTo8Bit(aParentFont->mScriptLevel +
+ (aParentFont->mMathDisplay ==
+ NS_MATHML_DISPLAYSTYLE_INLINE ? 1 : 0));
+ }
+ else if (eCSSUnit_Inherit == scriptLevelValue->GetUnit() ||
+ eCSSUnit_Unset == scriptLevelValue->GetUnit()) {
+ aConditions.SetUncacheable();
+ aFont->mScriptLevel = aParentFont->mScriptLevel;
+ }
+ else if (eCSSUnit_Initial == scriptLevelValue->GetUnit()) {
+ aFont->mScriptLevel = 0;
+ }
+
+ // font-kerning: none, enum, inherit, initial, -moz-system-font
+ SetValue(*aRuleData->ValueForFontKerning(),
+ aFont->mFont.kerning, aConditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ aParentFont->mFont.kerning,
+ defaultVariableFont->kerning,
+ Unused, Unused, Unused, systemFont.kerning);
+
+ // font-synthesis: none, enum (bit field), inherit, initial, -moz-system-font
+ SetValue(*aRuleData->ValueForFontSynthesis(),
+ aFont->mFont.synthesis, aConditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ aParentFont->mFont.synthesis,
+ defaultVariableFont->synthesis,
+ Unused, /* none */ 0, Unused, systemFont.synthesis);
+
+ // font-variant-alternates: normal, enum (bit field) + functions, inherit,
+ // initial, -moz-system-font
+ const nsCSSValue* variantAlternatesValue =
+ aRuleData->ValueForFontVariantAlternates();
+ int32_t variantAlternates = 0;
+
+ switch (variantAlternatesValue->GetUnit()) {
+ case eCSSUnit_Inherit:
+ case eCSSUnit_Unset:
+ aFont->mFont.CopyAlternates(aParentFont->mFont);
+ aConditions.SetUncacheable();
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Normal:
+ aFont->mFont.variantAlternates = 0;
+ aFont->mFont.alternateValues.Clear();
+ aFont->mFont.featureValueLookup = nullptr;
+ break;
+
+ case eCSSUnit_Pair:
+ NS_ASSERTION(variantAlternatesValue->GetPairValue().mXValue.GetUnit() ==
+ eCSSUnit_Enumerated, "strange unit for variantAlternates");
+ variantAlternates =
+ variantAlternatesValue->GetPairValue().mXValue.GetIntValue();
+ aFont->mFont.variantAlternates = variantAlternates;
+
+ if (variantAlternates & NS_FONT_VARIANT_ALTERNATES_FUNCTIONAL_MASK) {
+ // fetch the feature lookup object from the styleset
+ MOZ_ASSERT(aPresContext->StyleSet()->IsGecko(),
+ "ServoStyleSets should not have rule nodes");
+ aFont->mFont.featureValueLookup =
+ aPresContext->StyleSet()->AsGecko()->GetFontFeatureValuesLookup();
+
+ NS_ASSERTION(variantAlternatesValue->GetPairValue().mYValue.GetUnit() ==
+ eCSSUnit_List, "function list not a list value");
+ nsStyleUtil::ComputeFunctionalAlternates(
+ variantAlternatesValue->GetPairValue().mYValue.GetListValue(),
+ aFont->mFont.alternateValues);
+ }
+ break;
+
+ default:
+ break;
+ }
+
+ // font-variant-caps: normal, enum, inherit, initial, -moz-system-font
+ SetValue(*aRuleData->ValueForFontVariantCaps(),
+ aFont->mFont.variantCaps, aConditions,
+ SETVAL_ENUMERATED |SETVAL_UNSET_INHERIT,
+ aParentFont->mFont.variantCaps,
+ defaultVariableFont->variantCaps,
+ Unused, Unused, /* normal */ 0, systemFont.variantCaps);
+
+ // font-variant-east-asian: normal, enum (bit field), inherit, initial,
+ // -moz-system-font
+ SetValue(*aRuleData->ValueForFontVariantEastAsian(),
+ aFont->mFont.variantEastAsian, aConditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ aParentFont->mFont.variantEastAsian,
+ defaultVariableFont->variantEastAsian,
+ Unused, Unused, /* normal */ 0, systemFont.variantEastAsian);
+
+ // font-variant-ligatures: normal, none, enum (bit field), inherit, initial,
+ // -moz-system-font
+ SetValue(*aRuleData->ValueForFontVariantLigatures(),
+ aFont->mFont.variantLigatures, aConditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ aParentFont->mFont.variantLigatures,
+ defaultVariableFont->variantLigatures,
+ Unused, NS_FONT_VARIANT_LIGATURES_NONE, /* normal */ 0,
+ systemFont.variantLigatures);
+
+ // font-variant-numeric: normal, enum (bit field), inherit, initial,
+ // -moz-system-font
+ SetValue(*aRuleData->ValueForFontVariantNumeric(),
+ aFont->mFont.variantNumeric, aConditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ aParentFont->mFont.variantNumeric,
+ defaultVariableFont->variantNumeric,
+ Unused, Unused, /* normal */ 0, systemFont.variantNumeric);
+
+ // font-variant-position: normal, enum, inherit, initial,
+ // -moz-system-font
+ SetValue(*aRuleData->ValueForFontVariantPosition(),
+ aFont->mFont.variantPosition, aConditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ aParentFont->mFont.variantPosition,
+ defaultVariableFont->variantPosition,
+ Unused, Unused, /* normal */ 0, systemFont.variantPosition);
+
+ // font-feature-settings
+ const nsCSSValue* featureSettingsValue =
+ aRuleData->ValueForFontFeatureSettings();
+
+ switch (featureSettingsValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Normal:
+ case eCSSUnit_Initial:
+ aFont->mFont.fontFeatureSettings.Clear();
+ break;
+
+ case eCSSUnit_Inherit:
+ case eCSSUnit_Unset:
+ aConditions.SetUncacheable();
+ aFont->mFont.fontFeatureSettings = aParentFont->mFont.fontFeatureSettings;
+ break;
+
+ case eCSSUnit_System_Font:
+ aFont->mFont.fontFeatureSettings = systemFont.fontFeatureSettings;
+ break;
+
+ case eCSSUnit_PairList:
+ case eCSSUnit_PairListDep:
+ ComputeFontFeatures(featureSettingsValue->GetPairListValue(),
+ aFont->mFont.fontFeatureSettings);
+ break;
+
+ default:
+ MOZ_ASSERT(false, "unexpected value unit");
+ break;
+ }
+
+ // font-language-override
+ const nsCSSValue* languageOverrideValue =
+ aRuleData->ValueForFontLanguageOverride();
+ if (eCSSUnit_Inherit == languageOverrideValue->GetUnit() ||
+ eCSSUnit_Unset == languageOverrideValue->GetUnit()) {
+ aConditions.SetUncacheable();
+ aFont->mFont.languageOverride = aParentFont->mFont.languageOverride;
+ } else if (eCSSUnit_Normal == languageOverrideValue->GetUnit() ||
+ eCSSUnit_Initial == languageOverrideValue->GetUnit()) {
+ aFont->mFont.languageOverride.Truncate();
+ } else if (eCSSUnit_System_Font == languageOverrideValue->GetUnit()) {
+ aFont->mFont.languageOverride = systemFont.languageOverride;
+ } else if (eCSSUnit_String == languageOverrideValue->GetUnit()) {
+ languageOverrideValue->GetStringValue(aFont->mFont.languageOverride);
+ }
+
+ // -moz-min-font-size-ratio: percent, inherit
+ const nsCSSValue* minFontSizeRatio = aRuleData->ValueForMinFontSizeRatio();
+ switch (minFontSizeRatio->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_Unset:
+ case eCSSUnit_Inherit:
+ aFont->mMinFontSizeRatio = aParentFont->mMinFontSizeRatio;
+ aConditions.SetUncacheable();
+ break;
+ case eCSSUnit_Initial:
+ aFont->mMinFontSizeRatio = 100; // 100%
+ break;
+ case eCSSUnit_Percent: {
+ // While percentages are parsed as floating point numbers, we
+ // only store an integer in the range [0, 255] since that's all
+ // we need for now.
+ float percent = minFontSizeRatio->GetPercentValue() * 100;
+ if (percent < 0) {
+ percent = 0;
+ } else if (percent > 255) {
+ percent = 255;
+ }
+ aFont->mMinFontSizeRatio = uint8_t(percent);
+ break;
+ }
+ default:
+ MOZ_ASSERT_UNREACHABLE("Unknown unit for -moz-min-font-size-ratio");
+ }
+
+ nscoord scriptLevelAdjustedUnconstrainedParentSize;
+
+ // font-size: enum, length, percent, inherit
+ nscoord scriptLevelAdjustedParentSize =
+ ComputeScriptLevelSize(aFont, aParentFont, aPresContext,
+ &scriptLevelAdjustedUnconstrainedParentSize);
+ NS_ASSERTION(!aUsedStartStruct || aFont->mScriptUnconstrainedSize == aFont->mSize,
+ "If we have a start struct, we should have reset everything coming in here");
+
+ // Compute whether we're affected by scriptMinSize *before* calling
+ // SetFontSize, since aParentFont might be the same as aFont. If it
+ // is, calling SetFontSize might throw off our calculation.
+ bool affectedByScriptMinSize =
+ aParentFont->mSize != aParentFont->mScriptUnconstrainedSize ||
+ scriptLevelAdjustedParentSize !=
+ scriptLevelAdjustedUnconstrainedParentSize;
+
+ SetFontSize(aPresContext, aContext,
+ aRuleData, aFont, aParentFont,
+ &aFont->mSize,
+ systemFont, aParentFont->mSize, scriptLevelAdjustedParentSize,
+ aUsedStartStruct, atRoot, aConditions);
+ if (!aPresContext->Document()->GetMathMLEnabled()) {
+ MOZ_ASSERT(!affectedByScriptMinSize);
+ // If MathML is not enabled, we don't need to mark this node as
+ // uncacheable. If it becomes enabled, code in
+ // nsMathMLElementFactory will rebuild the rule tree and style data
+ // when MathML is first enabled (see nsMathMLElement::BindToTree).
+ aFont->mScriptUnconstrainedSize = aFont->mSize;
+ } else if (!affectedByScriptMinSize) {
+ // Fast path: we have not been affected by scriptminsize so we don't
+ // need to call SetFontSize again to compute the
+ // scriptminsize-unconstrained size. This is OK even if we have a
+ // start struct, because if we have a start struct then 'font-size'
+ // was specified and so scriptminsize has no effect.
+ aFont->mScriptUnconstrainedSize = aFont->mSize;
+ // It's possible we could, in the future, have a different parent,
+ // which would lead to a different affectedByScriptMinSize.
+ aConditions.SetUncacheable();
+ } else {
+ // see previous else-if
+ aConditions.SetUncacheable();
+
+ // Use a separate conditions object because it might get a
+ // *different* font-size dependency. We can ignore it because we've
+ // already called SetUncacheable.
+ RuleNodeCacheConditions unconstrainedConditions;
+
+ SetFontSize(aPresContext, aContext,
+ aRuleData, aFont, aParentFont,
+ &aFont->mScriptUnconstrainedSize,
+ systemFont, aParentFont->mScriptUnconstrainedSize,
+ scriptLevelAdjustedUnconstrainedParentSize,
+ aUsedStartStruct, atRoot, unconstrainedConditions);
+ }
+ NS_ASSERTION(aFont->mScriptUnconstrainedSize <= aFont->mSize,
+ "scriptminsize should never be making things bigger");
+
+ nscoord fontSize = aFont->mSize;
+
+ // enforce the user' specified minimum font-size on the value that we expose
+ // (but don't change font-size:0, since that would unhide hidden text)
+ if (fontSize > 0) {
+ nscoord minFontSize = aPresContext->MinFontSize(aFont->mLanguage);
+ if (minFontSize < 0) {
+ minFontSize = 0;
+ } else {
+ minFontSize = (minFontSize * aFont->mMinFontSizeRatio) / 100;
+ }
+ if (fontSize < minFontSize && !aPresContext->IsChrome()) {
+ // override the minimum font-size constraint
+ fontSize = minFontSize;
+ }
+ }
+ aFont->mFont.size = fontSize;
+
+ // font-size-adjust: number, none, inherit, initial, -moz-system-font
+ const nsCSSValue* sizeAdjustValue = aRuleData->ValueForFontSizeAdjust();
+ if (eCSSUnit_System_Font == sizeAdjustValue->GetUnit()) {
+ aFont->mFont.sizeAdjust = systemFont.sizeAdjust;
+ } else
+ SetFactor(*sizeAdjustValue, aFont->mFont.sizeAdjust,
+ aConditions, aParentFont->mFont.sizeAdjust, -1.0f,
+ SETFCT_NONE | SETFCT_UNSET_INHERIT);
+}
+
+/* static */ void
+nsRuleNode::ComputeFontFeatures(const nsCSSValuePairList *aFeaturesList,
+ nsTArray<gfxFontFeature>& aFeatureSettings)
+{
+ aFeatureSettings.Clear();
+ for (const nsCSSValuePairList* p = aFeaturesList; p; p = p->mNext) {
+ gfxFontFeature feat = {0, 0};
+
+ MOZ_ASSERT(aFeaturesList->mXValue.GetUnit() == eCSSUnit_String,
+ "unexpected value unit");
+
+ // tag is a 4-byte ASCII sequence
+ nsAutoString tag;
+ p->mXValue.GetStringValue(tag);
+ if (tag.Length() != 4) {
+ continue;
+ }
+ // parsing validates that these are ASCII chars
+ // tags are always big-endian
+ feat.mTag = (tag[0] << 24) | (tag[1] << 16) | (tag[2] << 8) | tag[3];
+
+ // value
+ NS_ASSERTION(p->mYValue.GetUnit() == eCSSUnit_Integer,
+ "should have found an integer unit");
+ feat.mValue = p->mYValue.GetIntValue();
+
+ aFeatureSettings.AppendElement(feat);
+ }
+}
+
+// This should die (bug 380915).
+//
+// SetGenericFont:
+// - backtrack to an ancestor with the same generic font name (possibly
+// up to the root where default values come from the presentation context)
+// - re-apply cascading rules from there without caching intermediate values
+/* static */ void
+nsRuleNode::SetGenericFont(nsPresContext* aPresContext,
+ nsStyleContext* aContext,
+ uint8_t aGenericFontID,
+ nsStyleFont* aFont)
+{
+ // walk up the contexts until a context with the desired generic font
+ AutoTArray<nsStyleContext*, 8> contextPath;
+ contextPath.AppendElement(aContext);
+ nsStyleContext* higherContext = aContext->GetParent();
+ while (higherContext) {
+ if (higherContext->StyleFont()->mGenericID == aGenericFontID) {
+ // done walking up the higher contexts
+ break;
+ }
+ contextPath.AppendElement(higherContext);
+ higherContext = higherContext->GetParent();
+ }
+
+ // re-apply the cascading rules, starting from the higher context
+
+ // If we stopped earlier because we reached the root of the style tree,
+ // we will start with the default generic font from the presentation
+ // context. Otherwise we start with the higher context.
+ const nsFont* defaultFont =
+ aPresContext->GetDefaultFont(aGenericFontID, aFont->mLanguage);
+ nsStyleFont parentFont(*defaultFont, aPresContext);
+ if (higherContext) {
+ const nsStyleFont* tmpFont = higherContext->StyleFont();
+ parentFont = *tmpFont;
+ }
+ *aFont = parentFont;
+
+ uint32_t fontBit = nsCachedStyleData::GetBitForSID(eStyleStruct_Font);
+
+ // use placement new[] on the result of alloca() to allocate a
+ // variable-sized stack array, including execution of constructors,
+ // and use an RAII class to run the destructors too.
+ size_t nprops = nsCSSProps::PropertyCountInStruct(eStyleStruct_Font);
+ void* dataStorage = alloca(nprops * sizeof(nsCSSValue));
+
+ for (int32_t i = contextPath.Length() - 1; i >= 0; --i) {
+ nsStyleContext* context = contextPath[i];
+ AutoCSSValueArray dataArray(dataStorage, nprops);
+
+ nsRuleData ruleData(NS_STYLE_INHERIT_BIT(Font), dataArray.get(),
+ aPresContext, context);
+ ruleData.mValueOffsets[eStyleStruct_Font] = 0;
+
+ // Trimmed down version of ::WalkRuleTree() to re-apply the style rules
+ // Note that we *do* need to do this for our own data, since what is
+ // in |fontData| in ComputeFontData is only for the rules below
+ // aStartStruct.
+ for (nsRuleNode* ruleNode = context->RuleNode(); ruleNode;
+ ruleNode = ruleNode->GetParent()) {
+ if (ruleNode->mNoneBits & fontBit)
+ // no more font rules on this branch, get out
+ break;
+
+ nsIStyleRule *rule = ruleNode->GetRule();
+ if (rule) {
+ ruleData.mLevel = ruleNode->GetLevel();
+ ruleData.mIsImportantRule = ruleNode->IsImportantRule();
+ rule->MapRuleInfoInto(&ruleData);
+ }
+ }
+
+ // Compute the delta from the information that the rules specified
+
+ // Avoid unnecessary operations in SetFont(). But we care if it's
+ // the final value that we're computing.
+ if (i != 0)
+ ruleData.ValueForFontFamily()->Reset();
+
+ ResolveVariableReferences(eStyleStruct_Font, &ruleData, aContext);
+
+ RuleNodeCacheConditions dummy;
+ nsRuleNode::SetFont(aPresContext, context,
+ aGenericFontID, &ruleData, &parentFont, aFont,
+ false, dummy);
+
+ parentFont = *aFont;
+ }
+
+ if (higherContext && contextPath.Length() > 1) {
+ // contextPath is a list of all ancestor style contexts, so it must have
+ // at least two elements for it to result in a dependency on grandancestor
+ // styles.
+ PropagateGrandancestorBit(aContext, higherContext);
+ }
+}
+
+const void*
+nsRuleNode::ComputeFontData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_INHERITED(Font, font, parentFont)
+
+ // NOTE: The |aRuleDetail| passed in is a little bit conservative due
+ // to the -moz-system-font property. We really don't need to consider
+ // it here in determining whether to cache in the rule tree. However,
+ // we do need to consider it in WalkRuleTree when deciding whether to
+ // walk further up the tree. So this means that when the font struct
+ // is fully specified using *longhand* properties (excluding
+ // -moz-system-font), we won't cache in the rule tree even though we
+ // could. However, it's pretty unlikely authors will do that
+ // (although there is a pretty good chance they'll fully specify it
+ // using the 'font' shorthand).
+
+ // Figure out if we are a generic font
+ uint8_t generic = kGenericFont_NONE;
+ // XXXldb What if we would have had a string if we hadn't been doing
+ // the optimization with a non-null aStartStruct?
+ const nsCSSValue* familyValue = aRuleData->ValueForFontFamily();
+ if (eCSSUnit_FontFamilyList == familyValue->GetUnit()) {
+ const FontFamilyList* fontlist = familyValue->GetFontFamilyListValue();
+ FontFamilyList& fl = font->mFont.fontlist;
+ fl = *fontlist;
+
+ // extract the first generic in the fontlist, if exists
+ FontFamilyType fontType = fontlist->FirstGeneric();
+
+ // if only a single generic, set the generic type
+ if (fontlist->Length() == 1) {
+ switch (fontType) {
+ case eFamily_serif:
+ generic = kGenericFont_serif;
+ break;
+ case eFamily_sans_serif:
+ generic = kGenericFont_sans_serif;
+ break;
+ case eFamily_monospace:
+ generic = kGenericFont_monospace;
+ break;
+ case eFamily_cursive:
+ generic = kGenericFont_cursive;
+ break;
+ case eFamily_fantasy:
+ generic = kGenericFont_fantasy;
+ break;
+ case eFamily_moz_fixed:
+ generic = kGenericFont_moz_fixed;
+ break;
+ default:
+ break;
+ }
+ }
+ }
+
+ // Now compute our font struct
+ if (generic == kGenericFont_NONE) {
+ // continue the normal processing
+ nsRuleNode::SetFont(mPresContext, aContext, generic,
+ aRuleData, parentFont, font,
+ aStartStruct != nullptr, conditions);
+ }
+ else {
+ // re-calculate the font as a generic font
+ conditions.SetUncacheable();
+ nsRuleNode::SetGenericFont(mPresContext, aContext, generic,
+ font);
+ }
+
+ COMPUTE_END_INHERITED(Font, font)
+}
+
+template <typename T>
+inline uint32_t ListLength(const T* aList)
+{
+ uint32_t len = 0;
+ while (aList) {
+ len++;
+ aList = aList->mNext;
+ }
+ return len;
+}
+
+static already_AddRefed<nsCSSShadowArray>
+GetShadowData(const nsCSSValueList* aList,
+ nsStyleContext* aContext,
+ bool aIsBoxShadow,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+{
+ uint32_t arrayLength = ListLength(aList);
+
+ MOZ_ASSERT(arrayLength > 0,
+ "Non-null text-shadow list, yet we counted 0 items.");
+ RefPtr<nsCSSShadowArray> shadowList =
+ new(arrayLength) nsCSSShadowArray(arrayLength);
+
+ if (!shadowList)
+ return nullptr;
+
+ nsStyleCoord tempCoord;
+ DebugOnly<bool> unitOK;
+ for (nsCSSShadowItem* item = shadowList->ShadowAt(0);
+ aList;
+ aList = aList->mNext, ++item) {
+ MOZ_ASSERT(aList->mValue.GetUnit() == eCSSUnit_Array,
+ "expecting a plain array value");
+ nsCSSValue::Array *arr = aList->mValue.GetArrayValue();
+ // OK to pass bad aParentCoord since we're not passing SETCOORD_INHERIT
+ unitOK = SetCoord(arr->Item(0), tempCoord, nsStyleCoord(),
+ SETCOORD_LENGTH | SETCOORD_CALC_LENGTH_ONLY,
+ aContext, aPresContext, aConditions);
+ NS_ASSERTION(unitOK, "unexpected unit");
+ item->mXOffset = tempCoord.GetCoordValue();
+
+ unitOK = SetCoord(arr->Item(1), tempCoord, nsStyleCoord(),
+ SETCOORD_LENGTH | SETCOORD_CALC_LENGTH_ONLY,
+ aContext, aPresContext, aConditions);
+ NS_ASSERTION(unitOK, "unexpected unit");
+ item->mYOffset = tempCoord.GetCoordValue();
+
+ // Blur radius is optional in the current box-shadow spec
+ if (arr->Item(2).GetUnit() != eCSSUnit_Null) {
+ unitOK = SetCoord(arr->Item(2), tempCoord, nsStyleCoord(),
+ SETCOORD_LENGTH | SETCOORD_CALC_LENGTH_ONLY |
+ SETCOORD_CALC_CLAMP_NONNEGATIVE,
+ aContext, aPresContext, aConditions);
+ NS_ASSERTION(unitOK, "unexpected unit");
+ item->mRadius = tempCoord.GetCoordValue();
+ } else {
+ item->mRadius = 0;
+ }
+
+ // Find the spread radius
+ if (aIsBoxShadow && arr->Item(3).GetUnit() != eCSSUnit_Null) {
+ unitOK = SetCoord(arr->Item(3), tempCoord, nsStyleCoord(),
+ SETCOORD_LENGTH | SETCOORD_CALC_LENGTH_ONLY,
+ aContext, aPresContext, aConditions);
+ NS_ASSERTION(unitOK, "unexpected unit");
+ item->mSpread = tempCoord.GetCoordValue();
+ } else {
+ item->mSpread = 0;
+ }
+
+ if (arr->Item(4).GetUnit() != eCSSUnit_Null) {
+ item->mHasColor = true;
+ // 2nd argument can be bogus since inherit is not a valid color
+ unitOK = SetColor(arr->Item(4), 0, aPresContext, aContext, item->mColor,
+ aConditions);
+ NS_ASSERTION(unitOK, "unexpected unit");
+ }
+
+ if (aIsBoxShadow && arr->Item(5).GetUnit() == eCSSUnit_Enumerated) {
+ NS_ASSERTION(arr->Item(5).GetIntValue()
+ == uint8_t(StyleBoxShadowType::Inset),
+ "invalid keyword type for box shadow");
+ item->mInset = true;
+ } else {
+ item->mInset = false;
+ }
+ }
+
+ return shadowList.forget();
+}
+
+struct TextEmphasisChars
+{
+ const char16_t* mFilled;
+ const char16_t* mOpen;
+};
+
+#define TEXT_EMPHASIS_CHARS_LIST() \
+ TEXT_EMPHASIS_CHARS_ITEM(u"", u"", NONE) \
+ TEXT_EMPHASIS_CHARS_ITEM(u"\u2022", u"\u25e6", DOT) \
+ TEXT_EMPHASIS_CHARS_ITEM(u"\u25cf", u"\u25cb", CIRCLE) \
+ TEXT_EMPHASIS_CHARS_ITEM(u"\u25c9", u"\u25ce", DOUBLE_CIRCLE) \
+ TEXT_EMPHASIS_CHARS_ITEM(u"\u25b2", u"\u25b3", TRIANGLE) \
+ TEXT_EMPHASIS_CHARS_ITEM(u"\ufe45", u"\ufe46", SESAME)
+
+static constexpr TextEmphasisChars kTextEmphasisChars[] =
+{
+#define TEXT_EMPHASIS_CHARS_ITEM(filled_, open_, type_) \
+ { filled_, open_ }, // type_
+ TEXT_EMPHASIS_CHARS_LIST()
+#undef TEXT_EMPHASIS_CHARS_ITEM
+};
+
+#define TEXT_EMPHASIS_CHARS_ITEM(filled_, open_, type_) \
+ static_assert(ArrayLength(filled_) <= 2 && \
+ ArrayLength(open_) <= 2, \
+ "emphasis marks should have no more than one char"); \
+ static_assert( \
+ *kTextEmphasisChars[NS_STYLE_TEXT_EMPHASIS_STYLE_##type_].mFilled == \
+ *filled_, "filled " #type_ " should be " #filled_); \
+ static_assert( \
+ *kTextEmphasisChars[NS_STYLE_TEXT_EMPHASIS_STYLE_##type_].mOpen == \
+ *open_, "open " #type_ " should be " #open_);
+TEXT_EMPHASIS_CHARS_LIST()
+#undef TEXT_EMPHASIS_CHARS_ITEM
+
+#undef TEXT_EMPHASIS_CHARS_LIST
+
+static void
+TruncateStringToSingleGrapheme(nsAString& aStr)
+{
+ unicode::ClusterIterator iter(aStr.Data(), aStr.Length());
+ if (!iter.AtEnd()) {
+ iter.Next();
+ if (!iter.AtEnd()) {
+ // Not mutating the string for common cases helps memory use
+ // since we share the buffer from the specified style into the
+ // computed style.
+ aStr.Truncate(iter - aStr.Data());
+ }
+ }
+}
+
+struct LineHeightCalcObj
+{
+ float mLineHeight;
+ bool mIsNumber;
+};
+
+struct SetLineHeightCalcOps : public css::NumbersAlreadyNormalizedOps
+{
+ typedef LineHeightCalcObj result_type;
+ nsStyleContext* const mStyleContext;
+ nsPresContext* const mPresContext;
+ RuleNodeCacheConditions& mConditions;
+
+ SetLineHeightCalcOps(nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+ : mStyleContext(aStyleContext),
+ mPresContext(aPresContext),
+ mConditions(aConditions)
+ {
+ }
+
+ result_type
+ MergeAdditive(nsCSSUnit aCalcFunction,
+ result_type aValue1, result_type aValue2)
+ {
+ MOZ_ASSERT(aValue1.mIsNumber == aValue2.mIsNumber);
+
+ LineHeightCalcObj result;
+ result.mIsNumber = aValue1.mIsNumber;
+ if (aCalcFunction == eCSSUnit_Calc_Plus) {
+ result.mLineHeight = aValue1.mLineHeight + aValue2.mLineHeight;
+ return result;
+ }
+ MOZ_ASSERT(aCalcFunction == eCSSUnit_Calc_Minus,
+ "unexpected unit");
+ result.mLineHeight = aValue1.mLineHeight - aValue2.mLineHeight;
+ return result;
+ }
+
+ result_type
+ MergeMultiplicativeL(nsCSSUnit aCalcFunction,
+ float aValue1, result_type aValue2)
+ {
+ MOZ_ASSERT(aCalcFunction == eCSSUnit_Calc_Times_L,
+ "unexpected unit");
+ LineHeightCalcObj result;
+ result.mIsNumber = aValue2.mIsNumber;
+ result.mLineHeight = aValue1 * aValue2.mLineHeight;
+ return result;
+ }
+
+ result_type
+ MergeMultiplicativeR(nsCSSUnit aCalcFunction,
+ result_type aValue1, float aValue2)
+ {
+ LineHeightCalcObj result;
+ result.mIsNumber = aValue1.mIsNumber;
+ if (aCalcFunction == eCSSUnit_Calc_Times_R) {
+ result.mLineHeight = aValue1.mLineHeight * aValue2;
+ return result;
+ }
+ MOZ_ASSERT(aCalcFunction == eCSSUnit_Calc_Divided,
+ "unexpected unit");
+ result.mLineHeight = aValue1.mLineHeight / aValue2;
+ return result;
+ }
+
+ result_type ComputeLeafValue(const nsCSSValue& aValue)
+ {
+ LineHeightCalcObj result;
+ if (aValue.IsLengthUnit()) {
+ result.mIsNumber = false;
+ result.mLineHeight = CalcLength(aValue, mStyleContext,
+ mPresContext, mConditions);
+ }
+ else if (eCSSUnit_Percent == aValue.GetUnit()) {
+ mConditions.SetUncacheable();
+ result.mIsNumber = false;
+ nscoord fontSize = mStyleContext->StyleFont()->mFont.size;
+ result.mLineHeight = fontSize * aValue.GetPercentValue();
+ }
+ else if (eCSSUnit_Number == aValue.GetUnit()) {
+ result.mIsNumber = true;
+ result.mLineHeight = aValue.GetFloatValue();
+ } else {
+ MOZ_ASSERT(false, "unexpected value");
+ result.mIsNumber = true;
+ result.mLineHeight = 1.0f;
+ }
+
+ return result;
+ }
+};
+
+const void*
+nsRuleNode::ComputeTextData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_INHERITED(Text, text, parentText)
+
+ auto setComplexColor = [&](const nsCSSValue* aValue,
+ StyleComplexColor nsStyleText::* aField) {
+ SetComplexColor<eUnsetInherit>(*aValue, parentText->*aField,
+ StyleComplexColor::CurrentColor(),
+ mPresContext, text->*aField, conditions);
+ };
+
+ // tab-size: integer, inherit
+ SetValue(*aRuleData->ValueForTabSize(),
+ text->mTabSize, conditions,
+ SETVAL_INTEGER | SETVAL_UNSET_INHERIT, parentText->mTabSize,
+ NS_STYLE_TABSIZE_INITIAL);
+
+ // letter-spacing: normal, length, inherit
+ SetCoord(*aRuleData->ValueForLetterSpacing(),
+ text->mLetterSpacing, parentText->mLetterSpacing,
+ SETCOORD_LH | SETCOORD_NORMAL | SETCOORD_INITIAL_NORMAL |
+ SETCOORD_CALC_LENGTH_ONLY | SETCOORD_UNSET_INHERIT,
+ aContext, mPresContext, conditions);
+
+ // text-shadow: none, list, inherit, initial
+ const nsCSSValue* textShadowValue = aRuleData->ValueForTextShadow();
+ if (textShadowValue->GetUnit() != eCSSUnit_Null) {
+ text->mTextShadow = nullptr;
+
+ // Don't need to handle none/initial explicitly: The above assignment
+ // takes care of that
+ if (textShadowValue->GetUnit() == eCSSUnit_Inherit ||
+ textShadowValue->GetUnit() == eCSSUnit_Unset) {
+ conditions.SetUncacheable();
+ text->mTextShadow = parentText->mTextShadow;
+ } else if (textShadowValue->GetUnit() == eCSSUnit_List ||
+ textShadowValue->GetUnit() == eCSSUnit_ListDep) {
+ // List of arrays
+ text->mTextShadow = GetShadowData(textShadowValue->GetListValue(),
+ aContext, false, mPresContext, conditions);
+ }
+ }
+
+ // line-height: normal, number, length, percent, calc, inherit
+ const nsCSSValue* lineHeightValue = aRuleData->ValueForLineHeight();
+ if (eCSSUnit_Percent == lineHeightValue->GetUnit()) {
+ conditions.SetUncacheable();
+ // Use |mFont.size| to pick up minimum font size.
+ text->mLineHeight.SetCoordValue(
+ NSToCoordRound(float(aContext->StyleFont()->mFont.size) *
+ lineHeightValue->GetPercentValue()));
+ }
+ else if (eCSSUnit_Initial == lineHeightValue->GetUnit() ||
+ eCSSUnit_System_Font == lineHeightValue->GetUnit()) {
+ text->mLineHeight.SetNormalValue();
+ }
+ else if (eCSSUnit_Calc == lineHeightValue->GetUnit()) {
+ SetLineHeightCalcOps ops(aContext, mPresContext, conditions);
+ LineHeightCalcObj obj = css::ComputeCalc(*lineHeightValue, ops);
+ if (obj.mIsNumber) {
+ text->mLineHeight.SetFactorValue(obj.mLineHeight);
+ } else {
+ text->mLineHeight.SetCoordValue(
+ NSToCoordRoundWithClamp(obj.mLineHeight));
+ }
+ }
+ else {
+ SetCoord(*lineHeightValue, text->mLineHeight, parentText->mLineHeight,
+ SETCOORD_LEH | SETCOORD_FACTOR | SETCOORD_NORMAL |
+ SETCOORD_UNSET_INHERIT,
+ aContext, mPresContext, conditions);
+ if (lineHeightValue->IsLengthUnit() &&
+ !lineHeightValue->IsRelativeLengthUnit()) {
+ nscoord lh = nsStyleFont::ZoomText(mPresContext,
+ text->mLineHeight.GetCoordValue());
+
+ conditions.SetUncacheable();
+ const nsStyleFont *font = aContext->StyleFont();
+ nscoord minimumFontSize = mPresContext->MinFontSize(font->mLanguage);
+
+ if (minimumFontSize > 0 && !mPresContext->IsChrome()) {
+ if (font->mSize != 0) {
+ lh = nscoord(float(lh) * float(font->mFont.size) / float(font->mSize));
+ } else {
+ lh = minimumFontSize;
+ }
+ }
+ text->mLineHeight.SetCoordValue(lh);
+ }
+ }
+
+
+ // text-align: enum, string, pair(enum|string), inherit, initial
+ // NOTE: string is not implemented yet.
+ const nsCSSValue* textAlignValue = aRuleData->ValueForTextAlign();
+ text->mTextAlignTrue = false;
+ if (eCSSUnit_String == textAlignValue->GetUnit()) {
+ NS_NOTYETIMPLEMENTED("align string");
+ } else if (eCSSUnit_Enumerated == textAlignValue->GetUnit() &&
+ NS_STYLE_TEXT_ALIGN_MOZ_CENTER_OR_INHERIT ==
+ textAlignValue->GetIntValue()) {
+ conditions.SetUncacheable();
+ uint8_t parentAlign = parentText->mTextAlign;
+ text->mTextAlign = (NS_STYLE_TEXT_ALIGN_START == parentAlign) ?
+ NS_STYLE_TEXT_ALIGN_CENTER : parentAlign;
+ } else if (eCSSUnit_Enumerated == textAlignValue->GetUnit() &&
+ NS_STYLE_TEXT_ALIGN_MATCH_PARENT ==
+ textAlignValue->GetIntValue()) {
+ conditions.SetUncacheable();
+ nsStyleContext* parent = aContext->GetParent();
+ if (parent) {
+ uint8_t parentAlign = parentText->mTextAlign;
+ uint8_t parentDirection = parent->StyleVisibility()->mDirection;
+ switch (parentAlign) {
+ case NS_STYLE_TEXT_ALIGN_START:
+ text->mTextAlign = parentDirection == NS_STYLE_DIRECTION_RTL ?
+ NS_STYLE_TEXT_ALIGN_RIGHT : NS_STYLE_TEXT_ALIGN_LEFT;
+ break;
+
+ case NS_STYLE_TEXT_ALIGN_END:
+ text->mTextAlign = parentDirection == NS_STYLE_DIRECTION_RTL ?
+ NS_STYLE_TEXT_ALIGN_LEFT : NS_STYLE_TEXT_ALIGN_RIGHT;
+ break;
+
+ default:
+ text->mTextAlign = parentAlign;
+ }
+ }
+ } else {
+ if (eCSSUnit_Pair == textAlignValue->GetUnit()) {
+ // Two values were specified, one must be 'true'.
+ text->mTextAlignTrue = true;
+ const nsCSSValuePair& textAlignValuePair = textAlignValue->GetPairValue();
+ textAlignValue = &textAlignValuePair.mXValue;
+ if (eCSSUnit_Enumerated == textAlignValue->GetUnit()) {
+ if (textAlignValue->GetIntValue() == NS_STYLE_TEXT_ALIGN_UNSAFE) {
+ textAlignValue = &textAlignValuePair.mYValue;
+ }
+ } else if (eCSSUnit_String == textAlignValue->GetUnit()) {
+ NS_NOTYETIMPLEMENTED("align string");
+ }
+ } else if (eCSSUnit_Inherit == textAlignValue->GetUnit() ||
+ eCSSUnit_Unset == textAlignValue->GetUnit()) {
+ text->mTextAlignTrue = parentText->mTextAlignTrue;
+ }
+ SetValue(*textAlignValue, text->mTextAlign, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mTextAlign,
+ NS_STYLE_TEXT_ALIGN_START);
+ }
+
+ // text-align-last: enum, pair(enum), inherit, initial
+ const nsCSSValue* textAlignLastValue = aRuleData->ValueForTextAlignLast();
+ text->mTextAlignLastTrue = false;
+ if (eCSSUnit_Pair == textAlignLastValue->GetUnit()) {
+ // Two values were specified, one must be 'true'.
+ text->mTextAlignLastTrue = true;
+ const nsCSSValuePair& textAlignLastValuePair = textAlignLastValue->GetPairValue();
+ textAlignLastValue = &textAlignLastValuePair.mXValue;
+ if (eCSSUnit_Enumerated == textAlignLastValue->GetUnit()) {
+ if (textAlignLastValue->GetIntValue() == NS_STYLE_TEXT_ALIGN_UNSAFE) {
+ textAlignLastValue = &textAlignLastValuePair.mYValue;
+ }
+ }
+ } else if (eCSSUnit_Inherit == textAlignLastValue->GetUnit() ||
+ eCSSUnit_Unset == textAlignLastValue->GetUnit()) {
+ text->mTextAlignLastTrue = parentText->mTextAlignLastTrue;
+ }
+ SetValue(*textAlignLastValue, text->mTextAlignLast,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mTextAlignLast,
+ NS_STYLE_TEXT_ALIGN_AUTO);
+
+ // text-indent: length, percent, calc, inherit, initial
+ SetCoord(*aRuleData->ValueForTextIndent(), text->mTextIndent, parentText->mTextIndent,
+ SETCOORD_LPH | SETCOORD_INITIAL_ZERO | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INHERIT,
+ aContext, mPresContext, conditions);
+
+ // text-transform: enum, inherit, initial
+ SetValue(*aRuleData->ValueForTextTransform(), text->mTextTransform, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mTextTransform,
+ NS_STYLE_TEXT_TRANSFORM_NONE);
+
+ // white-space: enum, inherit, initial
+ SetValue(*aRuleData->ValueForWhiteSpace(), text->mWhiteSpace, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mWhiteSpace,
+ NS_STYLE_WHITESPACE_NORMAL);
+
+ // word-break: enum, inherit, initial
+ SetValue(*aRuleData->ValueForWordBreak(), text->mWordBreak, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mWordBreak,
+ NS_STYLE_WORDBREAK_NORMAL);
+
+ // word-spacing: normal, length, percent, inherit
+ const nsCSSValue* wordSpacingValue = aRuleData->ValueForWordSpacing();
+ if (wordSpacingValue->GetUnit() == eCSSUnit_Normal) {
+ // Do this so that "normal" computes to 0px, as the CSS 2.1 spec requires.
+ text->mWordSpacing.SetCoordValue(0);
+ } else {
+ SetCoord(*aRuleData->ValueForWordSpacing(),
+ text->mWordSpacing, parentText->mWordSpacing,
+ SETCOORD_LPH | SETCOORD_INITIAL_ZERO |
+ SETCOORD_STORE_CALC | SETCOORD_UNSET_INHERIT,
+ aContext, mPresContext, conditions);
+ }
+
+ // overflow-wrap: enum, inherit, initial
+ SetValue(*aRuleData->ValueForOverflowWrap(), text->mOverflowWrap, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mOverflowWrap,
+ NS_STYLE_OVERFLOWWRAP_NORMAL);
+
+ // hyphens: enum, inherit, initial
+ SetValue(*aRuleData->ValueForHyphens(), text->mHyphens, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mHyphens,
+ NS_STYLE_HYPHENS_MANUAL);
+
+ // ruby-align: enum, inherit, initial
+ SetValue(*aRuleData->ValueForRubyAlign(),
+ text->mRubyAlign, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mRubyAlign,
+ NS_STYLE_RUBY_ALIGN_SPACE_AROUND);
+
+ // ruby-position: enum, inherit, initial
+ SetValue(*aRuleData->ValueForRubyPosition(),
+ text->mRubyPosition, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mRubyPosition,
+ NS_STYLE_RUBY_POSITION_OVER);
+
+ // text-size-adjust: none, auto, inherit, initial
+ SetValue(*aRuleData->ValueForTextSizeAdjust(), text->mTextSizeAdjust,
+ conditions, SETVAL_UNSET_INHERIT,
+ parentText->mTextSizeAdjust,
+ /* initial */ NS_STYLE_TEXT_SIZE_ADJUST_AUTO,
+ /* auto */ NS_STYLE_TEXT_SIZE_ADJUST_AUTO,
+ /* none */ NS_STYLE_TEXT_SIZE_ADJUST_NONE, Unused, Unused);
+
+ // text-combine-upright: enum, inherit, initial
+ SetValue(*aRuleData->ValueForTextCombineUpright(),
+ text->mTextCombineUpright,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mTextCombineUpright,
+ NS_STYLE_TEXT_COMBINE_UPRIGHT_NONE);
+
+ // text-emphasis-color: color, string, inherit, initial
+ setComplexColor(aRuleData->ValueForTextEmphasisColor(),
+ &nsStyleText::mTextEmphasisColor);
+
+ // text-emphasis-position: enum, inherit, initial
+ SetValue(*aRuleData->ValueForTextEmphasisPosition(),
+ text->mTextEmphasisPosition,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mTextEmphasisPosition,
+ NS_STYLE_TEXT_EMPHASIS_POSITION_OVER |
+ NS_STYLE_TEXT_EMPHASIS_POSITION_RIGHT);
+
+ // text-emphasis-style: string, enum, inherit, initial
+ const nsCSSValue* textEmphasisStyleValue =
+ aRuleData->ValueForTextEmphasisStyle();
+ switch (textEmphasisStyleValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_Initial:
+ case eCSSUnit_None: {
+ text->mTextEmphasisStyle = NS_STYLE_TEXT_EMPHASIS_STYLE_NONE;
+ text->mTextEmphasisStyleString = u"";
+ break;
+ }
+ case eCSSUnit_Inherit:
+ case eCSSUnit_Unset: {
+ conditions.SetUncacheable();
+ text->mTextEmphasisStyle = parentText->mTextEmphasisStyle;
+ text->mTextEmphasisStyleString = parentText->mTextEmphasisStyleString;
+ break;
+ }
+ case eCSSUnit_Enumerated: {
+ auto style = textEmphasisStyleValue->GetIntValue();
+ // If shape part is not specified, compute it according to the
+ // writing-mode. Note that, if the fill part (filled/open) is not
+ // specified, we compute it to filled per spec. Since that value
+ // is zero, no additional computation is needed. See the assertion
+ // in CSSParserImpl::ParseTextEmphasisStyle().
+ if (!(style & NS_STYLE_TEXT_EMPHASIS_STYLE_SHAPE_MASK)) {
+ conditions.SetUncacheable();
+ if (WritingMode(aContext).IsVertical()) {
+ style |= NS_STYLE_TEXT_EMPHASIS_STYLE_SESAME;
+ } else {
+ style |= NS_STYLE_TEXT_EMPHASIS_STYLE_CIRCLE;
+ }
+ }
+ text->mTextEmphasisStyle = style;
+ size_t shape = style & NS_STYLE_TEXT_EMPHASIS_STYLE_SHAPE_MASK;
+ MOZ_ASSERT(shape > 0 && shape < ArrayLength(kTextEmphasisChars));
+ const TextEmphasisChars& chars = kTextEmphasisChars[shape];
+ text->mTextEmphasisStyleString =
+ (style & NS_STYLE_TEXT_EMPHASIS_STYLE_FILL_MASK) ==
+ NS_STYLE_TEXT_EMPHASIS_STYLE_FILLED ? chars.mFilled : chars.mOpen;
+ break;
+ }
+ case eCSSUnit_String: {
+ text->mTextEmphasisStyle = NS_STYLE_TEXT_EMPHASIS_STYLE_STRING;
+ nsString strValue;
+ textEmphasisStyleValue->GetStringValue(strValue);
+ TruncateStringToSingleGrapheme(strValue);
+ text->mTextEmphasisStyleString = strValue;
+ break;
+ }
+ default:
+ MOZ_ASSERT_UNREACHABLE("Unknown value unit type");
+ }
+
+ // text-rendering: enum, inherit, initial
+ SetValue(*aRuleData->ValueForTextRendering(),
+ text->mTextRendering, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mTextRendering,
+ NS_STYLE_TEXT_RENDERING_AUTO);
+
+ // -webkit-text-fill-color: color, string, inherit, initial
+ setComplexColor(aRuleData->ValueForWebkitTextFillColor(),
+ &nsStyleText::mWebkitTextFillColor);
+
+ // -webkit-text-stroke-color: color, string, inherit, initial
+ setComplexColor(aRuleData->ValueForWebkitTextStrokeColor(),
+ &nsStyleText::mWebkitTextStrokeColor);
+
+ // -webkit-text-stroke-width: length, inherit, initial, enum
+ const nsCSSValue*
+ webkitTextStrokeWidthValue = aRuleData->ValueForWebkitTextStrokeWidth();
+ if (webkitTextStrokeWidthValue->GetUnit() == eCSSUnit_Enumerated) {
+ NS_ASSERTION(webkitTextStrokeWidthValue->GetIntValue() == NS_STYLE_BORDER_WIDTH_THIN ||
+ webkitTextStrokeWidthValue->GetIntValue() == NS_STYLE_BORDER_WIDTH_MEDIUM ||
+ webkitTextStrokeWidthValue->GetIntValue() == NS_STYLE_BORDER_WIDTH_THICK,
+ "Unexpected enum value");
+ text->mWebkitTextStrokeWidth.SetCoordValue(
+ mPresContext->GetBorderWidthTable()[webkitTextStrokeWidthValue->GetIntValue()]);
+ } else {
+ SetCoord(*webkitTextStrokeWidthValue, text->mWebkitTextStrokeWidth,
+ parentText->mWebkitTextStrokeWidth,
+ SETCOORD_LH | SETCOORD_CALC_LENGTH_ONLY |
+ SETCOORD_CALC_CLAMP_NONNEGATIVE |
+ SETCOORD_INITIAL_ZERO | SETCOORD_UNSET_INHERIT,
+ aContext, mPresContext, conditions);
+ }
+
+ // -moz-control-character-visibility: enum, inherit, initial
+ SetValue(*aRuleData->ValueForControlCharacterVisibility(),
+ text->mControlCharacterVisibility,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentText->mControlCharacterVisibility,
+ nsCSSParser::ControlCharVisibilityDefault());
+
+ COMPUTE_END_INHERITED(Text, text)
+}
+
+const void*
+nsRuleNode::ComputeTextResetData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(TextReset, text, parentText)
+
+ // text-decoration-line: enum (bit field), inherit, initial
+ const nsCSSValue* decorationLineValue =
+ aRuleData->ValueForTextDecorationLine();
+ if (eCSSUnit_Enumerated == decorationLineValue->GetUnit()) {
+ int32_t td = decorationLineValue->GetIntValue();
+ text->mTextDecorationLine = td;
+ if (td & NS_STYLE_TEXT_DECORATION_LINE_PREF_ANCHORS) {
+ bool underlineLinks =
+ mPresContext->GetCachedBoolPref(kPresContext_UnderlineLinks);
+ if (underlineLinks) {
+ text->mTextDecorationLine |= NS_STYLE_TEXT_DECORATION_LINE_UNDERLINE;
+ }
+ else {
+ text->mTextDecorationLine &= ~NS_STYLE_TEXT_DECORATION_LINE_UNDERLINE;
+ }
+ }
+ } else if (eCSSUnit_Inherit == decorationLineValue->GetUnit()) {
+ conditions.SetUncacheable();
+ text->mTextDecorationLine = parentText->mTextDecorationLine;
+ } else if (eCSSUnit_Initial == decorationLineValue->GetUnit() ||
+ eCSSUnit_Unset == decorationLineValue->GetUnit()) {
+ text->mTextDecorationLine = NS_STYLE_TEXT_DECORATION_LINE_NONE;
+ }
+
+ // text-decoration-color: color, string, enum, inherit, initial
+ SetComplexColor<eUnsetInitial>(*aRuleData->ValueForTextDecorationColor(),
+ parentText->mTextDecorationColor,
+ StyleComplexColor::CurrentColor(),
+ mPresContext,
+ text->mTextDecorationColor, conditions);
+
+ // text-decoration-style: enum, inherit, initial
+ const nsCSSValue* decorationStyleValue =
+ aRuleData->ValueForTextDecorationStyle();
+ if (eCSSUnit_Enumerated == decorationStyleValue->GetUnit()) {
+ text->mTextDecorationStyle = decorationStyleValue->GetIntValue();
+ } else if (eCSSUnit_Inherit == decorationStyleValue->GetUnit()) {
+ text->mTextDecorationStyle = parentText->mTextDecorationStyle;
+ conditions.SetUncacheable();
+ } else if (eCSSUnit_Initial == decorationStyleValue->GetUnit() ||
+ eCSSUnit_Unset == decorationStyleValue->GetUnit()) {
+ text->mTextDecorationStyle = NS_STYLE_TEXT_DECORATION_STYLE_SOLID;
+ }
+
+ // text-overflow: enum, string, pair(enum|string), inherit, initial
+ const nsCSSValue* textOverflowValue =
+ aRuleData->ValueForTextOverflow();
+ if (eCSSUnit_Initial == textOverflowValue->GetUnit() ||
+ eCSSUnit_Unset == textOverflowValue->GetUnit()) {
+ text->mTextOverflow = nsStyleTextOverflow();
+ } else if (eCSSUnit_Inherit == textOverflowValue->GetUnit()) {
+ conditions.SetUncacheable();
+ text->mTextOverflow = parentText->mTextOverflow;
+ } else if (eCSSUnit_Enumerated == textOverflowValue->GetUnit()) {
+ // A single enumerated value.
+ SetValue(*textOverflowValue, text->mTextOverflow.mRight.mType,
+ conditions,
+ SETVAL_ENUMERATED, parentText->mTextOverflow.mRight.mType,
+ NS_STYLE_TEXT_OVERFLOW_CLIP);
+ text->mTextOverflow.mRight.mString.Truncate();
+ text->mTextOverflow.mLeft.mType = NS_STYLE_TEXT_OVERFLOW_CLIP;
+ text->mTextOverflow.mLeft.mString.Truncate();
+ text->mTextOverflow.mLogicalDirections = true;
+ } else if (eCSSUnit_String == textOverflowValue->GetUnit()) {
+ // A single string value.
+ text->mTextOverflow.mRight.mType = NS_STYLE_TEXT_OVERFLOW_STRING;
+ textOverflowValue->GetStringValue(text->mTextOverflow.mRight.mString);
+ text->mTextOverflow.mLeft.mType = NS_STYLE_TEXT_OVERFLOW_CLIP;
+ text->mTextOverflow.mLeft.mString.Truncate();
+ text->mTextOverflow.mLogicalDirections = true;
+ } else if (eCSSUnit_Pair == textOverflowValue->GetUnit()) {
+ // Two values were specified.
+ text->mTextOverflow.mLogicalDirections = false;
+ const nsCSSValuePair& textOverflowValuePair =
+ textOverflowValue->GetPairValue();
+
+ const nsCSSValue *textOverflowLeftValue = &textOverflowValuePair.mXValue;
+ if (eCSSUnit_Enumerated == textOverflowLeftValue->GetUnit()) {
+ SetValue(*textOverflowLeftValue, text->mTextOverflow.mLeft.mType,
+ conditions,
+ SETVAL_ENUMERATED, parentText->mTextOverflow.mLeft.mType,
+ NS_STYLE_TEXT_OVERFLOW_CLIP);
+ text->mTextOverflow.mLeft.mString.Truncate();
+ } else if (eCSSUnit_String == textOverflowLeftValue->GetUnit()) {
+ textOverflowLeftValue->GetStringValue(text->mTextOverflow.mLeft.mString);
+ text->mTextOverflow.mLeft.mType = NS_STYLE_TEXT_OVERFLOW_STRING;
+ }
+
+ const nsCSSValue *textOverflowRightValue = &textOverflowValuePair.mYValue;
+ if (eCSSUnit_Enumerated == textOverflowRightValue->GetUnit()) {
+ SetValue(*textOverflowRightValue, text->mTextOverflow.mRight.mType,
+ conditions,
+ SETVAL_ENUMERATED, parentText->mTextOverflow.mRight.mType,
+ NS_STYLE_TEXT_OVERFLOW_CLIP);
+ text->mTextOverflow.mRight.mString.Truncate();
+ } else if (eCSSUnit_String == textOverflowRightValue->GetUnit()) {
+ textOverflowRightValue->GetStringValue(text->mTextOverflow.mRight.mString);
+ text->mTextOverflow.mRight.mType = NS_STYLE_TEXT_OVERFLOW_STRING;
+ }
+ }
+
+ // unicode-bidi: enum, inherit, initial
+ SetValue(*aRuleData->ValueForUnicodeBidi(), text->mUnicodeBidi, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentText->mUnicodeBidi,
+ NS_STYLE_UNICODE_BIDI_NORMAL);
+
+ // initial-letter: normal, number, array(number, integer?), initial
+ const nsCSSValue* initialLetterValue = aRuleData->ValueForInitialLetter();
+ if (initialLetterValue->GetUnit() == eCSSUnit_Null) {
+ // We don't want to change anything in this case.
+ } else if (initialLetterValue->GetUnit() == eCSSUnit_Inherit) {
+ conditions.SetUncacheable();
+ text->mInitialLetterSink = parentText->mInitialLetterSink;
+ text->mInitialLetterSize = parentText->mInitialLetterSize;
+ } else if (initialLetterValue->GetUnit() == eCSSUnit_Initial ||
+ initialLetterValue->GetUnit() == eCSSUnit_Unset ||
+ initialLetterValue->GetUnit() == eCSSUnit_Normal) {
+ // Use invalid values in initial-letter property to mean normal. So we can
+ // determine whether it is normal by checking mInitialLetterSink == 0.
+ text->mInitialLetterSink = 0;
+ text->mInitialLetterSize = 0.0f;
+ } else if (initialLetterValue->GetUnit() == eCSSUnit_Array) {
+ const nsCSSValue& firstValue = initialLetterValue->GetArrayValue()->Item(0);
+ const nsCSSValue& secondValue = initialLetterValue->GetArrayValue()->Item(1);
+ MOZ_ASSERT(firstValue.GetUnit() == eCSSUnit_Number &&
+ secondValue.GetUnit() == eCSSUnit_Integer,
+ "unexpected value unit");
+ text->mInitialLetterSize = firstValue.GetFloatValue();
+ text->mInitialLetterSink = secondValue.GetIntValue();
+ } else if (initialLetterValue->GetUnit() == eCSSUnit_Number) {
+ text->mInitialLetterSize = initialLetterValue->GetFloatValue();
+ text->mInitialLetterSink = NSToCoordFloorClamped(text->mInitialLetterSize);
+ } else {
+ MOZ_ASSERT_UNREACHABLE("unknown unit for initial-letter");
+ }
+
+ COMPUTE_END_RESET(TextReset, text)
+}
+
+const void*
+nsRuleNode::ComputeUserInterfaceData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_INHERITED(UserInterface, ui, parentUI)
+
+ // cursor: enum, url, inherit
+ const nsCSSValue* cursorValue = aRuleData->ValueForCursor();
+ nsCSSUnit cursorUnit = cursorValue->GetUnit();
+ if (cursorUnit != eCSSUnit_Null) {
+ ui->mCursorImages.Clear();
+
+ if (cursorUnit == eCSSUnit_Inherit ||
+ cursorUnit == eCSSUnit_Unset) {
+ conditions.SetUncacheable();
+ ui->mCursor = parentUI->mCursor;
+ ui->mCursorImages = parentUI->mCursorImages;
+ }
+ else if (cursorUnit == eCSSUnit_Initial) {
+ ui->mCursor = NS_STYLE_CURSOR_AUTO;
+ }
+ else {
+ // The parser will never create a list that is *all* URL values --
+ // that's invalid.
+ MOZ_ASSERT(cursorUnit == eCSSUnit_List || cursorUnit == eCSSUnit_ListDep,
+ "unrecognized cursor unit");
+ const nsCSSValueList* list = cursorValue->GetListValue();
+ for ( ; list->mValue.GetUnit() == eCSSUnit_Array; list = list->mNext) {
+ nsCSSValue::Array* arr = list->mValue.GetArrayValue();
+ imgRequestProxy* req =
+ GetImageRequest(aContext->PresContext(), arr->Item(0));
+ if (req) {
+ nsCursorImage* item = ui->mCursorImages.AppendElement();
+ item->SetImage(req);
+ if (arr->Item(1).GetUnit() != eCSSUnit_Null) {
+ item->mHaveHotspot = true;
+ item->mHotspotX = arr->Item(1).GetFloatValue();
+ item->mHotspotY = arr->Item(2).GetFloatValue();
+ }
+ }
+ }
+
+ NS_ASSERTION(list, "Must have non-array value at the end");
+ NS_ASSERTION(list->mValue.GetUnit() == eCSSUnit_Enumerated,
+ "Unexpected fallback value at end of cursor list");
+ ui->mCursor = list->mValue.GetIntValue();
+ }
+ }
+
+ // user-input: enum, inherit, initial
+ SetValue(*aRuleData->ValueForUserInput(),
+ ui->mUserInput, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentUI->mUserInput,
+ StyleUserInput::Auto);
+
+ // user-modify: enum, inherit, initial
+ SetValue(*aRuleData->ValueForUserModify(),
+ ui->mUserModify, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentUI->mUserModify,
+ StyleUserModify::ReadOnly);
+
+ // user-focus: enum, inherit, initial
+ SetValue(*aRuleData->ValueForUserFocus(),
+ ui->mUserFocus, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentUI->mUserFocus,
+ StyleUserFocus::None);
+
+ // pointer-events: enum, inherit, initial
+ SetValue(*aRuleData->ValueForPointerEvents(), ui->mPointerEvents,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentUI->mPointerEvents,
+ NS_STYLE_POINTER_EVENTS_AUTO);
+
+ COMPUTE_END_INHERITED(UserInterface, ui)
+}
+
+const void*
+nsRuleNode::ComputeUIResetData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(UIReset, ui, parentUI)
+
+ // user-select: enum, inherit, initial
+ SetValue(*aRuleData->ValueForUserSelect(),
+ ui->mUserSelect, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentUI->mUserSelect,
+ StyleUserSelect::Auto);
+
+ // ime-mode: enum, inherit, initial
+ SetValue(*aRuleData->ValueForImeMode(),
+ ui->mIMEMode, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentUI->mIMEMode,
+ NS_STYLE_IME_MODE_AUTO);
+
+ // force-broken-image-icons: integer, inherit, initial
+ SetValue(*aRuleData->ValueForForceBrokenImageIcon(),
+ ui->mForceBrokenImageIcon,
+ conditions,
+ SETVAL_INTEGER | SETVAL_UNSET_INITIAL,
+ parentUI->mForceBrokenImageIcon, 0);
+
+ // -moz-window-dragging: enum, inherit, initial
+ SetValue(*aRuleData->ValueForWindowDragging(),
+ ui->mWindowDragging, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentUI->mWindowDragging,
+ StyleWindowDragging::Default);
+
+ // -moz-window-shadow: enum, inherit, initial
+ SetValue(*aRuleData->ValueForWindowShadow(),
+ ui->mWindowShadow, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentUI->mWindowShadow,
+ NS_STYLE_WINDOW_SHADOW_DEFAULT);
+
+ COMPUTE_END_RESET(UIReset, ui)
+}
+
+// Information about each transition or animation property that is
+// constant.
+struct TransitionPropInfo {
+ nsCSSPropertyID property;
+ // Location of the count of the property's computed value.
+ uint32_t nsStyleDisplay::* sdCount;
+};
+
+// Each property's index in this array must match its index in the
+// mutable array |transitionPropData| below.
+static const TransitionPropInfo transitionPropInfo[4] = {
+ { eCSSProperty_transition_delay,
+ &nsStyleDisplay::mTransitionDelayCount },
+ { eCSSProperty_transition_duration,
+ &nsStyleDisplay::mTransitionDurationCount },
+ { eCSSProperty_transition_property,
+ &nsStyleDisplay::mTransitionPropertyCount },
+ { eCSSProperty_transition_timing_function,
+ &nsStyleDisplay::mTransitionTimingFunctionCount },
+};
+
+// Each property's index in this array must match its index in the
+// mutable array |animationPropData| below.
+static const TransitionPropInfo animationPropInfo[8] = {
+ { eCSSProperty_animation_delay,
+ &nsStyleDisplay::mAnimationDelayCount },
+ { eCSSProperty_animation_duration,
+ &nsStyleDisplay::mAnimationDurationCount },
+ { eCSSProperty_animation_name,
+ &nsStyleDisplay::mAnimationNameCount },
+ { eCSSProperty_animation_timing_function,
+ &nsStyleDisplay::mAnimationTimingFunctionCount },
+ { eCSSProperty_animation_direction,
+ &nsStyleDisplay::mAnimationDirectionCount },
+ { eCSSProperty_animation_fill_mode,
+ &nsStyleDisplay::mAnimationFillModeCount },
+ { eCSSProperty_animation_play_state,
+ &nsStyleDisplay::mAnimationPlayStateCount },
+ { eCSSProperty_animation_iteration_count,
+ &nsStyleDisplay::mAnimationIterationCountCount },
+};
+
+// Information about each transition or animation property that changes
+// during ComputeDisplayData.
+struct TransitionPropData {
+ const nsCSSValueList *list;
+ nsCSSUnit unit;
+ uint32_t num;
+};
+
+static uint32_t
+CountTransitionProps(const TransitionPropInfo* aInfo,
+ TransitionPropData* aData,
+ size_t aLength,
+ nsStyleDisplay* aDisplay,
+ const nsStyleDisplay* aParentDisplay,
+ const nsRuleData* aRuleData,
+ RuleNodeCacheConditions& aConditions)
+{
+ // The four transition properties or eight animation properties are
+ // stored in nsCSSDisplay in a single array for all properties. The
+ // number of transitions is equal to the number of items in the
+ // longest property's value. Properties that have fewer values than
+ // the longest are filled in by repeating the list. However, this
+ // repetition does not extend the computed value of that particular
+ // property (for purposes of inheritance, or, in our code, for when
+ // other properties are overridden by a more specific rule).
+
+ // But actually, since the spec isn't clear yet, we'll fully compute
+ // all of them (so we can switch easily later), but only care about
+ // the ones up to the number of items for 'transition-property', per
+ // http://lists.w3.org/Archives/Public/www-style/2009Aug/0109.html .
+
+ // Transitions are difficult to handle correctly because of this. For
+ // example, we need to handle scenarios such as:
+ // * a more general rule specifies transition-property: a, b, c;
+ // * a more specific rule overrides as transition-property: d;
+ //
+ // If only the general rule applied, we would fill in the extra
+ // properties (duration, delay, etc) with initial values to create 3
+ // fully-specified transitions. But when the more specific rule
+ // applies, we should only create a single transition. In order to do
+ // this we need to remember which properties were explicitly specified
+ // and which ones were just filled in with initial values to get a
+ // fully-specified transition, which we do by remembering the number
+ // of values for each property.
+
+ uint32_t numTransitions = 0;
+ for (size_t i = 0; i < aLength; ++i) {
+ const TransitionPropInfo& info = aInfo[i];
+ TransitionPropData& data = aData[i];
+
+ // cache whether any of the properties are specified as 'inherit' so
+ // we can use it below
+
+ const nsCSSValue& value = *aRuleData->ValueFor(info.property);
+ data.unit = value.GetUnit();
+ data.list = (value.GetUnit() == eCSSUnit_List ||
+ value.GetUnit() == eCSSUnit_ListDep)
+ ? value.GetListValue() : nullptr;
+
+ // General algorithm to determine how many total transitions we need
+ // to build. For each property:
+ // - if there is no value specified in for the property in
+ // displayData, use the values from the start struct, but only if
+ // they were explicitly specified
+ // - if there is a value specified for the property in displayData:
+ // - if the value is 'inherit', count the number of values for
+ // that property are specified by the parent, but only those
+ // that were explicitly specified
+ // - otherwise, count the number of values specified in displayData
+
+
+ // calculate number of elements
+ if (data.unit == eCSSUnit_Inherit) {
+ data.num = aParentDisplay->*(info.sdCount);
+ aConditions.SetUncacheable();
+ } else if (data.list) {
+ data.num = ListLength(data.list);
+ } else {
+ data.num = aDisplay->*(info.sdCount);
+ }
+ if (data.num > numTransitions)
+ numTransitions = data.num;
+ }
+
+ return numTransitions;
+}
+
+/* static */ void
+nsRuleNode::ComputeTimingFunction(const nsCSSValue& aValue,
+ nsTimingFunction& aResult)
+{
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Enumerated:
+ aResult = nsTimingFunction(aValue.GetIntValue());
+ break;
+ case eCSSUnit_Cubic_Bezier:
+ {
+ nsCSSValue::Array* array = aValue.GetArrayValue();
+ NS_ASSERTION(array && array->Count() == 4,
+ "Need 4 control points");
+ aResult = nsTimingFunction(array->Item(0).GetFloatValue(),
+ array->Item(1).GetFloatValue(),
+ array->Item(2).GetFloatValue(),
+ array->Item(3).GetFloatValue());
+ }
+ break;
+ case eCSSUnit_Steps:
+ {
+ nsCSSValue::Array* array = aValue.GetArrayValue();
+ NS_ASSERTION(array && array->Count() == 2,
+ "Need 2 items");
+ NS_ASSERTION(array->Item(0).GetUnit() == eCSSUnit_Integer,
+ "unexpected first value");
+ NS_ASSERTION(array->Item(1).GetUnit() == eCSSUnit_Enumerated &&
+ (array->Item(1).GetIntValue() ==
+ NS_STYLE_TRANSITION_TIMING_FUNCTION_STEP_START ||
+ array->Item(1).GetIntValue() ==
+ NS_STYLE_TRANSITION_TIMING_FUNCTION_STEP_END ||
+ array->Item(1).GetIntValue() == -1),
+ "unexpected second value");
+ nsTimingFunction::Type type =
+ (array->Item(1).GetIntValue() ==
+ NS_STYLE_TRANSITION_TIMING_FUNCTION_STEP_START) ?
+ nsTimingFunction::Type::StepStart :
+ nsTimingFunction::Type::StepEnd;
+ aResult = nsTimingFunction(type, array->Item(0).GetIntValue());
+ }
+ break;
+ default:
+ NS_NOTREACHED("Invalid transition property unit");
+ }
+}
+
+static uint8_t
+GetWillChangeBitFieldFromPropFlags(const nsCSSPropertyID& aProp)
+{
+ uint8_t willChangeBitField = 0;
+ if (nsCSSProps::PropHasFlags(aProp, CSS_PROPERTY_CREATES_STACKING_CONTEXT)) {
+ willChangeBitField |= NS_STYLE_WILL_CHANGE_STACKING_CONTEXT;
+ }
+
+ if (nsCSSProps::PropHasFlags(aProp, CSS_PROPERTY_FIXPOS_CB)) {
+ willChangeBitField |= NS_STYLE_WILL_CHANGE_FIXPOS_CB;
+ }
+
+ if (nsCSSProps::PropHasFlags(aProp, CSS_PROPERTY_ABSPOS_CB)) {
+ willChangeBitField |= NS_STYLE_WILL_CHANGE_ABSPOS_CB;
+ }
+
+ return willChangeBitField;
+}
+
+const void*
+nsRuleNode::ComputeDisplayData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(Display, display, parentDisplay)
+
+ // We may have ended up with aStartStruct's values of mDisplay and
+ // mFloat, but those may not be correct if our style data overrides
+ // its position or float properties. Reset to mOriginalDisplay and
+ // mOriginalFloat; if it turns out we still need the display/floats
+ // adjustments, we'll do them below.
+ display->mDisplay = display->mOriginalDisplay;
+ display->mFloat = display->mOriginalFloat;
+
+ // Each property's index in this array must match its index in the
+ // const array |transitionPropInfo| above.
+ TransitionPropData transitionPropData[4];
+ TransitionPropData& delay = transitionPropData[0];
+ TransitionPropData& duration = transitionPropData[1];
+ TransitionPropData& property = transitionPropData[2];
+ TransitionPropData& timingFunction = transitionPropData[3];
+
+#define FOR_ALL_TRANSITION_PROPS(var_) \
+ for (uint32_t var_ = 0; var_ < 4; ++var_)
+
+ // CSS Transitions
+ uint32_t numTransitions =
+ CountTransitionProps(transitionPropInfo, transitionPropData,
+ ArrayLength(transitionPropData),
+ display, parentDisplay, aRuleData,
+ conditions);
+
+ display->mTransitions.SetLengthNonZero(numTransitions);
+
+ FOR_ALL_TRANSITION_PROPS(p) {
+ const TransitionPropInfo& i = transitionPropInfo[p];
+ TransitionPropData& d = transitionPropData[p];
+
+ display->*(i.sdCount) = d.num;
+ }
+
+ // Fill in the transitions we just allocated with the appropriate values.
+ for (uint32_t i = 0; i < numTransitions; ++i) {
+ StyleTransition *transition = &display->mTransitions[i];
+
+ if (i >= delay.num) {
+ MOZ_ASSERT(delay.num, "delay.num must be greater than 0");
+ transition->SetDelay(display->mTransitions[i % delay.num].GetDelay());
+ } else if (delay.unit == eCSSUnit_Inherit) {
+ // FIXME (Bug 522599) (for all transition properties): write a test that
+ // detects when this was wrong for i >= delay.num if parent had
+ // count for this property not equal to length
+ MOZ_ASSERT(i < parentDisplay->mTransitionDelayCount,
+ "delay.num computed incorrectly");
+ MOZ_ASSERT(!conditions.Cacheable(),
+ "should have made conditions.Cacheable() false above");
+ transition->SetDelay(parentDisplay->mTransitions[i].GetDelay());
+ } else if (delay.unit == eCSSUnit_Initial ||
+ delay.unit == eCSSUnit_Unset) {
+ transition->SetDelay(0.0);
+ } else if (delay.list) {
+ switch (delay.list->mValue.GetUnit()) {
+ case eCSSUnit_Seconds:
+ transition->SetDelay(PR_MSEC_PER_SEC *
+ delay.list->mValue.GetFloatValue());
+ break;
+ case eCSSUnit_Milliseconds:
+ transition->SetDelay(delay.list->mValue.GetFloatValue());
+ break;
+ default:
+ NS_NOTREACHED("Invalid delay unit");
+ }
+ }
+
+ if (i >= duration.num) {
+ MOZ_ASSERT(duration.num, "duration.num must be greater than 0");
+ transition->SetDuration(
+ display->mTransitions[i % duration.num].GetDuration());
+ } else if (duration.unit == eCSSUnit_Inherit) {
+ MOZ_ASSERT(i < parentDisplay->mTransitionDurationCount,
+ "duration.num computed incorrectly");
+ MOZ_ASSERT(!conditions.Cacheable(),
+ "should have made conditions.Cacheable() false above");
+ transition->SetDuration(parentDisplay->mTransitions[i].GetDuration());
+ } else if (duration.unit == eCSSUnit_Initial ||
+ duration.unit == eCSSUnit_Unset) {
+ transition->SetDuration(0.0);
+ } else if (duration.list) {
+ switch (duration.list->mValue.GetUnit()) {
+ case eCSSUnit_Seconds:
+ transition->SetDuration(PR_MSEC_PER_SEC *
+ duration.list->mValue.GetFloatValue());
+ break;
+ case eCSSUnit_Milliseconds:
+ transition->SetDuration(duration.list->mValue.GetFloatValue());
+ break;
+ default:
+ NS_NOTREACHED("Invalid duration unit");
+ }
+ }
+
+ if (i >= property.num) {
+ MOZ_ASSERT(property.num, "property.num must be greater than 0");
+ transition->CopyPropertyFrom(display->mTransitions[i % property.num]);
+ } else if (property.unit == eCSSUnit_Inherit) {
+ MOZ_ASSERT(i < parentDisplay->mTransitionPropertyCount,
+ "property.num computed incorrectly");
+ MOZ_ASSERT(!conditions.Cacheable(),
+ "should have made conditions.Cacheable() false above");
+ transition->CopyPropertyFrom(parentDisplay->mTransitions[i]);
+ } else if (property.unit == eCSSUnit_Initial ||
+ property.unit == eCSSUnit_Unset) {
+ transition->SetProperty(eCSSPropertyExtra_all_properties);
+ } else if (property.unit == eCSSUnit_None) {
+ transition->SetProperty(eCSSPropertyExtra_no_properties);
+ } else if (property.list) {
+ const nsCSSValue &val = property.list->mValue;
+
+ if (val.GetUnit() == eCSSUnit_Ident) {
+ nsDependentString
+ propertyStr(property.list->mValue.GetStringBufferValue());
+ nsCSSPropertyID prop =
+ nsCSSProps::LookupProperty(propertyStr,
+ CSSEnabledState::eForAllContent);
+ if (prop == eCSSProperty_UNKNOWN ||
+ prop == eCSSPropertyExtra_variable) {
+ transition->SetUnknownProperty(prop, propertyStr);
+ } else {
+ transition->SetProperty(prop);
+ }
+ } else {
+ MOZ_ASSERT(val.GetUnit() == eCSSUnit_All,
+ "Invalid transition property unit");
+ transition->SetProperty(eCSSPropertyExtra_all_properties);
+ }
+ }
+
+ if (i >= timingFunction.num) {
+ MOZ_ASSERT(timingFunction.num,
+ "timingFunction.num must be greater than 0");
+ transition->SetTimingFunction(
+ display->mTransitions[i % timingFunction.num].GetTimingFunction());
+ } else if (timingFunction.unit == eCSSUnit_Inherit) {
+ MOZ_ASSERT(i < parentDisplay->mTransitionTimingFunctionCount,
+ "timingFunction.num computed incorrectly");
+ MOZ_ASSERT(!conditions.Cacheable(),
+ "should have made conditions.Cacheable() false above");
+ transition->SetTimingFunction(
+ parentDisplay->mTransitions[i].GetTimingFunction());
+ } else if (timingFunction.unit == eCSSUnit_Initial ||
+ timingFunction.unit == eCSSUnit_Unset) {
+ transition->SetTimingFunction(
+ nsTimingFunction(NS_STYLE_TRANSITION_TIMING_FUNCTION_EASE));
+ } else if (timingFunction.list) {
+ ComputeTimingFunction(timingFunction.list->mValue,
+ transition->TimingFunctionSlot());
+ }
+
+ FOR_ALL_TRANSITION_PROPS(p) {
+ const TransitionPropInfo& info = transitionPropInfo[p];
+ TransitionPropData& d = transitionPropData[p];
+
+ // if we're at the end of the list, start at the beginning and repeat
+ // until we're out of transitions to populate
+ if (d.list) {
+ d.list = d.list->mNext ? d.list->mNext :
+ aRuleData->ValueFor(info.property)->GetListValue();
+ }
+ }
+ }
+
+ // Each property's index in this array must match its index in the
+ // const array |animationPropInfo| above.
+ TransitionPropData animationPropData[8];
+ TransitionPropData& animDelay = animationPropData[0];
+ TransitionPropData& animDuration = animationPropData[1];
+ TransitionPropData& animName = animationPropData[2];
+ TransitionPropData& animTimingFunction = animationPropData[3];
+ TransitionPropData& animDirection = animationPropData[4];
+ TransitionPropData& animFillMode = animationPropData[5];
+ TransitionPropData& animPlayState = animationPropData[6];
+ TransitionPropData& animIterationCount = animationPropData[7];
+
+#define FOR_ALL_ANIMATION_PROPS(var_) \
+ for (uint32_t var_ = 0; var_ < 8; ++var_)
+
+ // CSS Animations.
+
+ uint32_t numAnimations =
+ CountTransitionProps(animationPropInfo, animationPropData,
+ ArrayLength(animationPropData),
+ display, parentDisplay, aRuleData,
+ conditions);
+
+ display->mAnimations.SetLengthNonZero(numAnimations);
+
+ FOR_ALL_ANIMATION_PROPS(p) {
+ const TransitionPropInfo& i = animationPropInfo[p];
+ TransitionPropData& d = animationPropData[p];
+
+ display->*(i.sdCount) = d.num;
+ }
+
+ // Fill in the animations we just allocated with the appropriate values.
+ for (uint32_t i = 0; i < numAnimations; ++i) {
+ StyleAnimation *animation = &display->mAnimations[i];
+
+ if (i >= animDelay.num) {
+ MOZ_ASSERT(animDelay.num, "animDelay.num must be greater than 0");
+ animation->SetDelay(display->mAnimations[i % animDelay.num].GetDelay());
+ } else if (animDelay.unit == eCSSUnit_Inherit) {
+ // FIXME (Bug 522599) (for all animation properties): write a test that
+ // detects when this was wrong for i >= animDelay.num if parent had
+ // count for this property not equal to length
+ MOZ_ASSERT(i < parentDisplay->mAnimationDelayCount,
+ "animDelay.num computed incorrectly");
+ MOZ_ASSERT(!conditions.Cacheable(),
+ "should have made conditions.Cacheable() false above");
+ animation->SetDelay(parentDisplay->mAnimations[i].GetDelay());
+ } else if (animDelay.unit == eCSSUnit_Initial ||
+ animDelay.unit == eCSSUnit_Unset) {
+ animation->SetDelay(0.0);
+ } else if (animDelay.list) {
+ switch (animDelay.list->mValue.GetUnit()) {
+ case eCSSUnit_Seconds:
+ animation->SetDelay(PR_MSEC_PER_SEC *
+ animDelay.list->mValue.GetFloatValue());
+ break;
+ case eCSSUnit_Milliseconds:
+ animation->SetDelay(animDelay.list->mValue.GetFloatValue());
+ break;
+ default:
+ NS_NOTREACHED("Invalid delay unit");
+ }
+ }
+
+ if (i >= animDuration.num) {
+ MOZ_ASSERT(animDuration.num, "animDuration.num must be greater than 0");
+ animation->SetDuration(
+ display->mAnimations[i % animDuration.num].GetDuration());
+ } else if (animDuration.unit == eCSSUnit_Inherit) {
+ MOZ_ASSERT(i < parentDisplay->mAnimationDurationCount,
+ "animDuration.num computed incorrectly");
+ MOZ_ASSERT(!conditions.Cacheable(),
+ "should have made conditions.Cacheable() false above");
+ animation->SetDuration(parentDisplay->mAnimations[i].GetDuration());
+ } else if (animDuration.unit == eCSSUnit_Initial ||
+ animDuration.unit == eCSSUnit_Unset) {
+ animation->SetDuration(0.0);
+ } else if (animDuration.list) {
+ switch (animDuration.list->mValue.GetUnit()) {
+ case eCSSUnit_Seconds:
+ animation->SetDuration(PR_MSEC_PER_SEC *
+ animDuration.list->mValue.GetFloatValue());
+ break;
+ case eCSSUnit_Milliseconds:
+ animation->SetDuration(animDuration.list->mValue.GetFloatValue());
+ break;
+ default:
+ NS_NOTREACHED("Invalid duration unit");
+ }
+ }
+
+ if (i >= animName.num) {
+ MOZ_ASSERT(animName.num, "animName.num must be greater than 0");
+ animation->SetName(display->mAnimations[i % animName.num].GetName());
+ } else if (animName.unit == eCSSUnit_Inherit) {
+ MOZ_ASSERT(i < parentDisplay->mAnimationNameCount,
+ "animName.num computed incorrectly");
+ MOZ_ASSERT(!conditions.Cacheable(),
+ "should have made conditions.Cacheable() false above");
+ animation->SetName(parentDisplay->mAnimations[i].GetName());
+ } else if (animName.unit == eCSSUnit_Initial ||
+ animName.unit == eCSSUnit_Unset) {
+ animation->SetName(EmptyString());
+ } else if (animName.list) {
+ switch (animName.list->mValue.GetUnit()) {
+ case eCSSUnit_Ident: {
+ nsDependentString
+ nameStr(animName.list->mValue.GetStringBufferValue());
+ animation->SetName(nameStr);
+ break;
+ }
+ case eCSSUnit_None: {
+ animation->SetName(EmptyString());
+ break;
+ }
+ default:
+ MOZ_ASSERT(false, "Invalid animation-name unit");
+ }
+ }
+
+ if (i >= animTimingFunction.num) {
+ MOZ_ASSERT(animTimingFunction.num,
+ "animTimingFunction.num must be greater than 0");
+ animation->SetTimingFunction(
+ display->mAnimations[i % animTimingFunction.num].GetTimingFunction());
+ } else if (animTimingFunction.unit == eCSSUnit_Inherit) {
+ MOZ_ASSERT(i < parentDisplay->mAnimationTimingFunctionCount,
+ "animTimingFunction.num computed incorrectly");
+ MOZ_ASSERT(!conditions.Cacheable(),
+ "should have made conditions.Cacheable() false above");
+ animation->SetTimingFunction(
+ parentDisplay->mAnimations[i].GetTimingFunction());
+ } else if (animTimingFunction.unit == eCSSUnit_Initial ||
+ animTimingFunction.unit == eCSSUnit_Unset) {
+ animation->SetTimingFunction(
+ nsTimingFunction(NS_STYLE_TRANSITION_TIMING_FUNCTION_EASE));
+ } else if (animTimingFunction.list) {
+ ComputeTimingFunction(animTimingFunction.list->mValue,
+ animation->TimingFunctionSlot());
+ }
+
+ if (i >= animDirection.num) {
+ MOZ_ASSERT(animDirection.num,
+ "animDirection.num must be greater than 0");
+ animation->SetDirection(display->mAnimations[i % animDirection.num].GetDirection());
+ } else if (animDirection.unit == eCSSUnit_Inherit) {
+ MOZ_ASSERT(i < parentDisplay->mAnimationDirectionCount,
+ "animDirection.num computed incorrectly");
+ MOZ_ASSERT(!conditions.Cacheable(),
+ "should have made conditions.Cacheable() false above");
+ animation->SetDirection(parentDisplay->mAnimations[i].GetDirection());
+ } else if (animDirection.unit == eCSSUnit_Initial ||
+ animDirection.unit == eCSSUnit_Unset) {
+ animation->SetDirection(dom::PlaybackDirection::Normal);
+ } else if (animDirection.list) {
+ MOZ_ASSERT(animDirection.list->mValue.GetUnit() == eCSSUnit_Enumerated,
+ "Invalid animation-direction unit");
+
+ animation->SetDirection(
+ static_cast<dom::PlaybackDirection>(animDirection.list->mValue.GetIntValue()));
+ }
+
+ if (i >= animFillMode.num) {
+ MOZ_ASSERT(animFillMode.num, "animFillMode.num must be greater than 0");
+ animation->SetFillMode(display->mAnimations[i % animFillMode.num].GetFillMode());
+ } else if (animFillMode.unit == eCSSUnit_Inherit) {
+ MOZ_ASSERT(i < parentDisplay->mAnimationFillModeCount,
+ "animFillMode.num computed incorrectly");
+ MOZ_ASSERT(!conditions.Cacheable(),
+ "should have made conditions.Cacheable() false above");
+ animation->SetFillMode(parentDisplay->mAnimations[i].GetFillMode());
+ } else if (animFillMode.unit == eCSSUnit_Initial ||
+ animFillMode.unit == eCSSUnit_Unset) {
+ animation->SetFillMode(dom::FillMode::None);
+ } else if (animFillMode.list) {
+ MOZ_ASSERT(animFillMode.list->mValue.GetUnit() == eCSSUnit_Enumerated,
+ "Invalid animation-fill-mode unit");
+
+ animation->SetFillMode(
+ static_cast<dom::FillMode>(animFillMode.list->mValue.GetIntValue()));
+ }
+
+ if (i >= animPlayState.num) {
+ MOZ_ASSERT(animPlayState.num,
+ "animPlayState.num must be greater than 0");
+ animation->SetPlayState(display->mAnimations[i % animPlayState.num].GetPlayState());
+ } else if (animPlayState.unit == eCSSUnit_Inherit) {
+ MOZ_ASSERT(i < parentDisplay->mAnimationPlayStateCount,
+ "animPlayState.num computed incorrectly");
+ MOZ_ASSERT(!conditions.Cacheable(),
+ "should have made conditions.Cacheable() false above");
+ animation->SetPlayState(parentDisplay->mAnimations[i].GetPlayState());
+ } else if (animPlayState.unit == eCSSUnit_Initial ||
+ animPlayState.unit == eCSSUnit_Unset) {
+ animation->SetPlayState(NS_STYLE_ANIMATION_PLAY_STATE_RUNNING);
+ } else if (animPlayState.list) {
+ MOZ_ASSERT(animPlayState.list->mValue.GetUnit() == eCSSUnit_Enumerated,
+ "Invalid animation-play-state unit");
+
+ animation->SetPlayState(animPlayState.list->mValue.GetIntValue());
+ }
+
+ if (i >= animIterationCount.num) {
+ MOZ_ASSERT(animIterationCount.num,
+ "animIterationCount.num must be greater than 0");
+ animation->SetIterationCount(display->mAnimations[i % animIterationCount.num].GetIterationCount());
+ } else if (animIterationCount.unit == eCSSUnit_Inherit) {
+ MOZ_ASSERT(i < parentDisplay->mAnimationIterationCountCount,
+ "animIterationCount.num computed incorrectly");
+ MOZ_ASSERT(!conditions.Cacheable(),
+ "should have made conditions.Cacheable() false above");
+ animation->SetIterationCount(parentDisplay->mAnimations[i].GetIterationCount());
+ } else if (animIterationCount.unit == eCSSUnit_Initial ||
+ animIterationCount.unit == eCSSUnit_Unset) {
+ animation->SetIterationCount(1.0f);
+ } else if (animIterationCount.list) {
+ switch (animIterationCount.list->mValue.GetUnit()) {
+ case eCSSUnit_Enumerated:
+ MOZ_ASSERT(animIterationCount.list->mValue.GetIntValue() ==
+ NS_STYLE_ANIMATION_ITERATION_COUNT_INFINITE,
+ "unexpected value");
+ animation->SetIterationCount(NS_IEEEPositiveInfinity());
+ break;
+ case eCSSUnit_Number:
+ animation->SetIterationCount(
+ animIterationCount.list->mValue.GetFloatValue());
+ break;
+ default:
+ MOZ_ASSERT(false,
+ "unexpected animation-iteration-count unit");
+ }
+ }
+
+ FOR_ALL_ANIMATION_PROPS(p) {
+ const TransitionPropInfo& info = animationPropInfo[p];
+ TransitionPropData& d = animationPropData[p];
+
+ // if we're at the end of the list, start at the beginning and repeat
+ // until we're out of animations to populate
+ if (d.list) {
+ d.list = d.list->mNext ? d.list->mNext :
+ aRuleData->ValueFor(info.property)->GetListValue();
+ }
+ }
+ }
+
+ // display: enum, inherit, initial
+ SetValue(*aRuleData->ValueForDisplay(), display->mDisplay, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mDisplay,
+ StyleDisplay::Inline);
+
+ // contain: none, enum, inherit, initial
+ SetValue(*aRuleData->ValueForContain(), display->mContain, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mContain,
+ NS_STYLE_CONTAIN_NONE, Unused,
+ NS_STYLE_CONTAIN_NONE, Unused, Unused);
+
+ // scroll-behavior: enum, inherit, initial
+ SetValue(*aRuleData->ValueForScrollBehavior(), display->mScrollBehavior,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mScrollBehavior, NS_STYLE_SCROLL_BEHAVIOR_AUTO);
+
+ // scroll-snap-type-x: none, enum, inherit, initial
+ SetValue(*aRuleData->ValueForScrollSnapTypeX(), display->mScrollSnapTypeX,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mScrollSnapTypeX, NS_STYLE_SCROLL_SNAP_TYPE_NONE);
+
+ // scroll-snap-type-y: none, enum, inherit, initial
+ SetValue(*aRuleData->ValueForScrollSnapTypeY(), display->mScrollSnapTypeY,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mScrollSnapTypeY, NS_STYLE_SCROLL_SNAP_TYPE_NONE);
+
+ // scroll-snap-points-x: none, inherit, initial
+ const nsCSSValue& scrollSnapPointsX = *aRuleData->ValueForScrollSnapPointsX();
+ switch (scrollSnapPointsX.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_None:
+ display->mScrollSnapPointsX.SetNoneValue();
+ break;
+ case eCSSUnit_Inherit:
+ display->mScrollSnapPointsX = parentDisplay->mScrollSnapPointsX;
+ conditions.SetUncacheable();
+ break;
+ case eCSSUnit_Function: {
+ nsCSSValue::Array* func = scrollSnapPointsX.GetArrayValue();
+ NS_ASSERTION(func->Item(0).GetKeywordValue() == eCSSKeyword_repeat,
+ "Expected repeat(), got another function name");
+ nsStyleCoord coord;
+ if (SetCoord(func->Item(1), coord, nsStyleCoord(),
+ SETCOORD_LP | SETCOORD_STORE_CALC |
+ SETCOORD_CALC_CLAMP_NONNEGATIVE,
+ aContext, mPresContext, conditions)) {
+ NS_ASSERTION(coord.GetUnit() == eStyleUnit_Coord ||
+ coord.GetUnit() == eStyleUnit_Percent ||
+ coord.GetUnit() == eStyleUnit_Calc,
+ "unexpected unit");
+ display->mScrollSnapPointsX = coord;
+ }
+ break;
+ }
+ default:
+ NS_NOTREACHED("unexpected unit");
+ }
+
+ // scroll-snap-points-y: none, inherit, initial
+ const nsCSSValue& scrollSnapPointsY = *aRuleData->ValueForScrollSnapPointsY();
+ switch (scrollSnapPointsY.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_None:
+ display->mScrollSnapPointsY.SetNoneValue();
+ break;
+ case eCSSUnit_Inherit:
+ display->mScrollSnapPointsY = parentDisplay->mScrollSnapPointsY;
+ conditions.SetUncacheable();
+ break;
+ case eCSSUnit_Function: {
+ nsCSSValue::Array* func = scrollSnapPointsY.GetArrayValue();
+ NS_ASSERTION(func->Item(0).GetKeywordValue() == eCSSKeyword_repeat,
+ "Expected repeat(), got another function name");
+ nsStyleCoord coord;
+ if (SetCoord(func->Item(1), coord, nsStyleCoord(),
+ SETCOORD_LP | SETCOORD_STORE_CALC |
+ SETCOORD_CALC_CLAMP_NONNEGATIVE,
+ aContext, mPresContext, conditions)) {
+ NS_ASSERTION(coord.GetUnit() == eStyleUnit_Coord ||
+ coord.GetUnit() == eStyleUnit_Percent ||
+ coord.GetUnit() == eStyleUnit_Calc,
+ "unexpected unit");
+ display->mScrollSnapPointsY = coord;
+ }
+ break;
+ }
+ default:
+ NS_NOTREACHED("unexpected unit");
+ }
+
+ // scroll-snap-destination: inherit, initial
+ const nsCSSValue& snapDestination = *aRuleData->ValueForScrollSnapDestination();
+ switch (snapDestination.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ display->mScrollSnapDestination.SetInitialZeroValues();
+ break;
+ case eCSSUnit_Inherit:
+ display->mScrollSnapDestination = parentDisplay->mScrollSnapDestination;
+ conditions.SetUncacheable();
+ break;
+ default: {
+ ComputePositionValue(aContext, snapDestination,
+ display->mScrollSnapDestination, conditions);
+ }
+ }
+
+ // scroll-snap-coordinate: none, inherit, initial
+ const nsCSSValue& snapCoordinate = *aRuleData->ValueForScrollSnapCoordinate();
+ switch (snapCoordinate.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_None:
+ // Unset and Initial is none, indicated by an empty array
+ display->mScrollSnapCoordinate.Clear();
+ break;
+ case eCSSUnit_Inherit:
+ display->mScrollSnapCoordinate = parentDisplay->mScrollSnapCoordinate;
+ conditions.SetUncacheable();
+ break;
+ case eCSSUnit_List: {
+ display->mScrollSnapCoordinate.Clear();
+ const nsCSSValueList* item = snapCoordinate.GetListValue();
+ do {
+ NS_ASSERTION(item->mValue.GetUnit() != eCSSUnit_Null &&
+ item->mValue.GetUnit() != eCSSUnit_Inherit &&
+ item->mValue.GetUnit() != eCSSUnit_Initial &&
+ item->mValue.GetUnit() != eCSSUnit_Unset,
+ "unexpected unit");
+ Position* pos = display->mScrollSnapCoordinate.AppendElement();
+ ComputePositionValue(aContext, item->mValue, *pos, conditions);
+ item = item->mNext;
+ } while(item);
+ break;
+ }
+ default:
+ NS_NOTREACHED("unexpected unit");
+ }
+
+ // isolation: enum, inherit, initial
+ SetValue(*aRuleData->ValueForIsolation(), display->mIsolation,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mIsolation, NS_STYLE_ISOLATION_AUTO);
+
+ // -moz-top-layer: enum, inherit, initial
+ SetValue(*aRuleData->ValueForTopLayer(), display->mTopLayer,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mTopLayer, NS_STYLE_TOP_LAYER_NONE);
+
+ // Backup original display value for calculation of a hypothetical
+ // box (CSS2 10.6.4/10.6.5), in addition to getting our style data right later.
+ // See ReflowInput::CalculateHypotheticalBox
+ display->mOriginalDisplay = display->mDisplay;
+
+ // appearance: enum, inherit, initial
+ SetValue(*aRuleData->ValueForAppearance(),
+ display->mAppearance, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mAppearance,
+ NS_THEME_NONE);
+
+ // binding: url, none, inherit
+ const nsCSSValue* bindingValue = aRuleData->ValueForBinding();
+ if (eCSSUnit_URL == bindingValue->GetUnit()) {
+ mozilla::css::URLValue* url = bindingValue->GetURLStructValue();
+ NS_ASSERTION(url, "What's going on here?");
+
+ if (MOZ_LIKELY(url->GetURI())) {
+ display->mBinding = url;
+ } else {
+ display->mBinding = nullptr;
+ }
+ }
+ else if (eCSSUnit_None == bindingValue->GetUnit() ||
+ eCSSUnit_Initial == bindingValue->GetUnit() ||
+ eCSSUnit_Unset == bindingValue->GetUnit()) {
+ display->mBinding = nullptr;
+ }
+ else if (eCSSUnit_Inherit == bindingValue->GetUnit()) {
+ conditions.SetUncacheable();
+ display->mBinding = parentDisplay->mBinding;
+ }
+
+ // position: enum, inherit, initial
+ SetValue(*aRuleData->ValueForPosition(), display->mPosition, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mPosition,
+ NS_STYLE_POSITION_STATIC);
+ // If an element is put in the top layer, while it is not absolutely
+ // positioned, the position value should be computed to 'absolute' per
+ // the Fullscreen API spec.
+ if (display->mTopLayer != NS_STYLE_TOP_LAYER_NONE &&
+ !display->IsAbsolutelyPositionedStyle()) {
+ display->mPosition = NS_STYLE_POSITION_ABSOLUTE;
+ // We cannot cache this struct because otherwise it may be used as
+ // an aStartStruct for some other elements.
+ conditions.SetUncacheable();
+ }
+
+ // clear: enum, inherit, initial
+ SetValue(*aRuleData->ValueForClear(), display->mBreakType, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mBreakType,
+ StyleClear::None);
+
+ // temp fix for bug 24000
+ // Map 'auto' and 'avoid' to false, and 'always', 'left', and
+ // 'right' to true.
+ // "A conforming user agent may interpret the values 'left' and
+ // 'right' as 'always'." - CSS2.1, section 13.3.1
+ const nsCSSValue* breakBeforeValue = aRuleData->ValueForPageBreakBefore();
+ if (eCSSUnit_Enumerated == breakBeforeValue->GetUnit()) {
+ display->mBreakBefore =
+ (NS_STYLE_PAGE_BREAK_AVOID != breakBeforeValue->GetIntValue() &&
+ NS_STYLE_PAGE_BREAK_AUTO != breakBeforeValue->GetIntValue());
+ }
+ else if (eCSSUnit_Initial == breakBeforeValue->GetUnit() ||
+ eCSSUnit_Unset == breakBeforeValue->GetUnit()) {
+ display->mBreakBefore = false;
+ }
+ else if (eCSSUnit_Inherit == breakBeforeValue->GetUnit()) {
+ conditions.SetUncacheable();
+ display->mBreakBefore = parentDisplay->mBreakBefore;
+ }
+
+ const nsCSSValue* breakAfterValue = aRuleData->ValueForPageBreakAfter();
+ if (eCSSUnit_Enumerated == breakAfterValue->GetUnit()) {
+ display->mBreakAfter =
+ (NS_STYLE_PAGE_BREAK_AVOID != breakAfterValue->GetIntValue() &&
+ NS_STYLE_PAGE_BREAK_AUTO != breakAfterValue->GetIntValue());
+ }
+ else if (eCSSUnit_Initial == breakAfterValue->GetUnit() ||
+ eCSSUnit_Unset == breakAfterValue->GetUnit()) {
+ display->mBreakAfter = false;
+ }
+ else if (eCSSUnit_Inherit == breakAfterValue->GetUnit()) {
+ conditions.SetUncacheable();
+ display->mBreakAfter = parentDisplay->mBreakAfter;
+ }
+ // end temp fix
+
+ // page-break-inside: enum, inherit, initial
+ SetValue(*aRuleData->ValueForPageBreakInside(),
+ display->mBreakInside, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mBreakInside,
+ NS_STYLE_PAGE_BREAK_AUTO);
+
+ // touch-action: none, auto, enum, inherit, initial
+ SetValue(*aRuleData->ValueForTouchAction(), display->mTouchAction,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mTouchAction,
+ /* initial */ NS_STYLE_TOUCH_ACTION_AUTO,
+ /* auto */ NS_STYLE_TOUCH_ACTION_AUTO,
+ /* none */ NS_STYLE_TOUCH_ACTION_NONE, Unused, Unused);
+
+ // float: enum, inherit, initial
+ SetValue(*aRuleData->ValueForFloat(),
+ display->mFloat, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mFloat,
+ StyleFloat::None);
+ // Save mFloat in mOriginalFloat in case we need it later
+ display->mOriginalFloat = display->mFloat;
+
+ // overflow-x: enum, inherit, initial
+ SetValue(*aRuleData->ValueForOverflowX(),
+ display->mOverflowX, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mOverflowX,
+ NS_STYLE_OVERFLOW_VISIBLE);
+
+ // overflow-y: enum, inherit, initial
+ SetValue(*aRuleData->ValueForOverflowY(),
+ display->mOverflowY, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mOverflowY,
+ NS_STYLE_OVERFLOW_VISIBLE);
+
+ // CSS3 overflow-x and overflow-y require some fixup as well in some
+ // cases. NS_STYLE_OVERFLOW_VISIBLE and NS_STYLE_OVERFLOW_CLIP are
+ // meaningful only when used in both dimensions.
+ if (display->mOverflowX != display->mOverflowY &&
+ (display->mOverflowX == NS_STYLE_OVERFLOW_VISIBLE ||
+ display->mOverflowX == NS_STYLE_OVERFLOW_CLIP ||
+ display->mOverflowY == NS_STYLE_OVERFLOW_VISIBLE ||
+ display->mOverflowY == NS_STYLE_OVERFLOW_CLIP)) {
+ // We can't store in the rule tree since a more specific rule might
+ // change these conditions.
+ conditions.SetUncacheable();
+
+ // NS_STYLE_OVERFLOW_CLIP is a deprecated value, so if it's specified
+ // in only one dimension, convert it to NS_STYLE_OVERFLOW_HIDDEN.
+ if (display->mOverflowX == NS_STYLE_OVERFLOW_CLIP)
+ display->mOverflowX = NS_STYLE_OVERFLOW_HIDDEN;
+ if (display->mOverflowY == NS_STYLE_OVERFLOW_CLIP)
+ display->mOverflowY = NS_STYLE_OVERFLOW_HIDDEN;
+
+ // If 'visible' is specified but doesn't match the other dimension, it
+ // turns into 'auto'.
+ if (display->mOverflowX == NS_STYLE_OVERFLOW_VISIBLE)
+ display->mOverflowX = NS_STYLE_OVERFLOW_AUTO;
+ if (display->mOverflowY == NS_STYLE_OVERFLOW_VISIBLE)
+ display->mOverflowY = NS_STYLE_OVERFLOW_AUTO;
+ }
+
+ // When 'contain: paint', update overflow from 'visible' to 'clip'.
+ if (display->IsContainPaint()) {
+ // XXX This actually sets overflow-[x|y] to -moz-hidden-unscrollable.
+ if (display->mOverflowX == NS_STYLE_OVERFLOW_VISIBLE) {
+ // This uncacheability (and the one below) could be fixed by adding
+ // mOriginalOverflowX and mOriginalOverflowY fields, if necessary.
+ display->mOverflowX = NS_STYLE_OVERFLOW_CLIP;
+ conditions.SetUncacheable();
+ }
+ if (display->mOverflowY == NS_STYLE_OVERFLOW_VISIBLE) {
+ display->mOverflowY = NS_STYLE_OVERFLOW_CLIP;
+ conditions.SetUncacheable();
+ }
+ }
+
+ SetValue(*aRuleData->ValueForOverflowClipBox(), display->mOverflowClipBox,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mOverflowClipBox,
+ NS_STYLE_OVERFLOW_CLIP_BOX_PADDING_BOX);
+
+ SetValue(*aRuleData->ValueForResize(), display->mResize, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mResize,
+ NS_STYLE_RESIZE_NONE);
+
+ if (display->mDisplay != StyleDisplay::None) {
+ // CSS2 9.7 specifies display type corrections dealing with 'float'
+ // and 'position'. Since generated content can't be floated or
+ // positioned, we can deal with it here.
+
+ nsIAtom* pseudo = aContext->GetPseudo();
+ if (pseudo && display->mDisplay == StyleDisplay::Contents) {
+ // We don't want to create frames for anonymous content using a parent
+ // frame that is for content above the root of the anon tree.
+ // (XXX what we really should check here is not GetPseudo() but if there's
+ // a 'content' property value that implies anon content but we can't
+ // check that here since that's a different struct(?))
+ // We might get display:contents to work for CSS_PSEUDO_ELEMENT_CONTAINS_ELEMENTS
+ // pseudos (:first-letter etc) in the future, but those have a lot of
+ // special handling in frame construction so they are also unsupported
+ // for now.
+ display->mOriginalDisplay = display->mDisplay = StyleDisplay::Inline;
+ conditions.SetUncacheable();
+ }
+
+ // Inherit a <fieldset> grid/flex display type into its anon content frame.
+ if (pseudo == nsCSSAnonBoxes::fieldsetContent) {
+ MOZ_ASSERT(display->mDisplay == StyleDisplay::Block,
+ "forms.css should have set 'display:block'");
+ switch (parentDisplay->mDisplay) {
+ case StyleDisplay::Grid:
+ case StyleDisplay::InlineGrid:
+ display->mDisplay = StyleDisplay::Grid;
+ conditions.SetUncacheable();
+ break;
+ case StyleDisplay::Flex:
+ case StyleDisplay::InlineFlex:
+ display->mDisplay = StyleDisplay::Flex;
+ conditions.SetUncacheable();
+ break;
+ default:
+ break; // Do nothing
+ }
+ }
+
+ if (nsCSSPseudoElements::firstLetter == pseudo) {
+ // a non-floating first-letter must be inline
+ // XXX this fix can go away once bug 103189 is fixed correctly
+ // Note that we reset mOriginalDisplay to enforce the invariant that it equals mDisplay if we're not positioned or floating.
+ display->mOriginalDisplay = display->mDisplay = StyleDisplay::Inline;
+
+ // We can't cache the data in the rule tree since if a more specific
+ // rule has 'float: left' we'll end up with the wrong 'display'
+ // property.
+ conditions.SetUncacheable();
+ }
+
+ if (display->IsAbsolutelyPositionedStyle()) {
+ // 1) if position is 'absolute' or 'fixed' then display must be
+ // block-level and float must be 'none'
+ EnsureBlockDisplay(display->mDisplay);
+ display->mFloat = StyleFloat::None;
+
+ // Note that it's OK to cache this struct in the ruletree
+ // because it's fine as-is for any style context that points to
+ // it directly, and any use of it as aStartStruct (e.g. if a
+ // more specific rule sets "position: static") will use
+ // mOriginalDisplay and mOriginalFloat, which we have carefully
+ // not changed.
+ } else if (display->mFloat != StyleFloat::None) {
+ // 2) if float is not none, and display is not none, then we must
+ // set a block-level 'display' type per CSS2.1 section 9.7.
+ EnsureBlockDisplay(display->mDisplay);
+
+ // Note that it's OK to cache this struct in the ruletree
+ // because it's fine as-is for any style context that points to
+ // it directly, and any use of it as aStartStruct (e.g. if a
+ // more specific rule sets "float: none") will use
+ // mOriginalDisplay, which we have carefully not changed.
+ }
+
+ if (display->IsContainPaint()) {
+ // An element with contain:paint or contain:layout needs to "be a
+ // formatting context". For the purposes of the "display" property, that
+ // just means we need to promote "display:inline" to "inline-block".
+ // XXX We may also need to promote ruby display vals; see bug 1179349.
+
+ // It's okay to cache this change in the rule tree for the same
+ // reasons as floats in the previous condition.
+ if (display->mDisplay == StyleDisplay::Inline) {
+ display->mDisplay = StyleDisplay::InlineBlock;
+ }
+ }
+ }
+
+ /* Convert the nsCSSValueList into an nsTArray<nsTransformFunction *>. */
+ const nsCSSValue* transformValue = aRuleData->ValueForTransform();
+ switch (transformValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_None:
+ display->mSpecifiedTransform = nullptr;
+ break;
+
+ case eCSSUnit_Inherit:
+ display->mSpecifiedTransform = parentDisplay->mSpecifiedTransform;
+ conditions.SetUncacheable();
+ break;
+
+ case eCSSUnit_SharedList: {
+ nsCSSValueSharedList* list = transformValue->GetSharedListValue();
+ nsCSSValueList* head = list->mHead;
+ MOZ_ASSERT(head, "transform list must have at least one item");
+ // can get a _None in here from transform animation
+ if (head->mValue.GetUnit() == eCSSUnit_None) {
+ MOZ_ASSERT(head->mNext == nullptr, "none must be alone");
+ display->mSpecifiedTransform = nullptr;
+ } else {
+ display->mSpecifiedTransform = list;
+ }
+ break;
+ }
+
+ default:
+ MOZ_ASSERT(false, "unrecognized transform unit");
+ }
+
+ /* Convert the nsCSSValueList into a will-change bitfield for fast lookup */
+ const nsCSSValue* willChangeValue = aRuleData->ValueForWillChange();
+ switch (willChangeValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_List:
+ case eCSSUnit_ListDep: {
+ display->mWillChange.Clear();
+ display->mWillChangeBitField = 0;
+ for (const nsCSSValueList* item = willChangeValue->GetListValue();
+ item; item = item->mNext)
+ {
+ if (item->mValue.UnitHasStringValue()) {
+ nsAutoString buffer;
+ item->mValue.GetStringValue(buffer);
+ display->mWillChange.AppendElement(buffer);
+
+ if (buffer.EqualsLiteral("transform")) {
+ display->mWillChangeBitField |= NS_STYLE_WILL_CHANGE_TRANSFORM;
+ }
+ if (buffer.EqualsLiteral("opacity")) {
+ display->mWillChangeBitField |= NS_STYLE_WILL_CHANGE_OPACITY;
+ }
+ if (buffer.EqualsLiteral("scroll-position")) {
+ display->mWillChangeBitField |= NS_STYLE_WILL_CHANGE_SCROLL;
+ }
+
+ nsCSSPropertyID prop =
+ nsCSSProps::LookupProperty(buffer, CSSEnabledState::eForAllContent);
+ if (prop != eCSSProperty_UNKNOWN &&
+ prop != eCSSPropertyExtra_variable) {
+ // If the property given is a shorthand, it indicates the expectation
+ // for all the longhands the shorthand expands to.
+ if (nsCSSProps::IsShorthand(prop)) {
+ for (const nsCSSPropertyID* shorthands =
+ nsCSSProps::SubpropertyEntryFor(prop);
+ *shorthands != eCSSProperty_UNKNOWN; ++shorthands) {
+ display->mWillChangeBitField |= GetWillChangeBitFieldFromPropFlags(*shorthands);
+ }
+ } else {
+ display->mWillChangeBitField |= GetWillChangeBitFieldFromPropFlags(prop);
+ }
+ }
+ }
+ }
+ break;
+ }
+
+ case eCSSUnit_Inherit:
+ display->mWillChange = parentDisplay->mWillChange;
+ display->mWillChangeBitField = parentDisplay->mWillChangeBitField;
+ conditions.SetUncacheable();
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_Auto:
+ display->mWillChange.Clear();
+ display->mWillChangeBitField = 0;
+ break;
+
+ default:
+ MOZ_ASSERT(false, "unrecognized will-change unit");
+ }
+
+ // vertical-align: enum, length, percent, calc, inherit
+ const nsCSSValue* verticalAlignValue = aRuleData->ValueForVerticalAlign();
+ if (!SetCoord(*verticalAlignValue, display->mVerticalAlign,
+ parentDisplay->mVerticalAlign,
+ SETCOORD_LPH | SETCOORD_ENUMERATED | SETCOORD_STORE_CALC,
+ aContext, mPresContext, conditions)) {
+ if (eCSSUnit_Initial == verticalAlignValue->GetUnit() ||
+ eCSSUnit_Unset == verticalAlignValue->GetUnit()) {
+ display->mVerticalAlign.SetIntValue(NS_STYLE_VERTICAL_ALIGN_BASELINE,
+ eStyleUnit_Enumerated);
+ }
+ }
+
+ /* Convert -moz-transform-origin. */
+ const nsCSSValue* transformOriginValue =
+ aRuleData->ValueForTransformOrigin();
+ if (transformOriginValue->GetUnit() != eCSSUnit_Null) {
+ const nsCSSValue& valX =
+ transformOriginValue->GetUnit() == eCSSUnit_Triplet ?
+ transformOriginValue->GetTripletValue().mXValue : *transformOriginValue;
+ const nsCSSValue& valY =
+ transformOriginValue->GetUnit() == eCSSUnit_Triplet ?
+ transformOriginValue->GetTripletValue().mYValue : *transformOriginValue;
+ const nsCSSValue& valZ =
+ transformOriginValue->GetUnit() == eCSSUnit_Triplet ?
+ transformOriginValue->GetTripletValue().mZValue : *transformOriginValue;
+
+ mozilla::DebugOnly<bool> cX =
+ SetCoord(valX, display->mTransformOrigin[0],
+ parentDisplay->mTransformOrigin[0],
+ SETCOORD_LPH | SETCOORD_INITIAL_HALF |
+ SETCOORD_BOX_POSITION | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+
+ mozilla::DebugOnly<bool> cY =
+ SetCoord(valY, display->mTransformOrigin[1],
+ parentDisplay->mTransformOrigin[1],
+ SETCOORD_LPH | SETCOORD_INITIAL_HALF |
+ SETCOORD_BOX_POSITION | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+
+ if (valZ.GetUnit() == eCSSUnit_Null) {
+ // Null for the z component means a 0 translation, not
+ // unspecified, as we have already checked the triplet
+ // value for Null.
+ display->mTransformOrigin[2].SetCoordValue(0);
+ } else {
+ mozilla::DebugOnly<bool> cZ =
+ SetCoord(valZ, display->mTransformOrigin[2],
+ parentDisplay->mTransformOrigin[2],
+ SETCOORD_LH | SETCOORD_INITIAL_ZERO | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+ MOZ_ASSERT(cY == cZ, "changed one but not the other");
+ }
+ MOZ_ASSERT(cX == cY, "changed one but not the other");
+ NS_ASSERTION(cX, "Malformed -moz-transform-origin parse!");
+ }
+
+ const nsCSSValue* perspectiveOriginValue =
+ aRuleData->ValueForPerspectiveOrigin();
+ if (perspectiveOriginValue->GetUnit() != eCSSUnit_Null) {
+ mozilla::DebugOnly<bool> result =
+ SetPairCoords(*perspectiveOriginValue,
+ display->mPerspectiveOrigin[0],
+ display->mPerspectiveOrigin[1],
+ parentDisplay->mPerspectiveOrigin[0],
+ parentDisplay->mPerspectiveOrigin[1],
+ SETCOORD_LPH | SETCOORD_INITIAL_HALF |
+ SETCOORD_BOX_POSITION | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+ NS_ASSERTION(result, "Malformed -moz-perspective-origin parse!");
+ }
+
+ SetCoord(*aRuleData->ValueForPerspective(),
+ display->mChildPerspective, parentDisplay->mChildPerspective,
+ SETCOORD_LAH | SETCOORD_INITIAL_NONE | SETCOORD_NONE |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+
+ SetValue(*aRuleData->ValueForBackfaceVisibility(),
+ display->mBackfaceVisibility, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mBackfaceVisibility,
+ NS_STYLE_BACKFACE_VISIBILITY_VISIBLE);
+
+ // transform-style: enum, inherit, initial
+ SetValue(*aRuleData->ValueForTransformStyle(),
+ display->mTransformStyle, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mTransformStyle,
+ NS_STYLE_TRANSFORM_STYLE_FLAT);
+
+ // transform-box: enum, inherit, initial
+ SetValue(*aRuleData->ValueForTransformBox(),
+ display->mTransformBox, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mTransformBox,
+ NS_STYLE_TRANSFORM_BOX_BORDER_BOX);
+
+ // orient: enum, inherit, initial
+ SetValue(*aRuleData->ValueForOrient(),
+ display->mOrient, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentDisplay->mOrient,
+ StyleOrient::Inline);
+
+ // shape-outside: none | [ <basic-shape> || <shape-box> ] | <image>
+ const nsCSSValue* shapeOutsideValue = aRuleData->ValueForShapeOutside();
+ switch (shapeOutsideValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_None:
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ display->mShapeOutside = StyleShapeOutside();
+ break;
+ case eCSSUnit_Inherit:
+ conditions.SetUncacheable();
+ display->mShapeOutside = parentDisplay->mShapeOutside;
+ break;
+ case eCSSUnit_URL: {
+ display->mShapeOutside = StyleShapeOutside();
+ display->mShapeOutside.SetURL(shapeOutsideValue->GetURLStructValue());
+ break;
+ }
+ case eCSSUnit_Array: {
+ display->mShapeOutside = StyleShapeOutside();
+ SetStyleShapeSourceToCSSValue(&display->mShapeOutside, shapeOutsideValue,
+ aContext, mPresContext, conditions);
+ break;
+ }
+ default:
+ MOZ_ASSERT_UNREACHABLE("Unrecognized shape-outside unit!");
+ }
+
+ COMPUTE_END_RESET(Display, display)
+}
+
+const void*
+nsRuleNode::ComputeVisibilityData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_INHERITED(Visibility, visibility, parentVisibility)
+
+ // IMPORTANT: No properties in this struct have lengths in them. We
+ // depend on this since CalcLengthWith can call StyleVisibility()
+ // to get the language for resolving fonts!
+
+ // direction: enum, inherit, initial
+ SetValue(*aRuleData->ValueForDirection(), visibility->mDirection,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentVisibility->mDirection,
+ (GET_BIDI_OPTION_DIRECTION(mPresContext->GetBidi())
+ == IBMBIDI_TEXTDIRECTION_RTL)
+ ? NS_STYLE_DIRECTION_RTL : NS_STYLE_DIRECTION_LTR);
+
+ // visibility: enum, inherit, initial
+ SetValue(*aRuleData->ValueForVisibility(), visibility->mVisible,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentVisibility->mVisible,
+ NS_STYLE_VISIBILITY_VISIBLE);
+
+ // image-rendering: enum, inherit
+ SetValue(*aRuleData->ValueForImageRendering(),
+ visibility->mImageRendering, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentVisibility->mImageRendering,
+ NS_STYLE_IMAGE_RENDERING_AUTO);
+
+ // writing-mode: enum, inherit, initial
+ SetValue(*aRuleData->ValueForWritingMode(), visibility->mWritingMode,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentVisibility->mWritingMode,
+ NS_STYLE_WRITING_MODE_HORIZONTAL_TB);
+
+ // text-orientation: enum, inherit, initial
+ SetValue(*aRuleData->ValueForTextOrientation(), visibility->mTextOrientation,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentVisibility->mTextOrientation,
+ NS_STYLE_TEXT_ORIENTATION_MIXED);
+
+ // image-orientation: enum, inherit, initial
+ const nsCSSValue* orientation = aRuleData->ValueForImageOrientation();
+ if (orientation->GetUnit() == eCSSUnit_Inherit ||
+ orientation->GetUnit() == eCSSUnit_Unset) {
+ conditions.SetUncacheable();
+ visibility->mImageOrientation = parentVisibility->mImageOrientation;
+ } else if (orientation->GetUnit() == eCSSUnit_Initial) {
+ visibility->mImageOrientation = nsStyleImageOrientation();
+ } else if (orientation->IsAngularUnit()) {
+ double angle = orientation->GetAngleValueInRadians();
+ visibility->mImageOrientation =
+ nsStyleImageOrientation::CreateAsAngleAndFlip(angle, false);
+ } else if (orientation->GetUnit() == eCSSUnit_Array) {
+ const nsCSSValue::Array* array = orientation->GetArrayValue();
+ MOZ_ASSERT(array->Item(0).IsAngularUnit(),
+ "First image-orientation value is not an angle");
+ MOZ_ASSERT(array->Item(1).GetUnit() == eCSSUnit_Enumerated &&
+ array->Item(1).GetIntValue() == NS_STYLE_IMAGE_ORIENTATION_FLIP,
+ "Second image-orientation value is not 'flip'");
+ double angle = array->Item(0).GetAngleValueInRadians();
+ visibility->mImageOrientation =
+ nsStyleImageOrientation::CreateAsAngleAndFlip(angle, true);
+
+ } else if (orientation->GetUnit() == eCSSUnit_Enumerated) {
+ switch (orientation->GetIntValue()) {
+ case NS_STYLE_IMAGE_ORIENTATION_FLIP:
+ visibility->mImageOrientation = nsStyleImageOrientation::CreateAsFlip();
+ break;
+ case NS_STYLE_IMAGE_ORIENTATION_FROM_IMAGE:
+ visibility->mImageOrientation = nsStyleImageOrientation::CreateAsFromImage();
+ break;
+ default:
+ NS_NOTREACHED("Invalid image-orientation enumerated value");
+ }
+ } else {
+ MOZ_ASSERT(orientation->GetUnit() == eCSSUnit_Null, "Should be null unit");
+ }
+
+ SetValue(*aRuleData->ValueForColorAdjust(), visibility->mColorAdjust,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentVisibility->mColorAdjust,
+ NS_STYLE_COLOR_ADJUST_ECONOMY);
+
+ COMPUTE_END_INHERITED(Visibility, visibility)
+}
+
+const void*
+nsRuleNode::ComputeColorData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_INHERITED(Color, color, parentColor)
+
+ // color: color, string, inherit
+ // Special case for currentColor. According to CSS3, setting color to 'currentColor'
+ // should behave as if it is inherited
+ const nsCSSValue* colorValue = aRuleData->ValueForColor();
+ if ((colorValue->GetUnit() == eCSSUnit_EnumColor &&
+ colorValue->GetIntValue() == NS_COLOR_CURRENTCOLOR) ||
+ colorValue->GetUnit() == eCSSUnit_Unset) {
+ color->mColor = parentColor->mColor;
+ conditions.SetUncacheable();
+ }
+ else if (colorValue->GetUnit() == eCSSUnit_Initial) {
+ color->mColor = mPresContext->DefaultColor();
+ }
+ else {
+ SetColor(*colorValue, parentColor->mColor, mPresContext, aContext,
+ color->mColor, conditions);
+ }
+
+ COMPUTE_END_INHERITED(Color, color)
+}
+
+// information about how to compute values for background-* properties
+template <class SpecifiedValueItem, class ComputedValueItem>
+struct BackgroundItemComputer {
+};
+
+template <>
+struct BackgroundItemComputer<nsCSSValueList, uint8_t>
+{
+ static void ComputeValue(nsStyleContext* aStyleContext,
+ const nsCSSValueList* aSpecifiedValue,
+ uint8_t& aComputedValue,
+ RuleNodeCacheConditions& aConditions)
+ {
+ SetValue(aSpecifiedValue->mValue, aComputedValue, aConditions,
+ SETVAL_ENUMERATED, uint8_t(0), 0);
+ }
+};
+
+template <>
+struct BackgroundItemComputer<nsCSSValuePairList, nsStyleImageLayers::Repeat>
+{
+ static void ComputeValue(nsStyleContext* aStyleContext,
+ const nsCSSValuePairList* aSpecifiedValue,
+ nsStyleImageLayers::Repeat& aComputedValue,
+ RuleNodeCacheConditions& aConditions)
+ {
+ NS_ASSERTION(aSpecifiedValue->mXValue.GetUnit() == eCSSUnit_Enumerated &&
+ (aSpecifiedValue->mYValue.GetUnit() == eCSSUnit_Enumerated ||
+ aSpecifiedValue->mYValue.GetUnit() == eCSSUnit_Null),
+ "Invalid unit");
+
+ bool hasContraction = true;
+ uint8_t value = aSpecifiedValue->mXValue.GetIntValue();
+ switch (value) {
+ case NS_STYLE_IMAGELAYER_REPEAT_REPEAT_X:
+ aComputedValue.mXRepeat = NS_STYLE_IMAGELAYER_REPEAT_REPEAT;
+ aComputedValue.mYRepeat = NS_STYLE_IMAGELAYER_REPEAT_NO_REPEAT;
+ break;
+ case NS_STYLE_IMAGELAYER_REPEAT_REPEAT_Y:
+ aComputedValue.mXRepeat = NS_STYLE_IMAGELAYER_REPEAT_NO_REPEAT;
+ aComputedValue.mYRepeat = NS_STYLE_IMAGELAYER_REPEAT_REPEAT;
+ break;
+ default:
+ NS_ASSERTION(value == NS_STYLE_IMAGELAYER_REPEAT_NO_REPEAT ||
+ value == NS_STYLE_IMAGELAYER_REPEAT_REPEAT ||
+ value == NS_STYLE_IMAGELAYER_REPEAT_SPACE ||
+ value == NS_STYLE_IMAGELAYER_REPEAT_ROUND, "Unexpected value");
+ aComputedValue.mXRepeat = value;
+ hasContraction = false;
+ break;
+ }
+
+ if (hasContraction) {
+ NS_ASSERTION(aSpecifiedValue->mYValue.GetUnit() == eCSSUnit_Null,
+ "Invalid unit.");
+ return;
+ }
+
+ switch (aSpecifiedValue->mYValue.GetUnit()) {
+ case eCSSUnit_Null:
+ aComputedValue.mYRepeat = aComputedValue.mXRepeat;
+ break;
+ case eCSSUnit_Enumerated:
+ value = aSpecifiedValue->mYValue.GetIntValue();
+ NS_ASSERTION(value == NS_STYLE_IMAGELAYER_REPEAT_NO_REPEAT ||
+ value == NS_STYLE_IMAGELAYER_REPEAT_REPEAT ||
+ value == NS_STYLE_IMAGELAYER_REPEAT_SPACE ||
+ value == NS_STYLE_IMAGELAYER_REPEAT_ROUND, "Unexpected value");
+ aComputedValue.mYRepeat = value;
+ break;
+ default:
+ NS_NOTREACHED("Unexpected CSS value");
+ break;
+ }
+ }
+};
+
+template <>
+struct BackgroundItemComputer<nsCSSValueList, nsStyleImage>
+{
+ static void ComputeValue(nsStyleContext* aStyleContext,
+ const nsCSSValueList* aSpecifiedValue,
+ nsStyleImage& aComputedValue,
+ RuleNodeCacheConditions& aConditions)
+ {
+ SetStyleImage(aStyleContext, aSpecifiedValue->mValue, aComputedValue,
+ aConditions);
+ }
+};
+
+template <>
+struct BackgroundItemComputer<nsCSSValueList, RefPtr<css::URLValueData>>
+{
+ static void ComputeValue(nsStyleContext* aStyleContext,
+ const nsCSSValueList* aSpecifiedValue,
+ RefPtr<css::URLValueData>& aComputedValue,
+ RuleNodeCacheConditions& aConditions)
+ {
+ switch (aSpecifiedValue->mValue.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_URL:
+ aComputedValue = aSpecifiedValue->mValue.GetURLStructValue();
+ break;
+ case eCSSUnit_Image:
+ aComputedValue = aSpecifiedValue->mValue.GetImageStructValue();
+ break;
+ default:
+ aComputedValue = nullptr;
+ break;
+ }
+ }
+};
+
+/* Helper function for ComputePositionValue.
+ * This function computes a single PositionCoord from two nsCSSValue objects,
+ * which represent an edge and an offset from that edge.
+ */
+static void
+ComputePositionCoord(nsStyleContext* aStyleContext,
+ const nsCSSValue& aEdge,
+ const nsCSSValue& aOffset,
+ Position::Coord* aResult,
+ RuleNodeCacheConditions& aConditions)
+{
+ if (eCSSUnit_Percent == aOffset.GetUnit()) {
+ aResult->mLength = 0;
+ aResult->mPercent = aOffset.GetPercentValue();
+ aResult->mHasPercent = true;
+ } else if (aOffset.IsLengthUnit()) {
+ aResult->mLength = CalcLength(aOffset, aStyleContext,
+ aStyleContext->PresContext(),
+ aConditions);
+ aResult->mPercent = 0.0f;
+ aResult->mHasPercent = false;
+ } else if (aOffset.IsCalcUnit()) {
+ LengthPercentPairCalcOps ops(aStyleContext,
+ aStyleContext->PresContext(),
+ aConditions);
+ nsRuleNode::ComputedCalc vals = ComputeCalc(aOffset, ops);
+ aResult->mLength = vals.mLength;
+ aResult->mPercent = vals.mPercent;
+ aResult->mHasPercent = ops.mHasPercent;
+ } else {
+ aResult->mLength = 0;
+ aResult->mPercent = 0.0f;
+ aResult->mHasPercent = false;
+ NS_ASSERTION(aOffset.GetUnit() == eCSSUnit_Null, "unexpected unit");
+ }
+
+ if (eCSSUnit_Enumerated == aEdge.GetUnit()) {
+ int sign;
+ if (aEdge.GetIntValue() & (NS_STYLE_IMAGELAYER_POSITION_BOTTOM |
+ NS_STYLE_IMAGELAYER_POSITION_RIGHT)) {
+ sign = -1;
+ } else {
+ sign = 1;
+ }
+ aResult->mPercent = GetFloatFromBoxPosition(aEdge.GetIntValue()) +
+ sign * aResult->mPercent;
+ aResult->mLength = sign * aResult->mLength;
+ aResult->mHasPercent = true;
+ } else {
+ NS_ASSERTION(eCSSUnit_Null == aEdge.GetUnit(), "unexpected unit");
+ }
+}
+
+/* Helper function to convert a CSS <position> specified value into its
+ * computed-style form. */
+static void
+ComputePositionValue(nsStyleContext* aStyleContext,
+ const nsCSSValue& aValue,
+ Position& aComputedValue,
+ RuleNodeCacheConditions& aConditions)
+{
+ NS_ASSERTION(aValue.GetUnit() == eCSSUnit_Array,
+ "unexpected unit for CSS <position> value");
+
+ RefPtr<nsCSSValue::Array> positionArray = aValue.GetArrayValue();
+ NS_ASSERTION(positionArray->Count() == 4,
+ "unexpected number of values in CSS <position> value");
+
+ const nsCSSValue &xEdge = positionArray->Item(0);
+ const nsCSSValue &xOffset = positionArray->Item(1);
+ const nsCSSValue &yEdge = positionArray->Item(2);
+ const nsCSSValue &yOffset = positionArray->Item(3);
+
+ NS_ASSERTION((eCSSUnit_Enumerated == xEdge.GetUnit() ||
+ eCSSUnit_Null == xEdge.GetUnit()) &&
+ (eCSSUnit_Enumerated == yEdge.GetUnit() ||
+ eCSSUnit_Null == yEdge.GetUnit()) &&
+ eCSSUnit_Enumerated != xOffset.GetUnit() &&
+ eCSSUnit_Enumerated != yOffset.GetUnit(),
+ "Invalid background position");
+
+ ComputePositionCoord(aStyleContext, xEdge, xOffset,
+ &aComputedValue.mXPosition,
+ aConditions);
+
+ ComputePositionCoord(aStyleContext, yEdge, yOffset,
+ &aComputedValue.mYPosition,
+ aConditions);
+}
+
+/* Helper function to convert the -x or -y part of a CSS <position> specified
+ * value into its computed-style form. */
+static void
+ComputePositionCoordValue(nsStyleContext* aStyleContext,
+ const nsCSSValue& aValue,
+ Position::Coord& aComputedValue,
+ RuleNodeCacheConditions& aConditions)
+{
+ NS_ASSERTION(aValue.GetUnit() == eCSSUnit_Array,
+ "unexpected unit for position coord value");
+
+ RefPtr<nsCSSValue::Array> positionArray = aValue.GetArrayValue();
+ NS_ASSERTION(positionArray->Count() == 2,
+ "unexpected number of values, expecting one edge and one offset");
+
+ const nsCSSValue &edge = positionArray->Item(0);
+ const nsCSSValue &offset = positionArray->Item(1);
+
+ NS_ASSERTION((eCSSUnit_Enumerated == edge.GetUnit() ||
+ eCSSUnit_Null == edge.GetUnit()) &&
+ eCSSUnit_Enumerated != offset.GetUnit(),
+ "Invalid background position");
+
+ ComputePositionCoord(aStyleContext, edge, offset,
+ &aComputedValue,
+ aConditions);
+}
+
+struct BackgroundSizeAxis {
+ nsCSSValue nsCSSValuePairList::* specified;
+ nsStyleImageLayers::Size::Dimension nsStyleImageLayers::Size::* result;
+ uint8_t nsStyleImageLayers::Size::* type;
+};
+
+static const BackgroundSizeAxis gBGSizeAxes[] = {
+ { &nsCSSValuePairList::mXValue,
+ &nsStyleImageLayers::Size::mWidth,
+ &nsStyleImageLayers::Size::mWidthType },
+ { &nsCSSValuePairList::mYValue,
+ &nsStyleImageLayers::Size::mHeight,
+ &nsStyleImageLayers::Size::mHeightType }
+};
+
+template <>
+struct BackgroundItemComputer<nsCSSValuePairList, nsStyleImageLayers::Size>
+{
+ static void ComputeValue(nsStyleContext* aStyleContext,
+ const nsCSSValuePairList* aSpecifiedValue,
+ nsStyleImageLayers::Size& aComputedValue,
+ RuleNodeCacheConditions& aConditions)
+ {
+ nsStyleImageLayers::Size &size = aComputedValue;
+ for (const BackgroundSizeAxis *axis = gBGSizeAxes,
+ *axis_end = ArrayEnd(gBGSizeAxes);
+ axis < axis_end; ++axis) {
+ const nsCSSValue &specified = aSpecifiedValue->*(axis->specified);
+ if (eCSSUnit_Auto == specified.GetUnit()) {
+ size.*(axis->type) = nsStyleImageLayers::Size::eAuto;
+ }
+ else if (eCSSUnit_Enumerated == specified.GetUnit()) {
+ static_assert(nsStyleImageLayers::Size::eContain ==
+ NS_STYLE_IMAGELAYER_SIZE_CONTAIN &&
+ nsStyleImageLayers::Size::eCover ==
+ NS_STYLE_IMAGELAYER_SIZE_COVER,
+ "background size constants out of sync");
+ MOZ_ASSERT(specified.GetIntValue() == NS_STYLE_IMAGELAYER_SIZE_CONTAIN ||
+ specified.GetIntValue() == NS_STYLE_IMAGELAYER_SIZE_COVER,
+ "invalid enumerated value for size coordinate");
+ size.*(axis->type) = specified.GetIntValue();
+ }
+ else if (eCSSUnit_Null == specified.GetUnit()) {
+ MOZ_ASSERT(axis == gBGSizeAxes + 1,
+ "null allowed only as height value, and only "
+ "for contain/cover/initial/inherit");
+#ifdef DEBUG
+ {
+ const nsCSSValue &widthValue = aSpecifiedValue->mXValue;
+ MOZ_ASSERT(widthValue.GetUnit() != eCSSUnit_Inherit &&
+ widthValue.GetUnit() != eCSSUnit_Initial &&
+ widthValue.GetUnit() != eCSSUnit_Unset,
+ "initial/inherit/unset should already have been handled");
+ MOZ_ASSERT(widthValue.GetUnit() == eCSSUnit_Enumerated &&
+ (widthValue.GetIntValue() == NS_STYLE_IMAGELAYER_SIZE_CONTAIN ||
+ widthValue.GetIntValue() == NS_STYLE_IMAGELAYER_SIZE_COVER),
+ "null height value not corresponding to allowable "
+ "non-null width value");
+ }
+#endif
+ size.*(axis->type) = size.mWidthType;
+ }
+ else if (eCSSUnit_Percent == specified.GetUnit()) {
+ (size.*(axis->result)).mLength = 0;
+ (size.*(axis->result)).mPercent = specified.GetPercentValue();
+ (size.*(axis->result)).mHasPercent = true;
+ size.*(axis->type) = nsStyleImageLayers::Size::eLengthPercentage;
+ }
+ else if (specified.IsLengthUnit()) {
+ (size.*(axis->result)).mLength =
+ CalcLength(specified, aStyleContext, aStyleContext->PresContext(),
+ aConditions);
+ (size.*(axis->result)).mPercent = 0.0f;
+ (size.*(axis->result)).mHasPercent = false;
+ size.*(axis->type) = nsStyleImageLayers::Size::eLengthPercentage;
+ } else {
+ MOZ_ASSERT(specified.IsCalcUnit(), "unexpected unit");
+ LengthPercentPairCalcOps ops(aStyleContext,
+ aStyleContext->PresContext(),
+ aConditions);
+ nsRuleNode::ComputedCalc vals = ComputeCalc(specified, ops);
+ (size.*(axis->result)).mLength = vals.mLength;
+ (size.*(axis->result)).mPercent = vals.mPercent;
+ (size.*(axis->result)).mHasPercent = ops.mHasPercent;
+ size.*(axis->type) = nsStyleImageLayers::Size::eLengthPercentage;
+ }
+ }
+
+ MOZ_ASSERT(size.mWidthType < nsStyleImageLayers::Size::eDimensionType_COUNT,
+ "bad width type");
+ MOZ_ASSERT(size.mHeightType < nsStyleImageLayers::Size::eDimensionType_COUNT,
+ "bad height type");
+ MOZ_ASSERT((size.mWidthType != nsStyleImageLayers::Size::eContain &&
+ size.mWidthType != nsStyleImageLayers::Size::eCover) ||
+ size.mWidthType == size.mHeightType,
+ "contain/cover apply to both dimensions or to neither");
+ }
+};
+
+template <class ComputedValueItem>
+static void
+SetImageLayerList(nsStyleContext* aStyleContext,
+ const nsCSSValue& aValue,
+ nsStyleAutoArray<nsStyleImageLayers::Layer>& aLayers,
+ const nsStyleAutoArray<nsStyleImageLayers::Layer>& aParentLayers,
+ ComputedValueItem nsStyleImageLayers::Layer::* aResultLocation,
+ ComputedValueItem aInitialValue,
+ uint32_t aParentItemCount,
+ uint32_t& aItemCount,
+ uint32_t& aMaxItemCount,
+ bool& aRebuild,
+ RuleNodeCacheConditions& aConditions)
+{
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Inherit:
+ aRebuild = true;
+ aConditions.SetUncacheable();
+ aLayers.EnsureLengthAtLeast(aParentItemCount);
+ aItemCount = aParentItemCount;
+ for (uint32_t i = 0; i < aParentItemCount; ++i) {
+ aLayers[i].*aResultLocation = aParentLayers[i].*aResultLocation;
+ }
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ aRebuild = true;
+ aItemCount = 1;
+ aLayers[0].*aResultLocation = aInitialValue;
+ break;
+
+ case eCSSUnit_List:
+ case eCSSUnit_ListDep: {
+ aRebuild = true;
+ aItemCount = 0;
+ const nsCSSValueList* item = aValue.GetListValue();
+ do {
+ NS_ASSERTION(item->mValue.GetUnit() != eCSSUnit_Null &&
+ item->mValue.GetUnit() != eCSSUnit_Inherit &&
+ item->mValue.GetUnit() != eCSSUnit_Initial &&
+ item->mValue.GetUnit() != eCSSUnit_Unset,
+ "unexpected unit");
+ ++aItemCount;
+ aLayers.EnsureLengthAtLeast(aItemCount);
+ BackgroundItemComputer<nsCSSValueList, ComputedValueItem>
+ ::ComputeValue(aStyleContext, item,
+ aLayers[aItemCount-1].*aResultLocation,
+ aConditions);
+ item = item->mNext;
+ } while (item);
+ break;
+ }
+
+ default:
+ MOZ_ASSERT(false, "unexpected unit");
+ }
+
+ if (aItemCount > aMaxItemCount)
+ aMaxItemCount = aItemCount;
+}
+
+// The same as SetImageLayerList, but for values stored in
+// layer.mPosition.*aResultLocation instead of layer.*aResultLocation.
+// This code is duplicated because it would be annoying to make
+// SetImageLayerList generic enough to handle both cases.
+static void
+SetImageLayerPositionCoordList(
+ nsStyleContext* aStyleContext,
+ const nsCSSValue& aValue,
+ nsStyleAutoArray<nsStyleImageLayers::Layer>& aLayers,
+ const nsStyleAutoArray<nsStyleImageLayers::Layer>& aParentLayers,
+ Position::Coord
+ Position::* aResultLocation,
+ Position::Coord aInitialValue,
+ uint32_t aParentItemCount,
+ uint32_t& aItemCount,
+ uint32_t& aMaxItemCount,
+ bool& aRebuild,
+ RuleNodeCacheConditions& aConditions)
+{
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Inherit:
+ aRebuild = true;
+ aConditions.SetUncacheable();
+ aLayers.EnsureLengthAtLeast(aParentItemCount);
+ aItemCount = aParentItemCount;
+ for (uint32_t i = 0; i < aParentItemCount; ++i) {
+ aLayers[i].mPosition.*aResultLocation = aParentLayers[i].mPosition.*aResultLocation;
+ }
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ aRebuild = true;
+ aItemCount = 1;
+ aLayers[0].mPosition.*aResultLocation = aInitialValue;
+ break;
+
+ case eCSSUnit_List:
+ case eCSSUnit_ListDep: {
+ aRebuild = true;
+ aItemCount = 0;
+ const nsCSSValueList* item = aValue.GetListValue();
+ do {
+ NS_ASSERTION(item->mValue.GetUnit() != eCSSUnit_Null &&
+ item->mValue.GetUnit() != eCSSUnit_Inherit &&
+ item->mValue.GetUnit() != eCSSUnit_Initial &&
+ item->mValue.GetUnit() != eCSSUnit_Unset,
+ "unexpected unit");
+ ++aItemCount;
+ aLayers.EnsureLengthAtLeast(aItemCount);
+
+ ComputePositionCoordValue(aStyleContext, item->mValue,
+ aLayers[aItemCount-1].mPosition.*aResultLocation,
+ aConditions);
+ item = item->mNext;
+ } while (item);
+ break;
+ }
+
+ default:
+ MOZ_ASSERT(false, "unexpected unit");
+ }
+
+ if (aItemCount > aMaxItemCount)
+ aMaxItemCount = aItemCount;
+}
+
+template <class ComputedValueItem>
+static void
+SetImageLayerPairList(nsStyleContext* aStyleContext,
+ const nsCSSValue& aValue,
+ nsStyleAutoArray<nsStyleImageLayers::Layer>& aLayers,
+ const nsStyleAutoArray<nsStyleImageLayers::Layer>& aParentLayers,
+ ComputedValueItem nsStyleImageLayers::Layer::*
+ aResultLocation,
+ ComputedValueItem aInitialValue,
+ uint32_t aParentItemCount,
+ uint32_t& aItemCount,
+ uint32_t& aMaxItemCount,
+ bool& aRebuild,
+ RuleNodeCacheConditions& aConditions)
+{
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Inherit:
+ aRebuild = true;
+ aConditions.SetUncacheable();
+ aLayers.EnsureLengthAtLeast(aParentItemCount);
+ aItemCount = aParentItemCount;
+ for (uint32_t i = 0; i < aParentItemCount; ++i) {
+ aLayers[i].*aResultLocation = aParentLayers[i].*aResultLocation;
+ }
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ aRebuild = true;
+ aItemCount = 1;
+ aLayers[0].*aResultLocation = aInitialValue;
+ break;
+
+ case eCSSUnit_PairList:
+ case eCSSUnit_PairListDep: {
+ aRebuild = true;
+ aItemCount = 0;
+ const nsCSSValuePairList* item = aValue.GetPairListValue();
+ do {
+ NS_ASSERTION(item->mXValue.GetUnit() != eCSSUnit_Inherit &&
+ item->mXValue.GetUnit() != eCSSUnit_Initial &&
+ item->mXValue.GetUnit() != eCSSUnit_Unset &&
+ item->mYValue.GetUnit() != eCSSUnit_Inherit &&
+ item->mYValue.GetUnit() != eCSSUnit_Initial &&
+ item->mYValue.GetUnit() != eCSSUnit_Unset,
+ "unexpected unit");
+ ++aItemCount;
+ aLayers.EnsureLengthAtLeast(aItemCount);
+ BackgroundItemComputer<nsCSSValuePairList, ComputedValueItem>
+ ::ComputeValue(aStyleContext, item,
+ aLayers[aItemCount-1].*aResultLocation,
+ aConditions);
+ item = item->mNext;
+ } while (item);
+ break;
+ }
+
+ default:
+ MOZ_ASSERT(false, "unexpected unit");
+ }
+
+ if (aItemCount > aMaxItemCount)
+ aMaxItemCount = aItemCount;
+}
+
+template <class ComputedValueItem>
+static void
+FillImageLayerList(
+ nsStyleAutoArray<nsStyleImageLayers::Layer>& aLayers,
+ ComputedValueItem nsStyleImageLayers::Layer::* aResultLocation,
+ uint32_t aItemCount, uint32_t aFillCount)
+{
+ NS_PRECONDITION(aFillCount <= aLayers.Length(), "unexpected array length");
+ for (uint32_t sourceLayer = 0, destLayer = aItemCount;
+ destLayer < aFillCount;
+ ++sourceLayer, ++destLayer) {
+ aLayers[destLayer].*aResultLocation =
+ aLayers[sourceLayer].*aResultLocation;
+ }
+}
+
+// The same as FillImageLayerList, but for values stored in
+// layer.mPosition.*aResultLocation instead of layer.*aResultLocation.
+static void
+FillImageLayerPositionCoordList(
+ nsStyleAutoArray<nsStyleImageLayers::Layer>& aLayers,
+ Position::Coord
+ Position::* aResultLocation,
+ uint32_t aItemCount, uint32_t aFillCount)
+{
+ NS_PRECONDITION(aFillCount <= aLayers.Length(), "unexpected array length");
+ for (uint32_t sourceLayer = 0, destLayer = aItemCount;
+ destLayer < aFillCount;
+ ++sourceLayer, ++destLayer) {
+ aLayers[destLayer].mPosition.*aResultLocation =
+ aLayers[sourceLayer].mPosition.*aResultLocation;
+ }
+}
+
+/* static */
+void
+nsRuleNode::FillAllBackgroundLists(nsStyleImageLayers& aImage,
+ uint32_t aMaxItemCount)
+{
+ // Delete any extra items. We need to keep layers in which any
+ // property was specified.
+ aImage.mLayers.TruncateLengthNonZero(aMaxItemCount);
+
+ uint32_t fillCount = aImage.mImageCount;
+ FillImageLayerList(aImage.mLayers,
+ &nsStyleImageLayers::Layer::mImage,
+ aImage.mImageCount, fillCount);
+ FillImageLayerList(aImage.mLayers,
+ &nsStyleImageLayers::Layer::mRepeat,
+ aImage.mRepeatCount, fillCount);
+ FillImageLayerList(aImage.mLayers,
+ &nsStyleImageLayers::Layer::mAttachment,
+ aImage.mAttachmentCount, fillCount);
+ FillImageLayerList(aImage.mLayers,
+ &nsStyleImageLayers::Layer::mClip,
+ aImage.mClipCount, fillCount);
+ FillImageLayerList(aImage.mLayers,
+ &nsStyleImageLayers::Layer::mBlendMode,
+ aImage.mBlendModeCount, fillCount);
+ FillImageLayerList(aImage.mLayers,
+ &nsStyleImageLayers::Layer::mOrigin,
+ aImage.mOriginCount, fillCount);
+ FillImageLayerPositionCoordList(aImage.mLayers,
+ &Position::mXPosition,
+ aImage.mPositionXCount, fillCount);
+ FillImageLayerPositionCoordList(aImage.mLayers,
+ &Position::mYPosition,
+ aImage.mPositionYCount, fillCount);
+ FillImageLayerList(aImage.mLayers,
+ &nsStyleImageLayers::Layer::mSize,
+ aImage.mSizeCount, fillCount);
+}
+
+const void*
+nsRuleNode::ComputeBackgroundData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(Background, bg, parentBG)
+
+ // background-color: color, string, inherit
+ const nsCSSValue* backColorValue = aRuleData->ValueForBackgroundColor();
+ if (eCSSUnit_Initial == backColorValue->GetUnit() ||
+ eCSSUnit_Unset == backColorValue->GetUnit()) {
+ bg->mBackgroundColor = NS_RGBA(0, 0, 0, 0);
+ } else if (!SetColor(*backColorValue, parentBG->mBackgroundColor,
+ mPresContext, aContext, bg->mBackgroundColor,
+ conditions)) {
+ NS_ASSERTION(eCSSUnit_Null == backColorValue->GetUnit(),
+ "unexpected color unit");
+ }
+
+ uint32_t maxItemCount = 1;
+ bool rebuild = false;
+
+ // background-image: url (stored as image), none, inherit [list]
+ nsStyleImage initialImage;
+ SetImageLayerList(aContext, *aRuleData->ValueForBackgroundImage(),
+ bg->mImage.mLayers,
+ parentBG->mImage.mLayers,
+ &nsStyleImageLayers::Layer::mImage,
+ initialImage, parentBG->mImage.mImageCount,
+ bg->mImage.mImageCount,
+ maxItemCount, rebuild, conditions);
+
+ // background-repeat: enum, inherit, initial [pair list]
+ nsStyleImageLayers::Repeat initialRepeat;
+ initialRepeat.SetInitialValues();
+ SetImageLayerPairList(aContext, *aRuleData->ValueForBackgroundRepeat(),
+ bg->mImage.mLayers,
+ parentBG->mImage.mLayers,
+ &nsStyleImageLayers::Layer::mRepeat,
+ initialRepeat, parentBG->mImage.mRepeatCount,
+ bg->mImage.mRepeatCount, maxItemCount, rebuild,
+ conditions);
+
+ // background-attachment: enum, inherit, initial [list]
+ SetImageLayerList(aContext, *aRuleData->ValueForBackgroundAttachment(),
+ bg->mImage.mLayers, parentBG->mImage.mLayers,
+ &nsStyleImageLayers::Layer::mAttachment,
+ uint8_t(NS_STYLE_IMAGELAYER_ATTACHMENT_SCROLL),
+ parentBG->mImage.mAttachmentCount,
+ bg->mImage.mAttachmentCount, maxItemCount, rebuild,
+ conditions);
+
+ // background-clip: enum, inherit, initial [list]
+ SetImageLayerList(aContext, *aRuleData->ValueForBackgroundClip(),
+ bg->mImage.mLayers,
+ parentBG->mImage.mLayers,
+ &nsStyleImageLayers::Layer::mClip,
+ uint8_t(NS_STYLE_IMAGELAYER_CLIP_BORDER),
+ parentBG->mImage.mClipCount,
+ bg->mImage.mClipCount, maxItemCount, rebuild, conditions);
+
+ // background-blend-mode: enum, inherit, initial [list]
+ SetImageLayerList(aContext, *aRuleData->ValueForBackgroundBlendMode(),
+ bg->mImage.mLayers,
+ parentBG->mImage.mLayers,
+ &nsStyleImageLayers::Layer::mBlendMode,
+ uint8_t(NS_STYLE_BLEND_NORMAL),
+ parentBG->mImage.mBlendModeCount,
+ bg->mImage.mBlendModeCount, maxItemCount, rebuild,
+ conditions);
+
+ // background-origin: enum, inherit, initial [list]
+ SetImageLayerList(aContext, *aRuleData->ValueForBackgroundOrigin(),
+ bg->mImage.mLayers,
+ parentBG->mImage.mLayers,
+ &nsStyleImageLayers::Layer::mOrigin,
+ uint8_t(NS_STYLE_IMAGELAYER_ORIGIN_PADDING),
+ parentBG->mImage.mOriginCount,
+ bg->mImage.mOriginCount, maxItemCount, rebuild,
+ conditions);
+
+ // background-position-x/y: enum, length, percent (flags), inherit [list]
+ Position::Coord initialPositionCoord;
+ initialPositionCoord.mPercent = 0.0f;
+ initialPositionCoord.mLength = 0;
+ initialPositionCoord.mHasPercent = true;
+
+ SetImageLayerPositionCoordList(
+ aContext, *aRuleData->ValueForBackgroundPositionX(),
+ bg->mImage.mLayers,
+ parentBG->mImage.mLayers,
+ &Position::mXPosition,
+ initialPositionCoord, parentBG->mImage.mPositionXCount,
+ bg->mImage.mPositionXCount, maxItemCount, rebuild,
+ conditions);
+ SetImageLayerPositionCoordList(
+ aContext, *aRuleData->ValueForBackgroundPositionY(),
+ bg->mImage.mLayers,
+ parentBG->mImage.mLayers,
+ &Position::mYPosition,
+ initialPositionCoord, parentBG->mImage.mPositionYCount,
+ bg->mImage.mPositionYCount, maxItemCount, rebuild,
+ conditions);
+
+ // background-size: enum, length, auto, inherit, initial [pair list]
+ nsStyleImageLayers::Size initialSize;
+ initialSize.SetInitialValues();
+ SetImageLayerPairList(aContext, *aRuleData->ValueForBackgroundSize(),
+ bg->mImage.mLayers,
+ parentBG->mImage.mLayers,
+ &nsStyleImageLayers::Layer::mSize,
+ initialSize, parentBG->mImage.mSizeCount,
+ bg->mImage.mSizeCount, maxItemCount, rebuild,
+ conditions);
+
+ if (rebuild) {
+ FillAllBackgroundLists(bg->mImage, maxItemCount);
+ }
+
+ COMPUTE_END_RESET(Background, bg)
+}
+
+const void*
+nsRuleNode::ComputeMarginData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(Margin, margin, parentMargin)
+
+ // margin: length, percent, calc, inherit
+ const nsCSSPropertyID* subprops =
+ nsCSSProps::SubpropertyEntryFor(eCSSProperty_margin);
+ nsStyleCoord coord;
+ NS_FOR_CSS_SIDES(side) {
+ nsStyleCoord parentCoord = parentMargin->mMargin.Get(side);
+ if (SetCoord(*aRuleData->ValueFor(subprops[side]),
+ coord, parentCoord,
+ SETCOORD_LPAH | SETCOORD_INITIAL_ZERO | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions)) {
+ margin->mMargin.Set(side, coord);
+ }
+ }
+
+ COMPUTE_END_RESET(Margin, margin)
+}
+
+static void
+SetBorderImageRect(const nsCSSValue& aValue,
+ /** outparam */ nsCSSRect& aRect)
+{
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Null:
+ aRect.Reset();
+ break;
+ case eCSSUnit_Rect:
+ aRect = aValue.GetRectValue();
+ break;
+ case eCSSUnit_Inherit:
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ aRect.SetAllSidesTo(aValue);
+ break;
+ default:
+ NS_ASSERTION(false, "Unexpected border image value for rect.");
+ }
+}
+
+static void
+SetBorderImagePair(const nsCSSValue& aValue,
+ /** outparam */ nsCSSValuePair& aPair)
+{
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Null:
+ aPair.Reset();
+ break;
+ case eCSSUnit_Pair:
+ aPair = aValue.GetPairValue();
+ break;
+ case eCSSUnit_Inherit:
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ aPair.SetBothValuesTo(aValue);
+ break;
+ default:
+ NS_ASSERTION(false, "Unexpected border image value for pair.");
+ }
+}
+
+static void
+SetBorderImageSlice(const nsCSSValue& aValue,
+ /** outparam */ nsCSSValue& aSlice,
+ /** outparam */ nsCSSValue& aFill)
+{
+ const nsCSSValueList* valueList;
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Null:
+ aSlice.Reset();
+ aFill.Reset();
+ break;
+ case eCSSUnit_List:
+ // Get slice dimensions.
+ valueList = aValue.GetListValue();
+ aSlice = valueList->mValue;
+
+ // Get "fill" keyword.
+ valueList = valueList->mNext;
+ if (valueList) {
+ aFill = valueList->mValue;
+ } else {
+ aFill.SetInitialValue();
+ }
+ break;
+ case eCSSUnit_Inherit:
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ aSlice = aValue;
+ aFill = aValue;
+ break;
+ default:
+ NS_ASSERTION(false, "Unexpected border image value for pair.");
+ }
+}
+
+const void*
+nsRuleNode::ComputeBorderData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(Border, border, parentBorder)
+
+ // box-decoration-break: enum, inherit, initial
+ SetValue(*aRuleData->ValueForBoxDecorationBreak(),
+ border->mBoxDecorationBreak, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentBorder->mBoxDecorationBreak,
+ StyleBoxDecorationBreak::Slice);
+
+ // border-width, border-*-width: length, enum, inherit
+ nsStyleCoord coord;
+ {
+ const nsCSSPropertyID* subprops =
+ nsCSSProps::SubpropertyEntryFor(eCSSProperty_border_width);
+ NS_FOR_CSS_SIDES(side) {
+ const nsCSSValue& value = *aRuleData->ValueFor(subprops[side]);
+ NS_ASSERTION(eCSSUnit_Percent != value.GetUnit(),
+ "Percentage borders not implemented yet "
+ "If implementing, make sure to fix all consumers of "
+ "nsStyleBorder, the IsPercentageAwareChild method, "
+ "the nsAbsoluteContainingBlock::FrameDependsOnContainer "
+ "method, the "
+ "nsLineLayout::IsPercentageAwareReplacedElement method "
+ "and probably some other places");
+ if (eCSSUnit_Enumerated == value.GetUnit()) {
+ NS_ASSERTION(value.GetIntValue() == NS_STYLE_BORDER_WIDTH_THIN ||
+ value.GetIntValue() == NS_STYLE_BORDER_WIDTH_MEDIUM ||
+ value.GetIntValue() == NS_STYLE_BORDER_WIDTH_THICK,
+ "Unexpected enum value");
+ border->SetBorderWidth(side,
+ (mPresContext->GetBorderWidthTable())[value.GetIntValue()]);
+ }
+ // OK to pass bad aParentCoord since we're not passing SETCOORD_INHERIT
+ else if (SetCoord(value, coord, nsStyleCoord(),
+ SETCOORD_LENGTH | SETCOORD_CALC_LENGTH_ONLY,
+ aContext, mPresContext, conditions)) {
+ NS_ASSERTION(coord.GetUnit() == eStyleUnit_Coord, "unexpected unit");
+ // clamp negative calc() to 0.
+ border->SetBorderWidth(side, std::max(coord.GetCoordValue(), 0));
+ }
+ else if (eCSSUnit_Inherit == value.GetUnit()) {
+ conditions.SetUncacheable();
+ border->SetBorderWidth(side,
+ parentBorder->GetComputedBorder().Side(side));
+ }
+ else if (eCSSUnit_Initial == value.GetUnit() ||
+ eCSSUnit_Unset == value.GetUnit()) {
+ border->SetBorderWidth(side,
+ (mPresContext->GetBorderWidthTable())[NS_STYLE_BORDER_WIDTH_MEDIUM]);
+ }
+ else {
+ NS_ASSERTION(eCSSUnit_Null == value.GetUnit(),
+ "missing case handling border width");
+ }
+ }
+ }
+
+ // border-style, border-*-style: enum, inherit
+ {
+ const nsCSSPropertyID* subprops =
+ nsCSSProps::SubpropertyEntryFor(eCSSProperty_border_style);
+ NS_FOR_CSS_SIDES(side) {
+ const nsCSSValue& value = *aRuleData->ValueFor(subprops[side]);
+ nsCSSUnit unit = value.GetUnit();
+ MOZ_ASSERT(eCSSUnit_None != unit,
+ "'none' should be handled as enumerated value");
+ if (eCSSUnit_Enumerated == unit) {
+ border->SetBorderStyle(side, value.GetIntValue());
+ }
+ else if (eCSSUnit_Initial == unit ||
+ eCSSUnit_Unset == unit) {
+ border->SetBorderStyle(side, NS_STYLE_BORDER_STYLE_NONE);
+ }
+ else if (eCSSUnit_Inherit == unit) {
+ conditions.SetUncacheable();
+ border->SetBorderStyle(side, parentBorder->GetBorderStyle(side));
+ }
+ }
+ }
+
+ // -moz-border-*-colors: color, string, enum, none, inherit/initial
+ nscolor borderColor;
+ nscolor unused = NS_RGB(0,0,0);
+
+ static const nsCSSPropertyID borderColorsProps[] = {
+ eCSSProperty_border_top_colors,
+ eCSSProperty_border_right_colors,
+ eCSSProperty_border_bottom_colors,
+ eCSSProperty_border_left_colors
+ };
+
+ NS_FOR_CSS_SIDES(side) {
+ const nsCSSValue& value = *aRuleData->ValueFor(borderColorsProps[side]);
+ switch (value.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_None:
+ border->ClearBorderColors(side);
+ break;
+
+ case eCSSUnit_Inherit: {
+ conditions.SetUncacheable();
+ border->ClearBorderColors(side);
+ if (parentContext) {
+ nsBorderColors *parentColors;
+ parentBorder->GetCompositeColors(side, &parentColors);
+ if (parentColors) {
+ border->EnsureBorderColors();
+ border->mBorderColors[side] = parentColors->Clone();
+ }
+ }
+ break;
+ }
+
+ case eCSSUnit_List:
+ case eCSSUnit_ListDep: {
+ // Some composite border color information has been specified for this
+ // border side.
+ border->EnsureBorderColors();
+ border->ClearBorderColors(side);
+ const nsCSSValueList* list = value.GetListValue();
+ while (list) {
+ if (SetColor(list->mValue, unused, mPresContext,
+ aContext, borderColor, conditions))
+ border->AppendBorderColor(side, borderColor);
+ else {
+ NS_NOTREACHED("unexpected item in -moz-border-*-colors list");
+ }
+ list = list->mNext;
+ }
+ break;
+ }
+
+ default:
+ MOZ_ASSERT(false, "unrecognized border color unit");
+ }
+ }
+
+ // border-color, border-*-color: color, string, enum, inherit
+ {
+ const nsCSSPropertyID* subprops =
+ nsCSSProps::SubpropertyEntryFor(eCSSProperty_border_color);
+ NS_FOR_CSS_SIDES(side) {
+ SetComplexColor<eUnsetInitial>(*aRuleData->ValueFor(subprops[side]),
+ parentBorder->mBorderColor[side],
+ StyleComplexColor::CurrentColor(),
+ mPresContext,
+ border->mBorderColor[side], conditions);
+ }
+ }
+
+ // border-radius: length, percent, inherit
+ {
+ const nsCSSPropertyID* subprops =
+ nsCSSProps::SubpropertyEntryFor(eCSSProperty_border_radius);
+ NS_FOR_CSS_FULL_CORNERS(corner) {
+ int cx = NS_FULL_TO_HALF_CORNER(corner, false);
+ int cy = NS_FULL_TO_HALF_CORNER(corner, true);
+ const nsCSSValue& radius = *aRuleData->ValueFor(subprops[corner]);
+ nsStyleCoord parentX = parentBorder->mBorderRadius.Get(cx);
+ nsStyleCoord parentY = parentBorder->mBorderRadius.Get(cy);
+ nsStyleCoord coordX, coordY;
+
+ if (SetPairCoords(radius, coordX, coordY, parentX, parentY,
+ SETCOORD_LPH | SETCOORD_INITIAL_ZERO |
+ SETCOORD_STORE_CALC | SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions)) {
+ border->mBorderRadius.Set(cx, coordX);
+ border->mBorderRadius.Set(cy, coordY);
+ }
+ }
+ }
+
+ // float-edge: enum, inherit, initial
+ SetValue(*aRuleData->ValueForFloatEdge(),
+ border->mFloatEdge, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentBorder->mFloatEdge,
+ StyleFloatEdge::ContentBox);
+
+ // border-image-source
+ const nsCSSValue* borderImageSource = aRuleData->ValueForBorderImageSource();
+ if (borderImageSource->GetUnit() == eCSSUnit_Inherit) {
+ conditions.SetUncacheable();
+ border->mBorderImageSource = parentBorder->mBorderImageSource;
+ } else {
+ SetStyleImage(aContext,
+ *borderImageSource,
+ border->mBorderImageSource,
+ conditions);
+ }
+
+ nsCSSValue borderImageSliceValue;
+ nsCSSValue borderImageSliceFill;
+ SetBorderImageSlice(*aRuleData->ValueForBorderImageSlice(),
+ borderImageSliceValue, borderImageSliceFill);
+
+ // border-image-slice: fill
+ SetValue(borderImageSliceFill,
+ border->mBorderImageFill,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentBorder->mBorderImageFill,
+ NS_STYLE_BORDER_IMAGE_SLICE_NOFILL);
+
+ nsCSSRect borderImageSlice;
+ SetBorderImageRect(borderImageSliceValue, borderImageSlice);
+
+ nsCSSRect borderImageWidth;
+ SetBorderImageRect(*aRuleData->ValueForBorderImageWidth(),
+ borderImageWidth);
+
+ nsCSSRect borderImageOutset;
+ SetBorderImageRect(*aRuleData->ValueForBorderImageOutset(),
+ borderImageOutset);
+
+ NS_FOR_CSS_SIDES (side) {
+ // border-image-slice
+ if (SetCoord(borderImageSlice.*(nsCSSRect::sides[side]), coord,
+ parentBorder->mBorderImageSlice.Get(side),
+ SETCOORD_FACTOR | SETCOORD_PERCENT |
+ SETCOORD_INHERIT | SETCOORD_INITIAL_HUNDRED_PCT |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions)) {
+ border->mBorderImageSlice.Set(side, coord);
+ }
+
+ // border-image-width
+ // 'auto' here means "same as slice"
+ if (SetCoord(borderImageWidth.*(nsCSSRect::sides[side]), coord,
+ parentBorder->mBorderImageWidth.Get(side),
+ SETCOORD_LPAH | SETCOORD_FACTOR | SETCOORD_INITIAL_FACTOR_ONE |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions)) {
+ border->mBorderImageWidth.Set(side, coord);
+ }
+
+ // border-image-outset
+ if (SetCoord(borderImageOutset.*(nsCSSRect::sides[side]), coord,
+ parentBorder->mBorderImageOutset.Get(side),
+ SETCOORD_LENGTH | SETCOORD_FACTOR |
+ SETCOORD_INHERIT | SETCOORD_INITIAL_FACTOR_ZERO |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions)) {
+ border->mBorderImageOutset.Set(side, coord);
+ }
+ }
+
+ // border-image-repeat
+ nsCSSValuePair borderImageRepeat;
+ SetBorderImagePair(*aRuleData->ValueForBorderImageRepeat(),
+ borderImageRepeat);
+
+ SetValue(borderImageRepeat.mXValue,
+ border->mBorderImageRepeatH,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentBorder->mBorderImageRepeatH,
+ NS_STYLE_BORDER_IMAGE_REPEAT_STRETCH);
+
+ SetValue(borderImageRepeat.mYValue,
+ border->mBorderImageRepeatV,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentBorder->mBorderImageRepeatV,
+ NS_STYLE_BORDER_IMAGE_REPEAT_STRETCH);
+
+ COMPUTE_END_RESET(Border, border)
+}
+
+const void*
+nsRuleNode::ComputePaddingData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(Padding, padding, parentPadding)
+
+ // padding: length, percent, calc, inherit
+ const nsCSSPropertyID* subprops =
+ nsCSSProps::SubpropertyEntryFor(eCSSProperty_padding);
+ nsStyleCoord coord;
+ NS_FOR_CSS_SIDES(side) {
+ nsStyleCoord parentCoord = parentPadding->mPadding.Get(side);
+ if (SetCoord(*aRuleData->ValueFor(subprops[side]),
+ coord, parentCoord,
+ SETCOORD_LPH | SETCOORD_INITIAL_ZERO | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions)) {
+ padding->mPadding.Set(side, coord);
+ }
+ }
+
+ COMPUTE_END_RESET(Padding, padding)
+}
+
+const void*
+nsRuleNode::ComputeOutlineData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(Outline, outline, parentOutline)
+
+ // outline-width: length, enum, inherit
+ const nsCSSValue* outlineWidthValue = aRuleData->ValueForOutlineWidth();
+ if (eCSSUnit_Initial == outlineWidthValue->GetUnit() ||
+ eCSSUnit_Unset == outlineWidthValue->GetUnit()) {
+ outline->mOutlineWidth =
+ nsStyleCoord(NS_STYLE_BORDER_WIDTH_MEDIUM, eStyleUnit_Enumerated);
+ }
+ else {
+ SetCoord(*outlineWidthValue, outline->mOutlineWidth,
+ parentOutline->mOutlineWidth,
+ SETCOORD_LEH | SETCOORD_CALC_LENGTH_ONLY, aContext,
+ mPresContext, conditions);
+ }
+
+ // outline-offset: length, inherit
+ nsStyleCoord tempCoord;
+ const nsCSSValue* outlineOffsetValue = aRuleData->ValueForOutlineOffset();
+ if (SetCoord(*outlineOffsetValue, tempCoord,
+ nsStyleCoord(parentOutline->mOutlineOffset,
+ nsStyleCoord::CoordConstructor),
+ SETCOORD_LH | SETCOORD_INITIAL_ZERO | SETCOORD_CALC_LENGTH_ONLY |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions)) {
+ outline->mOutlineOffset = tempCoord.GetCoordValue();
+ } else {
+ NS_ASSERTION(outlineOffsetValue->GetUnit() == eCSSUnit_Null,
+ "unexpected unit");
+ }
+
+ // outline-color: color, string, enum, inherit
+ SetComplexColor<eUnsetInitial>(*aRuleData->ValueForOutlineColor(),
+ parentOutline->mOutlineColor,
+ StyleComplexColor::CurrentColor(),
+ mPresContext,
+ outline->mOutlineColor, conditions);
+
+ // -moz-outline-radius: length, percent, inherit
+ {
+ const nsCSSPropertyID* subprops =
+ nsCSSProps::SubpropertyEntryFor(eCSSProperty__moz_outline_radius);
+ NS_FOR_CSS_FULL_CORNERS(corner) {
+ int cx = NS_FULL_TO_HALF_CORNER(corner, false);
+ int cy = NS_FULL_TO_HALF_CORNER(corner, true);
+ const nsCSSValue& radius = *aRuleData->ValueFor(subprops[corner]);
+ nsStyleCoord parentX = parentOutline->mOutlineRadius.Get(cx);
+ nsStyleCoord parentY = parentOutline->mOutlineRadius.Get(cy);
+ nsStyleCoord coordX, coordY;
+
+ if (SetPairCoords(radius, coordX, coordY, parentX, parentY,
+ SETCOORD_LPH | SETCOORD_INITIAL_ZERO |
+ SETCOORD_STORE_CALC | SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions)) {
+ outline->mOutlineRadius.Set(cx, coordX);
+ outline->mOutlineRadius.Set(cy, coordY);
+ }
+ }
+ }
+
+ // outline-style: enum, inherit, initial
+ // cannot use SetValue because of SetOutlineStyle
+ const nsCSSValue* outlineStyleValue = aRuleData->ValueForOutlineStyle();
+ nsCSSUnit unit = outlineStyleValue->GetUnit();
+ MOZ_ASSERT(eCSSUnit_None != unit && eCSSUnit_Auto != unit,
+ "'none' and 'auto' should be handled as enumerated values");
+ if (eCSSUnit_Enumerated == unit) {
+ outline->mOutlineStyle = outlineStyleValue->GetIntValue();
+ } else if (eCSSUnit_Initial == unit ||
+ eCSSUnit_Unset == unit) {
+ outline->mOutlineStyle = NS_STYLE_BORDER_STYLE_NONE;
+ } else if (eCSSUnit_Inherit == unit) {
+ conditions.SetUncacheable();
+ outline->mOutlineStyle = parentOutline->mOutlineStyle;
+ }
+
+ outline->RecalcData();
+ COMPUTE_END_RESET(Outline, outline)
+}
+
+const void*
+nsRuleNode::ComputeListData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_INHERITED(List, list, parentList)
+
+ // quotes: inherit, initial, none, [string string]+
+ const nsCSSValue* quotesValue = aRuleData->ValueForQuotes();
+ switch (quotesValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_Inherit:
+ case eCSSUnit_Unset:
+ conditions.SetUncacheable();
+ list->SetQuotesInherit(parentList);
+ break;
+ case eCSSUnit_Initial:
+ list->SetQuotesInitial();
+ break;
+ case eCSSUnit_None:
+ list->SetQuotesNone();
+ break;
+ case eCSSUnit_PairList:
+ case eCSSUnit_PairListDep: {
+ const nsCSSValuePairList* ourQuotes = quotesValue->GetPairListValue();
+
+ nsStyleQuoteValues::QuotePairArray quotePairs;
+ quotePairs.SetLength(ListLength(ourQuotes));
+
+ size_t index = 0;
+ nsAutoString buffer;
+ while (ourQuotes) {
+ MOZ_ASSERT(ourQuotes->mXValue.GetUnit() == eCSSUnit_String &&
+ ourQuotes->mYValue.GetUnit() == eCSSUnit_String,
+ "improper list contents for quotes");
+ quotePairs[index].first = ourQuotes->mXValue.GetStringValue(buffer);
+ quotePairs[index].second = ourQuotes->mYValue.GetStringValue(buffer);
+ ++index;
+ ourQuotes = ourQuotes->mNext;
+ }
+ list->SetQuotes(Move(quotePairs));
+ break;
+ }
+ default:
+ MOZ_ASSERT(false, "unexpected value unit");
+ }
+
+ // list-style-type: string, none, inherit, initial
+ const nsCSSValue* typeValue = aRuleData->ValueForListStyleType();
+ switch (typeValue->GetUnit()) {
+ case eCSSUnit_Unset:
+ case eCSSUnit_Inherit: {
+ conditions.SetUncacheable();
+ list->SetCounterStyle(parentList->GetCounterStyle());
+ break;
+ }
+ case eCSSUnit_Initial:
+ list->SetListStyleType(NS_LITERAL_STRING("disc"), mPresContext);
+ break;
+ case eCSSUnit_Ident: {
+ nsString typeIdent;
+ typeValue->GetStringValue(typeIdent);
+ list->SetListStyleType(typeIdent, mPresContext);
+ break;
+ }
+ case eCSSUnit_String: {
+ nsString str;
+ typeValue->GetStringValue(str);
+ list->SetCounterStyle(new AnonymousCounterStyle(str));
+ break;
+ }
+ case eCSSUnit_Enumerated: {
+ // For compatibility with html attribute map.
+ // This branch should never be called for value from CSS.
+ int32_t intValue = typeValue->GetIntValue();
+ nsAutoString name;
+ switch (intValue) {
+ case NS_STYLE_LIST_STYLE_LOWER_ROMAN:
+ name.AssignLiteral(u"lower-roman");
+ break;
+ case NS_STYLE_LIST_STYLE_UPPER_ROMAN:
+ name.AssignLiteral(u"upper-roman");
+ break;
+ case NS_STYLE_LIST_STYLE_LOWER_ALPHA:
+ name.AssignLiteral(u"lower-alpha");
+ break;
+ case NS_STYLE_LIST_STYLE_UPPER_ALPHA:
+ name.AssignLiteral(u"upper-alpha");
+ break;
+ default:
+ CopyASCIItoUTF16(nsCSSProps::ValueToKeyword(
+ intValue, nsCSSProps::kListStyleKTable), name);
+ break;
+ }
+ list->SetListStyleType(name, mPresContext);
+ break;
+ }
+ case eCSSUnit_Symbols:
+ list->SetCounterStyle(new AnonymousCounterStyle(typeValue->GetArrayValue()));
+ break;
+ case eCSSUnit_Null:
+ break;
+ default:
+ NS_NOTREACHED("Unexpected value unit");
+ }
+
+ // list-style-image: url, none, inherit
+ const nsCSSValue* imageValue = aRuleData->ValueForListStyleImage();
+ if (eCSSUnit_Image == imageValue->GetUnit()) {
+ SetStyleImageRequest([&](nsStyleImageRequest* req) {
+ list->mListStyleImage = req;
+ }, mPresContext, *imageValue, nsStyleImageRequest::Mode(0));
+ }
+ else if (eCSSUnit_None == imageValue->GetUnit() ||
+ eCSSUnit_Initial == imageValue->GetUnit()) {
+ list->mListStyleImage = nullptr;
+ }
+ else if (eCSSUnit_Inherit == imageValue->GetUnit() ||
+ eCSSUnit_Unset == imageValue->GetUnit()) {
+ conditions.SetUncacheable();
+ list->mListStyleImage = parentList->mListStyleImage;
+ }
+
+ // list-style-position: enum, inherit, initial
+ SetValue(*aRuleData->ValueForListStylePosition(),
+ list->mListStylePosition, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentList->mListStylePosition,
+ NS_STYLE_LIST_STYLE_POSITION_OUTSIDE);
+
+ // image region property: length, auto, inherit
+ const nsCSSValue* imageRegionValue = aRuleData->ValueForImageRegion();
+ switch (imageRegionValue->GetUnit()) {
+ case eCSSUnit_Inherit:
+ case eCSSUnit_Unset:
+ conditions.SetUncacheable();
+ list->mImageRegion = parentList->mImageRegion;
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Auto:
+ list->mImageRegion.SetRect(0,0,0,0);
+ break;
+
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Rect: {
+ const nsCSSRect& rgnRect = imageRegionValue->GetRectValue();
+
+ if (rgnRect.mTop.GetUnit() == eCSSUnit_Auto)
+ list->mImageRegion.y = 0;
+ else if (rgnRect.mTop.IsLengthUnit())
+ list->mImageRegion.y =
+ CalcLength(rgnRect.mTop, aContext, mPresContext, conditions);
+
+ if (rgnRect.mBottom.GetUnit() == eCSSUnit_Auto)
+ list->mImageRegion.height = 0;
+ else if (rgnRect.mBottom.IsLengthUnit())
+ list->mImageRegion.height =
+ CalcLength(rgnRect.mBottom, aContext, mPresContext,
+ conditions) - list->mImageRegion.y;
+
+ if (rgnRect.mLeft.GetUnit() == eCSSUnit_Auto)
+ list->mImageRegion.x = 0;
+ else if (rgnRect.mLeft.IsLengthUnit())
+ list->mImageRegion.x =
+ CalcLength(rgnRect.mLeft, aContext, mPresContext, conditions);
+
+ if (rgnRect.mRight.GetUnit() == eCSSUnit_Auto)
+ list->mImageRegion.width = 0;
+ else if (rgnRect.mRight.IsLengthUnit())
+ list->mImageRegion.width =
+ CalcLength(rgnRect.mRight, aContext, mPresContext,
+ conditions) - list->mImageRegion.x;
+ break;
+ }
+
+ default:
+ MOZ_ASSERT(false, "unrecognized image-region unit");
+ }
+
+ COMPUTE_END_INHERITED(List, list)
+}
+
+static void
+SetGridTrackBreadth(const nsCSSValue& aValue,
+ nsStyleCoord& aResult,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+{
+ nsCSSUnit unit = aValue.GetUnit();
+ if (unit == eCSSUnit_FlexFraction) {
+ aResult.SetFlexFractionValue(aValue.GetFloatValue());
+ } else if (unit == eCSSUnit_Auto) {
+ aResult.SetAutoValue();
+ } else if (unit == eCSSUnit_None) {
+ // For fit-content().
+ aResult.SetNoneValue();
+ } else {
+ MOZ_ASSERT(unit != eCSSUnit_Inherit && unit != eCSSUnit_Unset,
+ "Unexpected value that would use dummyParentCoord");
+ const nsStyleCoord dummyParentCoord;
+ DebugOnly<bool> stored =
+ SetCoord(aValue, aResult, dummyParentCoord,
+ SETCOORD_LPE | SETCOORD_STORE_CALC,
+ aStyleContext, aPresContext, aConditions);
+ MOZ_ASSERT(stored, "invalid <track-size> value");
+ }
+}
+
+static void
+SetGridTrackSize(const nsCSSValue& aValue,
+ nsStyleCoord& aResultMin,
+ nsStyleCoord& aResultMax,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+{
+ if (aValue.GetUnit() == eCSSUnit_Function) {
+ nsCSSValue::Array* func = aValue.GetArrayValue();
+ auto funcName = func->Item(0).GetKeywordValue();
+ if (funcName == eCSSKeyword_minmax) {
+ SetGridTrackBreadth(func->Item(1), aResultMin,
+ aStyleContext, aPresContext, aConditions);
+ SetGridTrackBreadth(func->Item(2), aResultMax,
+ aStyleContext, aPresContext, aConditions);
+ } else if (funcName == eCSSKeyword_fit_content) {
+ // We represent fit-content(L) as 'none' min-sizing and L max-sizing.
+ SetGridTrackBreadth(nsCSSValue(eCSSUnit_None), aResultMin,
+ aStyleContext, aPresContext, aConditions);
+ SetGridTrackBreadth(func->Item(1), aResultMax,
+ aStyleContext, aPresContext, aConditions);
+ } else {
+ NS_ERROR("Expected minmax() or fit-content(), got another function name");
+ }
+ } else {
+ // A single <track-breadth>,
+ // specifies identical min and max sizing functions.
+ SetGridTrackBreadth(aValue, aResultMin,
+ aStyleContext, aPresContext, aConditions);
+ aResultMax = aResultMin;
+ }
+}
+
+static void
+SetGridAutoColumnsRows(const nsCSSValue& aValue,
+ nsStyleCoord& aResultMin,
+ nsStyleCoord& aResultMax,
+ const nsStyleCoord& aParentValueMin,
+ const nsStyleCoord& aParentValueMax,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+
+{
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Inherit:
+ aConditions.SetUncacheable();
+ aResultMin = aParentValueMin;
+ aResultMax = aParentValueMax;
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ // The initial value is 'auto',
+ // which computes to 'minmax(auto, auto)'.
+ // (Explicitly-specified 'auto' values are handled in SetGridTrackSize.)
+ aResultMin.SetAutoValue();
+ aResultMax.SetAutoValue();
+ break;
+
+ default:
+ SetGridTrackSize(aValue, aResultMin, aResultMax,
+ aStyleContext, aPresContext, aConditions);
+ }
+}
+
+static void
+AppendGridLineNames(const nsCSSValue& aValue,
+ nsTArray<nsString>& aNameList)
+{
+ // Compute a <line-names> value
+ // Null unit means empty list, nothing more to do.
+ if (aValue.GetUnit() != eCSSUnit_Null) {
+ const nsCSSValueList* item = aValue.GetListValue();
+ do {
+ nsString* name = aNameList.AppendElement();
+ item->mValue.GetStringValue(*name);
+ item = item->mNext;
+ } while (item);
+ }
+}
+
+static void
+SetGridTrackList(const nsCSSValue& aValue,
+ nsStyleGridTemplate& aResult,
+ const nsStyleGridTemplate& aParentValue,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+
+{
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Inherit:
+ aConditions.SetUncacheable();
+ aResult.mIsSubgrid = aParentValue.mIsSubgrid;
+ aResult.mLineNameLists = aParentValue.mLineNameLists;
+ aResult.mMinTrackSizingFunctions = aParentValue.mMinTrackSizingFunctions;
+ aResult.mMaxTrackSizingFunctions = aParentValue.mMaxTrackSizingFunctions;
+ aResult.mRepeatAutoLineNameListBefore = aParentValue.mRepeatAutoLineNameListBefore;
+ aResult.mRepeatAutoLineNameListAfter = aParentValue.mRepeatAutoLineNameListAfter;
+ aResult.mRepeatAutoIndex = aParentValue.mRepeatAutoIndex;
+ aResult.mIsAutoFill = aParentValue.mIsAutoFill;
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_None:
+ aResult.mIsSubgrid = false;
+ aResult.mLineNameLists.Clear();
+ aResult.mMinTrackSizingFunctions.Clear();
+ aResult.mMaxTrackSizingFunctions.Clear();
+ aResult.mRepeatAutoLineNameListBefore.Clear();
+ aResult.mRepeatAutoLineNameListAfter.Clear();
+ aResult.mRepeatAutoIndex = -1;
+ aResult.mIsAutoFill = false;
+ break;
+
+ default:
+ aResult.mLineNameLists.Clear();
+ aResult.mMinTrackSizingFunctions.Clear();
+ aResult.mMaxTrackSizingFunctions.Clear();
+ aResult.mRepeatAutoLineNameListBefore.Clear();
+ aResult.mRepeatAutoLineNameListAfter.Clear();
+ aResult.mRepeatAutoIndex = -1;
+ aResult.mIsAutoFill = false;
+ const nsCSSValueList* item = aValue.GetListValue();
+ if (item->mValue.GetUnit() == eCSSUnit_Enumerated &&
+ item->mValue.GetIntValue() == NS_STYLE_GRID_TEMPLATE_SUBGRID) {
+ // subgrid <line-name-list>?
+ aResult.mIsSubgrid = true;
+ item = item->mNext;
+ for (int32_t i = 0; item && i < nsStyleGridLine::kMaxLine; ++i) {
+ if (item->mValue.GetUnit() == eCSSUnit_Pair) {
+ // This is a 'auto-fill' <name-repeat> expression.
+ const nsCSSValuePair& pair = item->mValue.GetPairValue();
+ MOZ_ASSERT(aResult.mRepeatAutoIndex == -1,
+ "can only have one <name-repeat> with auto-fill");
+ aResult.mRepeatAutoIndex = i;
+ aResult.mIsAutoFill = true;
+ MOZ_ASSERT(pair.mXValue.GetIntValue() == NS_STYLE_GRID_REPEAT_AUTO_FILL,
+ "unexpected repeat() enum value for subgrid");
+ const nsCSSValueList* list = pair.mYValue.GetListValue();
+ AppendGridLineNames(list->mValue, aResult.mRepeatAutoLineNameListBefore);
+ } else {
+ AppendGridLineNames(item->mValue,
+ *aResult.mLineNameLists.AppendElement());
+ }
+ item = item->mNext;
+ }
+ } else {
+ // <track-list>
+ // The list is expected to have odd number of items, at least 3
+ // starting with a <line-names> (sub list of identifiers),
+ // and alternating between that and <track-size>.
+ aResult.mIsSubgrid = false;
+ for (int32_t line = 1; ; ++line) {
+ AppendGridLineNames(item->mValue,
+ *aResult.mLineNameLists.AppendElement());
+ item = item->mNext;
+
+ if (!item || line == nsStyleGridLine::kMaxLine) {
+ break;
+ }
+
+ if (item->mValue.GetUnit() == eCSSUnit_Pair) {
+ // This is a 'auto-fill' / 'auto-fit' <auto-repeat> expression.
+ const nsCSSValuePair& pair = item->mValue.GetPairValue();
+ MOZ_ASSERT(aResult.mRepeatAutoIndex == -1,
+ "can only have one <auto-repeat>");
+ aResult.mRepeatAutoIndex = line - 1;
+ switch (pair.mXValue.GetIntValue()) {
+ case NS_STYLE_GRID_REPEAT_AUTO_FILL:
+ aResult.mIsAutoFill = true;
+ break;
+ case NS_STYLE_GRID_REPEAT_AUTO_FIT:
+ aResult.mIsAutoFill = false;
+ break;
+ default:
+ MOZ_ASSERT_UNREACHABLE("unexpected repeat() enum value");
+ }
+ const nsCSSValueList* list = pair.mYValue.GetListValue();
+ AppendGridLineNames(list->mValue, aResult.mRepeatAutoLineNameListBefore);
+ list = list->mNext;
+ nsStyleCoord& min = *aResult.mMinTrackSizingFunctions.AppendElement();
+ nsStyleCoord& max = *aResult.mMaxTrackSizingFunctions.AppendElement();
+ SetGridTrackSize(list->mValue, min, max,
+ aStyleContext, aPresContext, aConditions);
+ list = list->mNext;
+ AppendGridLineNames(list->mValue, aResult.mRepeatAutoLineNameListAfter);
+ } else {
+ nsStyleCoord& min = *aResult.mMinTrackSizingFunctions.AppendElement();
+ nsStyleCoord& max = *aResult.mMaxTrackSizingFunctions.AppendElement();
+ SetGridTrackSize(item->mValue, min, max,
+ aStyleContext, aPresContext, aConditions);
+ }
+
+ item = item->mNext;
+ MOZ_ASSERT(item, "Expected a eCSSUnit_List of odd length");
+ }
+ MOZ_ASSERT(!aResult.mMinTrackSizingFunctions.IsEmpty() &&
+ aResult.mMinTrackSizingFunctions.Length() ==
+ aResult.mMaxTrackSizingFunctions.Length() &&
+ aResult.mMinTrackSizingFunctions.Length() + 1 ==
+ aResult.mLineNameLists.Length(),
+ "Inconstistent array lengths for nsStyleGridTemplate");
+ }
+ }
+}
+
+static void
+SetGridTemplateAreas(const nsCSSValue& aValue,
+ RefPtr<css::GridTemplateAreasValue>* aResult,
+ css::GridTemplateAreasValue* aParentValue,
+ RuleNodeCacheConditions& aConditions)
+{
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Inherit:
+ aConditions.SetUncacheable();
+ *aResult = aParentValue;
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_None:
+ *aResult = nullptr;
+ break;
+
+ default:
+ *aResult = aValue.GetGridTemplateAreas();
+ }
+}
+
+static void
+SetGridLine(const nsCSSValue& aValue,
+ nsStyleGridLine& aResult,
+ const nsStyleGridLine& aParentValue,
+ RuleNodeCacheConditions& aConditions)
+
+{
+ switch (aValue.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Inherit:
+ aConditions.SetUncacheable();
+ aResult = aParentValue;
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_Auto:
+ aResult.SetAuto();
+ break;
+
+ default:
+ aResult.SetAuto(); // Reset any existing value.
+ const nsCSSValueList* item = aValue.GetListValue();
+ do {
+ if (item->mValue.GetUnit() == eCSSUnit_Enumerated) {
+ aResult.mHasSpan = true;
+ } else if (item->mValue.GetUnit() == eCSSUnit_Integer) {
+ aResult.mInteger = clamped(item->mValue.GetIntValue(),
+ nsStyleGridLine::kMinLine,
+ nsStyleGridLine::kMaxLine);
+ } else if (item->mValue.GetUnit() == eCSSUnit_Ident) {
+ item->mValue.GetStringValue(aResult.mLineName);
+ } else {
+ NS_ASSERTION(false, "Unexpected unit");
+ }
+ item = item->mNext;
+ } while (item);
+ MOZ_ASSERT(!aResult.IsAuto(),
+ "should have set something away from default value");
+ }
+}
+
+const void*
+nsRuleNode::ComputePositionData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(Position, pos, parentPos)
+
+ // box offsets: length, percent, calc, auto, inherit
+ static const nsCSSPropertyID offsetProps[] = {
+ eCSSProperty_top,
+ eCSSProperty_right,
+ eCSSProperty_bottom,
+ eCSSProperty_left
+ };
+ nsStyleCoord coord;
+ NS_FOR_CSS_SIDES(side) {
+ nsStyleCoord parentCoord = parentPos->mOffset.Get(side);
+ if (SetCoord(*aRuleData->ValueFor(offsetProps[side]),
+ coord, parentCoord,
+ SETCOORD_LPAH | SETCOORD_INITIAL_AUTO | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions)) {
+ pos->mOffset.Set(side, coord);
+ }
+ }
+
+ // We allow the enumerated box size property values -moz-min-content, etc. to
+ // be specified on both the {,min-,max-}width properties and the
+ // {,min-,max-}height properties, regardless of the writing mode. This is
+ // because the writing mode is not determined until here, at computed value
+ // time. Since we do not support layout behavior of these keywords on the
+ // block-axis properties, we turn them into unset if we find them in
+ // that case.
+
+ WritingMode wm(aContext);
+ bool vertical = wm.IsVertical();
+
+ const nsCSSValue* width = aRuleData->ValueForWidth();
+ if (width->GetUnit() == eCSSUnit_Enumerated) {
+ conditions.SetWritingModeDependency(wm.GetBits());
+ }
+ SetCoord(width->GetUnit() == eCSSUnit_Enumerated && vertical ?
+ nsCSSValue(eCSSUnit_Unset) : *width,
+ pos->mWidth, parentPos->mWidth,
+ SETCOORD_LPAEH | SETCOORD_INITIAL_AUTO | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+
+ const nsCSSValue* minWidth = aRuleData->ValueForMinWidth();
+ if (minWidth->GetUnit() == eCSSUnit_Enumerated) {
+ conditions.SetWritingModeDependency(wm.GetBits());
+ }
+ SetCoord(minWidth->GetUnit() == eCSSUnit_Enumerated && vertical ?
+ nsCSSValue(eCSSUnit_Unset) : *minWidth,
+ pos->mMinWidth, parentPos->mMinWidth,
+ SETCOORD_LPAEH | SETCOORD_INITIAL_AUTO | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+
+ const nsCSSValue* maxWidth = aRuleData->ValueForMaxWidth();
+ if (maxWidth->GetUnit() == eCSSUnit_Enumerated) {
+ conditions.SetWritingModeDependency(wm.GetBits());
+ }
+ SetCoord(maxWidth->GetUnit() == eCSSUnit_Enumerated && vertical ?
+ nsCSSValue(eCSSUnit_Unset) : *maxWidth,
+ pos->mMaxWidth, parentPos->mMaxWidth,
+ SETCOORD_LPOEH | SETCOORD_INITIAL_NONE | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+
+ const nsCSSValue* height = aRuleData->ValueForHeight();
+ if (height->GetUnit() == eCSSUnit_Enumerated) {
+ conditions.SetWritingModeDependency(wm.GetBits());
+ }
+ SetCoord(height->GetUnit() == eCSSUnit_Enumerated && !vertical ?
+ nsCSSValue(eCSSUnit_Unset) : *height,
+ pos->mHeight, parentPos->mHeight,
+ SETCOORD_LPAEH | SETCOORD_INITIAL_AUTO | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+
+ const nsCSSValue* minHeight = aRuleData->ValueForMinHeight();
+ if (minHeight->GetUnit() == eCSSUnit_Enumerated) {
+ conditions.SetWritingModeDependency(wm.GetBits());
+ }
+ SetCoord(minHeight->GetUnit() == eCSSUnit_Enumerated && !vertical ?
+ nsCSSValue(eCSSUnit_Unset) : *minHeight,
+ pos->mMinHeight, parentPos->mMinHeight,
+ SETCOORD_LPAEH | SETCOORD_INITIAL_AUTO | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+
+ const nsCSSValue* maxHeight = aRuleData->ValueForMaxHeight();
+ if (maxHeight->GetUnit() == eCSSUnit_Enumerated) {
+ conditions.SetWritingModeDependency(wm.GetBits());
+ }
+ SetCoord(maxHeight->GetUnit() == eCSSUnit_Enumerated && !vertical ?
+ nsCSSValue(eCSSUnit_Unset) : *maxHeight,
+ pos->mMaxHeight, parentPos->mMaxHeight,
+ SETCOORD_LPOEH | SETCOORD_INITIAL_NONE | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+
+ // box-sizing: enum, inherit, initial
+ SetValue(*aRuleData->ValueForBoxSizing(),
+ pos->mBoxSizing, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentPos->mBoxSizing,
+ StyleBoxSizing::Content);
+
+ // align-content: enum, inherit, initial
+ SetValue(*aRuleData->ValueForAlignContent(),
+ pos->mAlignContent, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentPos->mAlignContent,
+ NS_STYLE_ALIGN_NORMAL);
+
+ // align-items: enum, inherit, initial
+ SetValue(*aRuleData->ValueForAlignItems(),
+ pos->mAlignItems, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentPos->mAlignItems,
+ NS_STYLE_ALIGN_NORMAL);
+
+ // align-self: enum, inherit, initial
+ SetValue(*aRuleData->ValueForAlignSelf(),
+ pos->mAlignSelf, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentPos->mAlignSelf,
+ NS_STYLE_ALIGN_AUTO);
+
+ // justify-content: enum, inherit, initial
+ SetValue(*aRuleData->ValueForJustifyContent(),
+ pos->mJustifyContent, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentPos->mJustifyContent,
+ NS_STYLE_JUSTIFY_NORMAL);
+
+ // justify-items: enum, inherit, initial
+ const auto& justifyItemsValue = *aRuleData->ValueForJustifyItems();
+ if (MOZ_UNLIKELY(justifyItemsValue.GetUnit() == eCSSUnit_Inherit)) {
+ if (MOZ_LIKELY(parentContext)) {
+ pos->mJustifyItems =
+ parentPos->ComputedJustifyItems(parentContext->GetParent());
+ } else {
+ pos->mJustifyItems = NS_STYLE_JUSTIFY_NORMAL;
+ }
+ conditions.SetUncacheable();
+ } else {
+ SetValue(justifyItemsValue,
+ pos->mJustifyItems, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentPos->mJustifyItems, // unused, we handle 'inherit' above
+ NS_STYLE_JUSTIFY_AUTO);
+ }
+
+ // justify-self: enum, inherit, initial
+ SetValue(*aRuleData->ValueForJustifySelf(),
+ pos->mJustifySelf, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentPos->mJustifySelf,
+ NS_STYLE_JUSTIFY_AUTO);
+
+ // flex-basis: auto, length, percent, enum, calc, inherit, initial
+ // (Note: The flags here should match those used for 'width' property above.)
+ SetCoord(*aRuleData->ValueForFlexBasis(), pos->mFlexBasis, parentPos->mFlexBasis,
+ SETCOORD_LPAEH | SETCOORD_INITIAL_AUTO | SETCOORD_STORE_CALC |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+
+ // flex-direction: enum, inherit, initial
+ SetValue(*aRuleData->ValueForFlexDirection(),
+ pos->mFlexDirection, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentPos->mFlexDirection,
+ NS_STYLE_FLEX_DIRECTION_ROW);
+
+ // flex-grow: float, inherit, initial
+ SetFactor(*aRuleData->ValueForFlexGrow(),
+ pos->mFlexGrow, conditions,
+ parentPos->mFlexGrow, 0.0f,
+ SETFCT_UNSET_INITIAL);
+
+ // flex-shrink: float, inherit, initial
+ SetFactor(*aRuleData->ValueForFlexShrink(),
+ pos->mFlexShrink, conditions,
+ parentPos->mFlexShrink, 1.0f,
+ SETFCT_UNSET_INITIAL);
+
+ // flex-wrap: enum, inherit, initial
+ SetValue(*aRuleData->ValueForFlexWrap(),
+ pos->mFlexWrap, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentPos->mFlexWrap,
+ NS_STYLE_FLEX_WRAP_NOWRAP);
+
+ // order: integer, inherit, initial
+ SetValue(*aRuleData->ValueForOrder(),
+ pos->mOrder, conditions,
+ SETVAL_INTEGER | SETVAL_UNSET_INITIAL,
+ parentPos->mOrder,
+ NS_STYLE_ORDER_INITIAL);
+
+ // object-fit: enum, inherit, initial
+ SetValue(*aRuleData->ValueForObjectFit(),
+ pos->mObjectFit, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentPos->mObjectFit,
+ NS_STYLE_OBJECT_FIT_FILL);
+
+ // object-position
+ const nsCSSValue& objectPosition = *aRuleData->ValueForObjectPosition();
+ switch (objectPosition.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_Inherit:
+ conditions.SetUncacheable();
+ pos->mObjectPosition = parentPos->mObjectPosition;
+ break;
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ pos->mObjectPosition.SetInitialPercentValues(0.5f);
+ break;
+ default:
+ ComputePositionValue(aContext, objectPosition,
+ pos->mObjectPosition, conditions);
+ }
+
+ // grid-auto-flow
+ const nsCSSValue& gridAutoFlow = *aRuleData->ValueForGridAutoFlow();
+ switch (gridAutoFlow.GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_Inherit:
+ conditions.SetUncacheable();
+ pos->mGridAutoFlow = parentPos->mGridAutoFlow;
+ break;
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ pos->mGridAutoFlow = NS_STYLE_GRID_AUTO_FLOW_ROW;
+ break;
+ default:
+ NS_ASSERTION(gridAutoFlow.GetUnit() == eCSSUnit_Enumerated,
+ "Unexpected unit");
+ pos->mGridAutoFlow = gridAutoFlow.GetIntValue();
+ }
+
+ // grid-auto-columns
+ SetGridAutoColumnsRows(*aRuleData->ValueForGridAutoColumns(),
+ pos->mGridAutoColumnsMin,
+ pos->mGridAutoColumnsMax,
+ parentPos->mGridAutoColumnsMin,
+ parentPos->mGridAutoColumnsMax,
+ aContext, mPresContext, conditions);
+
+ // grid-auto-rows
+ SetGridAutoColumnsRows(*aRuleData->ValueForGridAutoRows(),
+ pos->mGridAutoRowsMin,
+ pos->mGridAutoRowsMax,
+ parentPos->mGridAutoRowsMin,
+ parentPos->mGridAutoRowsMax,
+ aContext, mPresContext, conditions);
+
+ // grid-template-columns
+ SetGridTrackList(*aRuleData->ValueForGridTemplateColumns(),
+ pos->mGridTemplateColumns, parentPos->mGridTemplateColumns,
+ aContext, mPresContext, conditions);
+
+ // grid-template-rows
+ SetGridTrackList(*aRuleData->ValueForGridTemplateRows(),
+ pos->mGridTemplateRows, parentPos->mGridTemplateRows,
+ aContext, mPresContext, conditions);
+
+ // grid-tempate-areas
+ SetGridTemplateAreas(*aRuleData->ValueForGridTemplateAreas(),
+ &pos->mGridTemplateAreas,
+ parentPos->mGridTemplateAreas,
+ conditions);
+
+ // grid-column-start
+ SetGridLine(*aRuleData->ValueForGridColumnStart(),
+ pos->mGridColumnStart,
+ parentPos->mGridColumnStart,
+ conditions);
+
+ // grid-column-end
+ SetGridLine(*aRuleData->ValueForGridColumnEnd(),
+ pos->mGridColumnEnd,
+ parentPos->mGridColumnEnd,
+ conditions);
+
+ // grid-row-start
+ SetGridLine(*aRuleData->ValueForGridRowStart(),
+ pos->mGridRowStart,
+ parentPos->mGridRowStart,
+ conditions);
+
+ // grid-row-end
+ SetGridLine(*aRuleData->ValueForGridRowEnd(),
+ pos->mGridRowEnd,
+ parentPos->mGridRowEnd,
+ conditions);
+
+ // grid-column-gap
+ if (SetCoord(*aRuleData->ValueForGridColumnGap(),
+ pos->mGridColumnGap, parentPos->mGridColumnGap,
+ SETCOORD_LPH | SETCOORD_INITIAL_ZERO | SETCOORD_STORE_CALC |
+ SETCOORD_CALC_CLAMP_NONNEGATIVE | SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions)) {
+ } else {
+ MOZ_ASSERT(aRuleData->ValueForGridColumnGap()->GetUnit() == eCSSUnit_Null,
+ "unexpected unit");
+ }
+
+ // grid-row-gap
+ if (SetCoord(*aRuleData->ValueForGridRowGap(),
+ pos->mGridRowGap, parentPos->mGridRowGap,
+ SETCOORD_LPH | SETCOORD_INITIAL_ZERO | SETCOORD_STORE_CALC |
+ SETCOORD_CALC_CLAMP_NONNEGATIVE | SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions)) {
+ } else {
+ MOZ_ASSERT(aRuleData->ValueForGridRowGap()->GetUnit() == eCSSUnit_Null,
+ "unexpected unit");
+ }
+
+ // z-index
+ const nsCSSValue* zIndexValue = aRuleData->ValueForZIndex();
+ if (! SetCoord(*zIndexValue, pos->mZIndex, parentPos->mZIndex,
+ SETCOORD_IA | SETCOORD_INITIAL_AUTO | SETCOORD_UNSET_INITIAL,
+ aContext, nullptr, conditions)) {
+ if (eCSSUnit_Inherit == zIndexValue->GetUnit()) {
+ // handle inherit, because it's ok to inherit 'auto' here
+ conditions.SetUncacheable();
+ pos->mZIndex = parentPos->mZIndex;
+ }
+ }
+
+ COMPUTE_END_RESET(Position, pos)
+}
+
+const void*
+nsRuleNode::ComputeTableData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(Table, table, parentTable)
+
+ // table-layout: enum, inherit, initial
+ SetValue(*aRuleData->ValueForTableLayout(),
+ table->mLayoutStrategy, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentTable->mLayoutStrategy,
+ NS_STYLE_TABLE_LAYOUT_AUTO);
+
+ // span: pixels (not a real CSS prop)
+ const nsCSSValue* spanValue = aRuleData->ValueForSpan();
+ if (eCSSUnit_Enumerated == spanValue->GetUnit() ||
+ eCSSUnit_Integer == spanValue->GetUnit())
+ table->mSpan = spanValue->GetIntValue();
+
+ COMPUTE_END_RESET(Table, table)
+}
+
+const void*
+nsRuleNode::ComputeTableBorderData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_INHERITED(TableBorder, table, parentTable)
+
+ // border-collapse: enum, inherit, initial
+ SetValue(*aRuleData->ValueForBorderCollapse(), table->mBorderCollapse,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentTable->mBorderCollapse,
+ NS_STYLE_BORDER_SEPARATE);
+
+ const nsCSSValue* borderSpacingValue = aRuleData->ValueForBorderSpacing();
+ // border-spacing: pair(length), inherit
+ if (borderSpacingValue->GetUnit() != eCSSUnit_Null) {
+ nsStyleCoord parentCol(parentTable->mBorderSpacingCol,
+ nsStyleCoord::CoordConstructor);
+ nsStyleCoord parentRow(parentTable->mBorderSpacingRow,
+ nsStyleCoord::CoordConstructor);
+ nsStyleCoord coordCol, coordRow;
+
+#ifdef DEBUG
+ bool result =
+#endif
+ SetPairCoords(*borderSpacingValue,
+ coordCol, coordRow, parentCol, parentRow,
+ SETCOORD_LH | SETCOORD_INITIAL_ZERO |
+ SETCOORD_CALC_LENGTH_ONLY |
+ SETCOORD_CALC_CLAMP_NONNEGATIVE | SETCOORD_UNSET_INHERIT,
+ aContext, mPresContext, conditions);
+ NS_ASSERTION(result, "malformed table border value");
+ table->mBorderSpacingCol = coordCol.GetCoordValue();
+ table->mBorderSpacingRow = coordRow.GetCoordValue();
+ }
+
+ // caption-side: enum, inherit, initial
+ SetValue(*aRuleData->ValueForCaptionSide(),
+ table->mCaptionSide, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentTable->mCaptionSide,
+ NS_STYLE_CAPTION_SIDE_TOP);
+
+ // empty-cells: enum, inherit, initial
+ SetValue(*aRuleData->ValueForEmptyCells(),
+ table->mEmptyCells, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentTable->mEmptyCells,
+ NS_STYLE_TABLE_EMPTY_CELLS_SHOW);
+
+ COMPUTE_END_INHERITED(TableBorder, table)
+}
+
+const void*
+nsRuleNode::ComputeContentData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ uint32_t count;
+ nsAutoString buffer;
+
+ COMPUTE_START_RESET(Content, content, parentContent)
+
+ // content: [string, url, counter, attr, enum]+, normal, none, inherit
+ const nsCSSValue* contentValue = aRuleData->ValueForContent();
+ switch (contentValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Normal:
+ case eCSSUnit_None:
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ // "normal", "none", "initial" and "unset" all mean no content
+ content->AllocateContents(0);
+ break;
+
+ case eCSSUnit_Inherit:
+ conditions.SetUncacheable();
+ count = parentContent->ContentCount();
+ content->AllocateContents(count);
+ while (0 < count--) {
+ content->ContentAt(count) = parentContent->ContentAt(count);
+ }
+ break;
+
+ case eCSSUnit_Enumerated: {
+ MOZ_ASSERT(contentValue->GetIntValue() == NS_STYLE_CONTENT_ALT_CONTENT,
+ "unrecognized solitary content keyword");
+ content->AllocateContents(1);
+ nsStyleContentData& data = content->ContentAt(0);
+ data.mType = eStyleContentType_AltContent;
+ data.mContent.mString = nullptr;
+ break;
+ }
+
+ case eCSSUnit_List:
+ case eCSSUnit_ListDep: {
+ const nsCSSValueList* contentValueList = contentValue->GetListValue();
+ count = 0;
+ while (contentValueList) {
+ count++;
+ contentValueList = contentValueList->mNext;
+ }
+ content->AllocateContents(count);
+ const nsAutoString nullStr;
+ count = 0;
+ contentValueList = contentValue->GetListValue();
+ while (contentValueList) {
+ const nsCSSValue& value = contentValueList->mValue;
+ nsCSSUnit unit = value.GetUnit();
+ nsStyleContentType type;
+ nsStyleContentData &data = content->ContentAt(count++);
+ switch (unit) {
+ case eCSSUnit_String: type = eStyleContentType_String; break;
+ case eCSSUnit_Image: type = eStyleContentType_Image; break;
+ case eCSSUnit_Attr: type = eStyleContentType_Attr; break;
+ case eCSSUnit_Counter: type = eStyleContentType_Counter; break;
+ case eCSSUnit_Counters: type = eStyleContentType_Counters; break;
+ case eCSSUnit_Enumerated:
+ switch (value.GetIntValue()) {
+ case NS_STYLE_CONTENT_OPEN_QUOTE:
+ type = eStyleContentType_OpenQuote; break;
+ case NS_STYLE_CONTENT_CLOSE_QUOTE:
+ type = eStyleContentType_CloseQuote; break;
+ case NS_STYLE_CONTENT_NO_OPEN_QUOTE:
+ type = eStyleContentType_NoOpenQuote; break;
+ case NS_STYLE_CONTENT_NO_CLOSE_QUOTE:
+ type = eStyleContentType_NoCloseQuote; break;
+ default:
+ NS_ERROR("bad content value");
+ type = eStyleContentType_Uninitialized;
+ }
+ break;
+ default:
+ NS_ERROR("bad content type");
+ type = eStyleContentType_Uninitialized;
+ }
+ data.mType = type;
+ if (type == eStyleContentType_Image) {
+ SetImageRequest([&](imgRequestProxy* req) {
+ data.SetImage(req);
+ }, mPresContext, value);
+ } else if (type <= eStyleContentType_Attr) {
+ value.GetStringValue(buffer);
+ data.mContent.mString = NS_strdup(buffer.get());
+ } else if (type <= eStyleContentType_Counters) {
+ data.mContent.mCounters = value.GetArrayValue();
+ data.mContent.mCounters->AddRef();
+ } else {
+ data.mContent.mString = nullptr;
+ }
+ contentValueList = contentValueList->mNext;
+ }
+ break;
+ }
+
+ default:
+ MOZ_ASSERT(false, "unrecognized content unit");
+ }
+
+ // counter-increment: [string [int]]+, none, inherit
+ const nsCSSValue* counterIncrementValue =
+ aRuleData->ValueForCounterIncrement();
+ switch (counterIncrementValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_None:
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ content->AllocateCounterIncrements(0);
+ break;
+
+ case eCSSUnit_Inherit:
+ conditions.SetUncacheable();
+ count = parentContent->CounterIncrementCount();
+ content->AllocateCounterIncrements(count);
+ while (count--) {
+ const nsStyleCounterData& data = parentContent->CounterIncrementAt(count);
+ content->SetCounterIncrementAt(count, data.mCounter, data.mValue);
+ }
+ break;
+
+ case eCSSUnit_PairList:
+ case eCSSUnit_PairListDep: {
+ const nsCSSValuePairList* ourIncrement =
+ counterIncrementValue->GetPairListValue();
+ MOZ_ASSERT(ourIncrement->mXValue.GetUnit() == eCSSUnit_Ident,
+ "unexpected value unit");
+ count = ListLength(ourIncrement);
+ content->AllocateCounterIncrements(count);
+
+ count = 0;
+ for (const nsCSSValuePairList* p = ourIncrement; p; p = p->mNext, count++) {
+ int32_t increment;
+ if (p->mYValue.GetUnit() == eCSSUnit_Integer) {
+ increment = p->mYValue.GetIntValue();
+ } else {
+ increment = 1;
+ }
+ p->mXValue.GetStringValue(buffer);
+ content->SetCounterIncrementAt(count, buffer, increment);
+ }
+ break;
+ }
+
+ default:
+ MOZ_ASSERT(false, "unexpected value unit");
+ }
+
+ // counter-reset: [string [int]]+, none, inherit
+ const nsCSSValue* counterResetValue = aRuleData->ValueForCounterReset();
+ switch (counterResetValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_None:
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ content->AllocateCounterResets(0);
+ break;
+
+ case eCSSUnit_Inherit:
+ conditions.SetUncacheable();
+ count = parentContent->CounterResetCount();
+ content->AllocateCounterResets(count);
+ while (0 < count--) {
+ const nsStyleCounterData& data = parentContent->CounterResetAt(count);
+ content->SetCounterResetAt(count, data.mCounter, data.mValue);
+ }
+ break;
+
+ case eCSSUnit_PairList:
+ case eCSSUnit_PairListDep: {
+ const nsCSSValuePairList* ourReset =
+ counterResetValue->GetPairListValue();
+ MOZ_ASSERT(ourReset->mXValue.GetUnit() == eCSSUnit_Ident,
+ "unexpected value unit");
+ count = ListLength(ourReset);
+ content->AllocateCounterResets(count);
+ count = 0;
+ for (const nsCSSValuePairList* p = ourReset; p; p = p->mNext, count++) {
+ int32_t reset;
+ if (p->mYValue.GetUnit() == eCSSUnit_Integer) {
+ reset = p->mYValue.GetIntValue();
+ } else {
+ reset = 0;
+ }
+ p->mXValue.GetStringValue(buffer);
+ content->SetCounterResetAt(count, buffer, reset);
+ }
+ break;
+ }
+
+ default:
+ MOZ_ASSERT(false, "unexpected value unit");
+ }
+
+ // If we ended up with an image, track it.
+ for (uint32_t i = 0; i < content->ContentCount(); ++i) {
+ if ((content->ContentAt(i).mType == eStyleContentType_Image) &&
+ content->ContentAt(i).mContent.mImage) {
+ content->ContentAt(i).TrackImage(
+ aContext->PresContext()->Document()->ImageTracker());
+ }
+ }
+
+ COMPUTE_END_RESET(Content, content)
+}
+
+const void*
+nsRuleNode::ComputeXULData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(XUL, xul, parentXUL)
+
+ // box-align: enum, inherit, initial
+ SetValue(*aRuleData->ValueForBoxAlign(),
+ xul->mBoxAlign, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentXUL->mBoxAlign,
+ StyleBoxAlign::Stretch);
+
+ // box-direction: enum, inherit, initial
+ SetValue(*aRuleData->ValueForBoxDirection(),
+ xul->mBoxDirection, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentXUL->mBoxDirection,
+ StyleBoxDirection::Normal);
+
+ // box-flex: factor, inherit
+ SetFactor(*aRuleData->ValueForBoxFlex(),
+ xul->mBoxFlex, conditions,
+ parentXUL->mBoxFlex, 0.0f,
+ SETFCT_UNSET_INITIAL);
+
+ // box-orient: enum, inherit, initial
+ SetValue(*aRuleData->ValueForBoxOrient(),
+ xul->mBoxOrient, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentXUL->mBoxOrient,
+ StyleBoxOrient::Horizontal);
+
+ // box-pack: enum, inherit, initial
+ SetValue(*aRuleData->ValueForBoxPack(),
+ xul->mBoxPack, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentXUL->mBoxPack,
+ StyleBoxPack::Start);
+
+ // box-ordinal-group: integer, inherit, initial
+ SetValue(*aRuleData->ValueForBoxOrdinalGroup(),
+ xul->mBoxOrdinal, conditions,
+ SETVAL_INTEGER | SETVAL_UNSET_INITIAL,
+ parentXUL->mBoxOrdinal, 1);
+
+ const nsCSSValue* stackSizingValue = aRuleData->ValueForStackSizing();
+ if (eCSSUnit_Inherit == stackSizingValue->GetUnit()) {
+ conditions.SetUncacheable();
+ xul->mStretchStack = parentXUL->mStretchStack;
+ } else if (eCSSUnit_Initial == stackSizingValue->GetUnit() ||
+ eCSSUnit_Unset == stackSizingValue->GetUnit()) {
+ xul->mStretchStack = true;
+ } else if (eCSSUnit_Enumerated == stackSizingValue->GetUnit()) {
+ xul->mStretchStack = stackSizingValue->GetIntValue() ==
+ NS_STYLE_STACK_SIZING_STRETCH_TO_FIT;
+ }
+
+ COMPUTE_END_RESET(XUL, xul)
+}
+
+const void*
+nsRuleNode::ComputeColumnData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(Column, column, parent)
+
+ // column-width: length, auto, inherit
+ SetCoord(*aRuleData->ValueForColumnWidth(),
+ column->mColumnWidth, parent->mColumnWidth,
+ SETCOORD_LAH | SETCOORD_INITIAL_AUTO |
+ SETCOORD_CALC_LENGTH_ONLY | SETCOORD_CALC_CLAMP_NONNEGATIVE |
+ SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+
+ // column-gap: length, inherit, normal
+ SetCoord(*aRuleData->ValueForColumnGap(),
+ column->mColumnGap, parent->mColumnGap,
+ SETCOORD_LH | SETCOORD_NORMAL | SETCOORD_INITIAL_NORMAL |
+ SETCOORD_CALC_LENGTH_ONLY | SETCOORD_UNSET_INITIAL,
+ aContext, mPresContext, conditions);
+ // clamp negative calc() to 0
+ if (column->mColumnGap.GetUnit() == eStyleUnit_Coord) {
+ column->mColumnGap.SetCoordValue(
+ std::max(column->mColumnGap.GetCoordValue(), 0));
+ }
+
+ // column-count: auto, integer, inherit
+ const nsCSSValue* columnCountValue = aRuleData->ValueForColumnCount();
+ if (eCSSUnit_Auto == columnCountValue->GetUnit() ||
+ eCSSUnit_Initial == columnCountValue->GetUnit() ||
+ eCSSUnit_Unset == columnCountValue->GetUnit()) {
+ column->mColumnCount = NS_STYLE_COLUMN_COUNT_AUTO;
+ } else if (eCSSUnit_Integer == columnCountValue->GetUnit()) {
+ column->mColumnCount = columnCountValue->GetIntValue();
+ // Max kMaxColumnCount columns - wallpaper for bug 345583.
+ column->mColumnCount = std::min(column->mColumnCount,
+ nsStyleColumn::kMaxColumnCount);
+ } else if (eCSSUnit_Inherit == columnCountValue->GetUnit()) {
+ conditions.SetUncacheable();
+ column->mColumnCount = parent->mColumnCount;
+ }
+
+ // column-rule-width: length, enum, inherit
+ const nsCSSValue& widthValue = *aRuleData->ValueForColumnRuleWidth();
+ if (eCSSUnit_Initial == widthValue.GetUnit() ||
+ eCSSUnit_Unset == widthValue.GetUnit()) {
+ column->SetColumnRuleWidth(
+ (mPresContext->GetBorderWidthTable())[NS_STYLE_BORDER_WIDTH_MEDIUM]);
+ }
+ else if (eCSSUnit_Enumerated == widthValue.GetUnit()) {
+ NS_ASSERTION(widthValue.GetIntValue() == NS_STYLE_BORDER_WIDTH_THIN ||
+ widthValue.GetIntValue() == NS_STYLE_BORDER_WIDTH_MEDIUM ||
+ widthValue.GetIntValue() == NS_STYLE_BORDER_WIDTH_THICK,
+ "Unexpected enum value");
+ column->SetColumnRuleWidth(
+ (mPresContext->GetBorderWidthTable())[widthValue.GetIntValue()]);
+ }
+ else if (eCSSUnit_Inherit == widthValue.GetUnit()) {
+ column->SetColumnRuleWidth(parent->GetComputedColumnRuleWidth());
+ conditions.SetUncacheable();
+ }
+ else if (widthValue.IsLengthUnit() || widthValue.IsCalcUnit()) {
+ nscoord len =
+ CalcLength(widthValue, aContext, mPresContext, conditions);
+ if (len < 0) {
+ // FIXME: This is untested (by test_value_storage.html) for
+ // column-rule-width since it gets covered up by the border
+ // rounding code.
+ NS_ASSERTION(widthValue.IsCalcUnit(),
+ "parser should have rejected negative length");
+ len = 0;
+ }
+ column->SetColumnRuleWidth(len);
+ }
+
+ // column-rule-style: enum, inherit
+ const nsCSSValue& styleValue = *aRuleData->ValueForColumnRuleStyle();
+ MOZ_ASSERT(eCSSUnit_None != styleValue.GetUnit(),
+ "'none' should be handled as enumerated value");
+ if (eCSSUnit_Enumerated == styleValue.GetUnit()) {
+ column->mColumnRuleStyle = styleValue.GetIntValue();
+ }
+ else if (eCSSUnit_Initial == styleValue.GetUnit() ||
+ eCSSUnit_Unset == styleValue.GetUnit()) {
+ column->mColumnRuleStyle = NS_STYLE_BORDER_STYLE_NONE;
+ }
+ else if (eCSSUnit_Inherit == styleValue.GetUnit()) {
+ conditions.SetUncacheable();
+ column->mColumnRuleStyle = parent->mColumnRuleStyle;
+ }
+
+ // column-rule-color: color, inherit
+ SetComplexColor<eUnsetInitial>(*aRuleData->ValueForColumnRuleColor(),
+ parent->mColumnRuleColor,
+ StyleComplexColor::CurrentColor(),
+ mPresContext,
+ column->mColumnRuleColor, conditions);
+
+ // column-fill: enum
+ SetValue(*aRuleData->ValueForColumnFill(),
+ column->mColumnFill, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parent->mColumnFill,
+ NS_STYLE_COLUMN_FILL_BALANCE);
+
+ COMPUTE_END_RESET(Column, column)
+}
+
+static void
+SetSVGPaint(const nsCSSValue& aValue, const nsStyleSVGPaint& parentPaint,
+ nsPresContext* aPresContext, nsStyleContext *aContext,
+ nsStyleSVGPaint& aResult, nsStyleSVGPaintType aInitialPaintType,
+ RuleNodeCacheConditions& aConditions)
+{
+ MOZ_ASSERT(aInitialPaintType == eStyleSVGPaintType_None ||
+ aInitialPaintType == eStyleSVGPaintType_Color,
+ "SetSVGPaint only supports initial values being either 'black' "
+ "(represented by eStyleSVGPaintType_Color) or none (by "
+ "eStyleSVGPaintType_None)");
+
+ nscolor color;
+
+ if (aValue.GetUnit() == eCSSUnit_Inherit ||
+ aValue.GetUnit() == eCSSUnit_Unset) {
+ aResult = parentPaint;
+ aConditions.SetUncacheable();
+ } else if (aValue.GetUnit() == eCSSUnit_None) {
+ aResult.SetNone();
+ } else if (aValue.GetUnit() == eCSSUnit_Initial) {
+ if (aInitialPaintType == eStyleSVGPaintType_None) {
+ aResult.SetNone();
+ } else {
+ aResult.SetColor(NS_RGB(0, 0, 0));
+ }
+ } else if (SetColor(aValue, NS_RGB(0, 0, 0), aPresContext, aContext,
+ color, aConditions)) {
+ aResult.SetColor(color);
+ } else if (aValue.GetUnit() == eCSSUnit_Pair) {
+ const nsCSSValuePair& pair = aValue.GetPairValue();
+
+ nscolor fallback;
+ if (pair.mYValue.GetUnit() == eCSSUnit_None) {
+ fallback = NS_RGBA(0, 0, 0, 0);
+ } else {
+ MOZ_ASSERT(pair.mYValue.GetUnit() != eCSSUnit_Inherit,
+ "cannot inherit fallback colour");
+ SetColor(pair.mYValue, NS_RGB(0, 0, 0), aPresContext, aContext,
+ fallback, aConditions);
+ }
+
+ if (pair.mXValue.GetUnit() == eCSSUnit_URL) {
+ aResult.SetPaintServer(pair.mXValue.GetURLStructValue(), fallback);
+ } else if (pair.mXValue.GetUnit() == eCSSUnit_Enumerated) {
+
+ switch (pair.mXValue.GetIntValue()) {
+ case NS_COLOR_CONTEXT_FILL:
+ aResult.SetContextValue(eStyleSVGPaintType_ContextFill, fallback);
+ break;
+ case NS_COLOR_CONTEXT_STROKE:
+ aResult.SetContextValue(eStyleSVGPaintType_ContextStroke, fallback);
+ break;
+ default:
+ NS_NOTREACHED("unknown keyword as paint server value");
+ }
+
+ } else {
+ NS_NOTREACHED("malformed paint server value");
+ }
+
+ } else {
+ MOZ_ASSERT(aValue.GetUnit() == eCSSUnit_Null,
+ "malformed paint server value");
+ }
+}
+
+static void
+SetSVGOpacity(const nsCSSValue& aValue,
+ float& aOpacityField, nsStyleSVGOpacitySource& aOpacityTypeField,
+ RuleNodeCacheConditions& aConditions,
+ float aParentOpacity, nsStyleSVGOpacitySource aParentOpacityType)
+{
+ if (eCSSUnit_Enumerated == aValue.GetUnit()) {
+ switch (aValue.GetIntValue()) {
+ case NS_STYLE_CONTEXT_FILL_OPACITY:
+ aOpacityTypeField = eStyleSVGOpacitySource_ContextFillOpacity;
+ break;
+ case NS_STYLE_CONTEXT_STROKE_OPACITY:
+ aOpacityTypeField = eStyleSVGOpacitySource_ContextStrokeOpacity;
+ break;
+ default:
+ NS_NOTREACHED("SetSVGOpacity: Unknown keyword");
+ }
+ // Fall back on fully opaque
+ aOpacityField = 1.0f;
+ } else if (eCSSUnit_Inherit == aValue.GetUnit() ||
+ eCSSUnit_Unset == aValue.GetUnit()) {
+ aConditions.SetUncacheable();
+ aOpacityField = aParentOpacity;
+ aOpacityTypeField = aParentOpacityType;
+ } else if (eCSSUnit_Null != aValue.GetUnit()) {
+ SetFactor(aValue, aOpacityField, aConditions,
+ aParentOpacity, 1.0f, SETFCT_OPACITY);
+ aOpacityTypeField = eStyleSVGOpacitySource_Normal;
+ }
+}
+
+/* static */
+void
+nsRuleNode::FillAllMaskLists(nsStyleImageLayers& aMask,
+ uint32_t aMaxItemCount)
+{
+
+ // Delete any extra items. We need to keep layers in which any
+ // property was specified.
+ aMask.mLayers.TruncateLengthNonZero(aMaxItemCount);
+
+ uint32_t fillCount = aMask.mImageCount;
+
+ FillImageLayerList(aMask.mLayers,
+ &nsStyleImageLayers::Layer::mImage,
+ aMask.mImageCount, fillCount);
+ FillImageLayerList(aMask.mLayers,
+ &nsStyleImageLayers::Layer::mSourceURI,
+ aMask.mImageCount, fillCount);
+ FillImageLayerList(aMask.mLayers,
+ &nsStyleImageLayers::Layer::mRepeat,
+ aMask.mRepeatCount, fillCount);
+ FillImageLayerList(aMask.mLayers,
+ &nsStyleImageLayers::Layer::mClip,
+ aMask.mClipCount, fillCount);
+ FillImageLayerList(aMask.mLayers,
+ &nsStyleImageLayers::Layer::mOrigin,
+ aMask.mOriginCount, fillCount);
+ FillImageLayerPositionCoordList(aMask.mLayers,
+ &Position::mXPosition,
+ aMask.mPositionXCount, fillCount);
+ FillImageLayerPositionCoordList(aMask.mLayers,
+ &Position::mYPosition,
+ aMask.mPositionYCount, fillCount);
+ FillImageLayerList(aMask.mLayers,
+ &nsStyleImageLayers::Layer::mSize,
+ aMask.mSizeCount, fillCount);
+ FillImageLayerList(aMask.mLayers,
+ &nsStyleImageLayers::Layer::mMaskMode,
+ aMask.mMaskModeCount, fillCount);
+ FillImageLayerList(aMask.mLayers,
+ &nsStyleImageLayers::Layer::mComposite,
+ aMask.mCompositeCount, fillCount);
+}
+
+const void*
+nsRuleNode::ComputeSVGData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_INHERITED(SVG, svg, parentSVG)
+
+ // clip-rule: enum, inherit, initial
+ SetValue(*aRuleData->ValueForClipRule(),
+ svg->mClipRule, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentSVG->mClipRule,
+ StyleFillRule::Nonzero);
+
+ // color-interpolation: enum, inherit, initial
+ SetValue(*aRuleData->ValueForColorInterpolation(),
+ svg->mColorInterpolation, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentSVG->mColorInterpolation,
+ NS_STYLE_COLOR_INTERPOLATION_SRGB);
+
+ // color-interpolation-filters: enum, inherit, initial
+ SetValue(*aRuleData->ValueForColorInterpolationFilters(),
+ svg->mColorInterpolationFilters, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentSVG->mColorInterpolationFilters,
+ NS_STYLE_COLOR_INTERPOLATION_LINEARRGB);
+
+ // fill:
+ SetSVGPaint(*aRuleData->ValueForFill(),
+ parentSVG->mFill, mPresContext, aContext,
+ svg->mFill, eStyleSVGPaintType_Color, conditions);
+
+ // fill-opacity: factor, inherit, initial,
+ // context-fill-opacity, context-stroke-opacity
+ nsStyleSVGOpacitySource contextFillOpacity = svg->FillOpacitySource();
+ SetSVGOpacity(*aRuleData->ValueForFillOpacity(),
+ svg->mFillOpacity, contextFillOpacity, conditions,
+ parentSVG->mFillOpacity, parentSVG->FillOpacitySource());
+ svg->SetFillOpacitySource(contextFillOpacity);
+
+ // fill-rule: enum, inherit, initial
+ SetValue(*aRuleData->ValueForFillRule(),
+ svg->mFillRule, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentSVG->mFillRule,
+ StyleFillRule::Nonzero);
+
+ // marker-end: url, none, inherit
+ const nsCSSValue* markerEndValue = aRuleData->ValueForMarkerEnd();
+ if (eCSSUnit_URL == markerEndValue->GetUnit()) {
+ svg->mMarkerEnd = markerEndValue->GetURLStructValue();
+ } else if (eCSSUnit_None == markerEndValue->GetUnit() ||
+ eCSSUnit_Initial == markerEndValue->GetUnit()) {
+ svg->mMarkerEnd = nullptr;
+ } else if (eCSSUnit_Inherit == markerEndValue->GetUnit() ||
+ eCSSUnit_Unset == markerEndValue->GetUnit()) {
+ conditions.SetUncacheable();
+ svg->mMarkerEnd = parentSVG->mMarkerEnd;
+ }
+
+ // marker-mid: url, none, inherit
+ const nsCSSValue* markerMidValue = aRuleData->ValueForMarkerMid();
+ if (eCSSUnit_URL == markerMidValue->GetUnit()) {
+ svg->mMarkerMid = markerMidValue->GetURLStructValue();
+ } else if (eCSSUnit_None == markerMidValue->GetUnit() ||
+ eCSSUnit_Initial == markerMidValue->GetUnit()) {
+ svg->mMarkerMid = nullptr;
+ } else if (eCSSUnit_Inherit == markerMidValue->GetUnit() ||
+ eCSSUnit_Unset == markerMidValue->GetUnit()) {
+ conditions.SetUncacheable();
+ svg->mMarkerMid = parentSVG->mMarkerMid;
+ }
+
+ // marker-start: url, none, inherit
+ const nsCSSValue* markerStartValue = aRuleData->ValueForMarkerStart();
+ if (eCSSUnit_URL == markerStartValue->GetUnit()) {
+ svg->mMarkerStart = markerStartValue->GetURLStructValue();
+ } else if (eCSSUnit_None == markerStartValue->GetUnit() ||
+ eCSSUnit_Initial == markerStartValue->GetUnit()) {
+ svg->mMarkerStart = nullptr;
+ } else if (eCSSUnit_Inherit == markerStartValue->GetUnit() ||
+ eCSSUnit_Unset == markerStartValue->GetUnit()) {
+ conditions.SetUncacheable();
+ svg->mMarkerStart = parentSVG->mMarkerStart;
+ }
+
+ // paint-order: enum (bit field), inherit, initial
+ const nsCSSValue* paintOrderValue = aRuleData->ValueForPaintOrder();
+ switch (paintOrderValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Enumerated:
+ static_assert
+ (NS_STYLE_PAINT_ORDER_BITWIDTH * NS_STYLE_PAINT_ORDER_LAST_VALUE <= 8,
+ "SVGStyleStruct::mPaintOrder not big enough");
+ svg->mPaintOrder = static_cast<uint8_t>(paintOrderValue->GetIntValue());
+ break;
+
+ case eCSSUnit_Inherit:
+ case eCSSUnit_Unset:
+ conditions.SetUncacheable();
+ svg->mPaintOrder = parentSVG->mPaintOrder;
+ break;
+
+ case eCSSUnit_Initial:
+ svg->mPaintOrder = NS_STYLE_PAINT_ORDER_NORMAL;
+ break;
+
+ default:
+ NS_NOTREACHED("unexpected unit");
+ }
+
+ // shape-rendering: enum, inherit
+ SetValue(*aRuleData->ValueForShapeRendering(),
+ svg->mShapeRendering, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentSVG->mShapeRendering,
+ NS_STYLE_SHAPE_RENDERING_AUTO);
+
+ // stroke:
+ SetSVGPaint(*aRuleData->ValueForStroke(),
+ parentSVG->mStroke, mPresContext, aContext,
+ svg->mStroke, eStyleSVGPaintType_None, conditions);
+
+ // stroke-dasharray: <dasharray>, none, inherit, context-value
+ const nsCSSValue* strokeDasharrayValue = aRuleData->ValueForStrokeDasharray();
+ switch (strokeDasharrayValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Inherit:
+ case eCSSUnit_Unset:
+ conditions.SetUncacheable();
+ svg->SetStrokeDasharrayFromObject(parentSVG->StrokeDasharrayFromObject());
+ svg->mStrokeDasharray = parentSVG->mStrokeDasharray;
+ break;
+
+ case eCSSUnit_Enumerated:
+ MOZ_ASSERT(strokeDasharrayValue->GetIntValue() ==
+ NS_STYLE_STROKE_PROP_CONTEXT_VALUE,
+ "Unknown keyword for stroke-dasharray");
+ svg->SetStrokeDasharrayFromObject(true);
+ svg->mStrokeDasharray.Clear();
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_None:
+ svg->SetStrokeDasharrayFromObject(false);
+ svg->mStrokeDasharray.Clear();
+ break;
+
+ case eCSSUnit_List:
+ case eCSSUnit_ListDep: {
+ svg->SetStrokeDasharrayFromObject(false);
+ svg->mStrokeDasharray.Clear();
+
+ // count number of values
+ const nsCSSValueList *value = strokeDasharrayValue->GetListValue();
+ uint32_t strokeDasharrayLength = ListLength(value);
+
+ MOZ_ASSERT(strokeDasharrayLength != 0, "no dasharray items");
+
+ svg->mStrokeDasharray.SetLength(strokeDasharrayLength);
+
+ uint32_t i = 0;
+ while (nullptr != value) {
+ SetCoord(value->mValue,
+ svg->mStrokeDasharray[i++], nsStyleCoord(),
+ SETCOORD_LP | SETCOORD_FACTOR,
+ aContext, mPresContext, conditions);
+ value = value->mNext;
+ }
+ break;
+ }
+
+ default:
+ MOZ_ASSERT(false, "unrecognized dasharray unit");
+ }
+
+ // stroke-dashoffset: <dashoffset>, inherit
+ const nsCSSValue *strokeDashoffsetValue =
+ aRuleData->ValueForStrokeDashoffset();
+ svg->SetStrokeDashoffsetFromObject(
+ strokeDashoffsetValue->GetUnit() == eCSSUnit_Enumerated &&
+ strokeDashoffsetValue->GetIntValue() == NS_STYLE_STROKE_PROP_CONTEXT_VALUE);
+ if (svg->StrokeDashoffsetFromObject()) {
+ svg->mStrokeDashoffset.SetCoordValue(0);
+ } else {
+ SetCoord(*aRuleData->ValueForStrokeDashoffset(),
+ svg->mStrokeDashoffset, parentSVG->mStrokeDashoffset,
+ SETCOORD_LPH | SETCOORD_FACTOR | SETCOORD_INITIAL_ZERO |
+ SETCOORD_UNSET_INHERIT,
+ aContext, mPresContext, conditions);
+ }
+
+ // stroke-linecap: enum, inherit, initial
+ SetValue(*aRuleData->ValueForStrokeLinecap(),
+ svg->mStrokeLinecap, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentSVG->mStrokeLinecap,
+ NS_STYLE_STROKE_LINECAP_BUTT);
+
+ // stroke-linejoin: enum, inherit, initial
+ SetValue(*aRuleData->ValueForStrokeLinejoin(),
+ svg->mStrokeLinejoin, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentSVG->mStrokeLinejoin,
+ NS_STYLE_STROKE_LINEJOIN_MITER);
+
+ // stroke-miterlimit: <miterlimit>, inherit
+ SetFactor(*aRuleData->ValueForStrokeMiterlimit(),
+ svg->mStrokeMiterlimit,
+ conditions,
+ parentSVG->mStrokeMiterlimit, 4.0f,
+ SETFCT_UNSET_INHERIT);
+
+ // stroke-opacity:
+ nsStyleSVGOpacitySource contextStrokeOpacity = svg->StrokeOpacitySource();
+ SetSVGOpacity(*aRuleData->ValueForStrokeOpacity(),
+ svg->mStrokeOpacity, contextStrokeOpacity, conditions,
+ parentSVG->mStrokeOpacity, parentSVG->StrokeOpacitySource());
+ svg->SetStrokeOpacitySource(contextStrokeOpacity);
+
+ // stroke-width:
+ const nsCSSValue* strokeWidthValue = aRuleData->ValueForStrokeWidth();
+ switch (strokeWidthValue->GetUnit()) {
+ case eCSSUnit_Enumerated:
+ MOZ_ASSERT(strokeWidthValue->GetIntValue() ==
+ NS_STYLE_STROKE_PROP_CONTEXT_VALUE,
+ "Unrecognized keyword for stroke-width");
+ svg->SetStrokeWidthFromObject(true);
+ svg->mStrokeWidth.SetCoordValue(nsPresContext::CSSPixelsToAppUnits(1));
+ break;
+
+ case eCSSUnit_Initial:
+ svg->SetStrokeWidthFromObject(false);
+ svg->mStrokeWidth.SetCoordValue(nsPresContext::CSSPixelsToAppUnits(1));
+ break;
+
+ default:
+ svg->SetStrokeWidthFromObject(false);
+ SetCoord(*strokeWidthValue,
+ svg->mStrokeWidth, parentSVG->mStrokeWidth,
+ SETCOORD_LPH | SETCOORD_FACTOR | SETCOORD_UNSET_INHERIT,
+ aContext, mPresContext, conditions);
+ }
+
+ // text-anchor: enum, inherit, initial
+ SetValue(*aRuleData->ValueForTextAnchor(),
+ svg->mTextAnchor, conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INHERIT,
+ parentSVG->mTextAnchor,
+ NS_STYLE_TEXT_ANCHOR_START);
+
+ COMPUTE_END_INHERITED(SVG, svg)
+}
+
+static already_AddRefed<StyleBasicShape>
+GetStyleBasicShapeFromCSSValue(const nsCSSValue& aValue,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+{
+ RefPtr<StyleBasicShape> basicShape;
+
+ nsCSSValue::Array* shapeFunction = aValue.GetArrayValue();
+ nsCSSKeyword functionName =
+ (nsCSSKeyword)shapeFunction->Item(0).GetIntValue();
+
+ if (functionName == eCSSKeyword_polygon) {
+ MOZ_ASSERT(!basicShape, "did not expect value");
+ basicShape = new StyleBasicShape(StyleBasicShapeType::Polygon);
+ MOZ_ASSERT(shapeFunction->Count() > 1,
+ "polygon has wrong number of arguments");
+ size_t j = 1;
+ if (shapeFunction->Item(j).GetUnit() == eCSSUnit_Enumerated) {
+ StyleFillRule rule;
+ SetEnumValueHelper::SetEnumeratedValue(rule, shapeFunction->Item(j));
+ basicShape->SetFillRule(rule);
+ ++j;
+ }
+ const int32_t mask = SETCOORD_PERCENT | SETCOORD_LENGTH |
+ SETCOORD_STORE_CALC;
+ const nsCSSValuePairList* curPair =
+ shapeFunction->Item(j).GetPairListValue();
+ nsTArray<nsStyleCoord>& coordinates = basicShape->Coordinates();
+ while (curPair) {
+ nsStyleCoord xCoord, yCoord;
+ DebugOnly<bool> didSetCoordX = SetCoord(curPair->mXValue, xCoord,
+ nsStyleCoord(), mask,
+ aStyleContext, aPresContext,
+ aConditions);
+ coordinates.AppendElement(xCoord);
+ MOZ_ASSERT(didSetCoordX, "unexpected x coordinate unit");
+ DebugOnly<bool> didSetCoordY = SetCoord(curPair->mYValue, yCoord,
+ nsStyleCoord(), mask,
+ aStyleContext, aPresContext,
+ aConditions);
+ coordinates.AppendElement(yCoord);
+ MOZ_ASSERT(didSetCoordY, "unexpected y coordinate unit");
+ curPair = curPair->mNext;
+ }
+ } else if (functionName == eCSSKeyword_circle ||
+ functionName == eCSSKeyword_ellipse) {
+ StyleBasicShapeType type = functionName == eCSSKeyword_circle ?
+ StyleBasicShapeType::Circle :
+ StyleBasicShapeType::Ellipse;
+ MOZ_ASSERT(!basicShape, "did not expect value");
+ basicShape = new StyleBasicShape(type);
+ const int32_t mask = SETCOORD_PERCENT | SETCOORD_LENGTH |
+ SETCOORD_STORE_CALC | SETCOORD_ENUMERATED;
+ size_t count = type == StyleBasicShapeType::Circle ? 2 : 3;
+ MOZ_ASSERT(shapeFunction->Count() == count + 1,
+ "unexpected arguments count");
+ MOZ_ASSERT(type == StyleBasicShapeType::Circle ||
+ (shapeFunction->Item(1).GetUnit() == eCSSUnit_Null) ==
+ (shapeFunction->Item(2).GetUnit() == eCSSUnit_Null),
+ "ellipse should have two radii or none");
+ for (size_t j = 1; j < count; ++j) {
+ const nsCSSValue& val = shapeFunction->Item(j);
+ nsStyleCoord radius;
+ if (val.GetUnit() != eCSSUnit_Null) {
+ DebugOnly<bool> didSetRadius = SetCoord(val, radius,
+ nsStyleCoord(), mask,
+ aStyleContext,
+ aPresContext,
+ aConditions);
+ MOZ_ASSERT(didSetRadius, "unexpected radius unit");
+ } else {
+ radius.SetIntValue(NS_RADIUS_CLOSEST_SIDE, eStyleUnit_Enumerated);
+ }
+ basicShape->Coordinates().AppendElement(radius);
+ }
+ const nsCSSValue& positionVal = shapeFunction->Item(count);
+ if (positionVal.GetUnit() == eCSSUnit_Array) {
+ ComputePositionValue(aStyleContext, positionVal,
+ basicShape->GetPosition(),
+ aConditions);
+ } else {
+ MOZ_ASSERT(positionVal.GetUnit() == eCSSUnit_Null,
+ "expected no value");
+ }
+ } else if (functionName == eCSSKeyword_inset) {
+ MOZ_ASSERT(!basicShape, "did not expect value");
+ basicShape = new StyleBasicShape(StyleBasicShapeType::Inset);
+ MOZ_ASSERT(shapeFunction->Count() == 6,
+ "inset function has wrong number of arguments");
+ MOZ_ASSERT(shapeFunction->Item(1).GetUnit() != eCSSUnit_Null,
+ "no shape arguments defined");
+ const int32_t mask = SETCOORD_PERCENT | SETCOORD_LENGTH |
+ SETCOORD_STORE_CALC;
+ nsTArray<nsStyleCoord>& coords = basicShape->Coordinates();
+ for (size_t j = 1; j <= 4; ++j) {
+ const nsCSSValue& val = shapeFunction->Item(j);
+ nsStyleCoord inset;
+ // Fill missing values to get 4 at the end.
+ if (val.GetUnit() == eCSSUnit_Null) {
+ if (j == 4) {
+ inset = coords[1];
+ } else {
+ MOZ_ASSERT(j != 1, "first argument not specified");
+ inset = coords[0];
+ }
+ } else {
+ DebugOnly<bool> didSetInset = SetCoord(val, inset,
+ nsStyleCoord(), mask,
+ aStyleContext, aPresContext,
+ aConditions);
+ MOZ_ASSERT(didSetInset, "unexpected inset unit");
+ }
+ coords.AppendElement(inset);
+ }
+
+ nsStyleCorners& insetRadius = basicShape->GetRadius();
+ if (shapeFunction->Item(5).GetUnit() == eCSSUnit_Array) {
+ nsCSSValue::Array* radiiArray = shapeFunction->Item(5).GetArrayValue();
+ NS_FOR_CSS_FULL_CORNERS(corner) {
+ int cx = NS_FULL_TO_HALF_CORNER(corner, false);
+ int cy = NS_FULL_TO_HALF_CORNER(corner, true);
+ const nsCSSValue& radius = radiiArray->Item(corner);
+ nsStyleCoord coordX, coordY;
+ DebugOnly<bool> didSetRadii = SetPairCoords(radius, coordX, coordY,
+ nsStyleCoord(),
+ nsStyleCoord(), mask,
+ aStyleContext,
+ aPresContext,
+ aConditions);
+ MOZ_ASSERT(didSetRadii, "unexpected radius unit");
+ insetRadius.Set(cx, coordX);
+ insetRadius.Set(cy, coordY);
+ }
+ } else {
+ MOZ_ASSERT(shapeFunction->Item(5).GetUnit() == eCSSUnit_Null,
+ "unexpected value");
+ // Initialize border-radius
+ nsStyleCoord zero;
+ zero.SetCoordValue(0);
+ NS_FOR_CSS_HALF_CORNERS(j) {
+ insetRadius.Set(j, zero);
+ }
+ }
+ } else {
+ NS_NOTREACHED("unexpected basic shape function");
+ }
+
+ return basicShape.forget();
+}
+
+template<typename ReferenceBox>
+static void
+SetStyleShapeSourceToCSSValue(
+ StyleShapeSource<ReferenceBox>* aShapeSource,
+ const nsCSSValue* aValue,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+{
+ MOZ_ASSERT(aValue->GetUnit() == eCSSUnit_Array,
+ "expected a basic shape or reference box");
+
+ const nsCSSValue::Array* array = aValue->GetArrayValue();
+ MOZ_ASSERT(array->Count() == 1 || array->Count() == 2,
+ "Expect one or both of a shape function and a reference box");
+
+ ReferenceBox referenceBox = ReferenceBox::NoBox;
+ RefPtr<StyleBasicShape> basicShape;
+
+ for (size_t i = 0; i < array->Count(); ++i) {
+ const nsCSSValue& item = array->Item(i);
+ if (item.GetUnit() == eCSSUnit_Enumerated) {
+ referenceBox = static_cast<ReferenceBox>(item.GetIntValue());
+ } else if (item.GetUnit() == eCSSUnit_Function) {
+ basicShape = GetStyleBasicShapeFromCSSValue(item, aStyleContext,
+ aPresContext, aConditions);
+ } else {
+ MOZ_ASSERT_UNREACHABLE("Unexpected unit!");
+ return;
+ }
+ }
+
+ if (basicShape) {
+ aShapeSource->SetBasicShape(basicShape, referenceBox);
+ } else {
+ aShapeSource->SetReferenceBox(referenceBox);
+ }
+}
+
+// Returns true if the nsStyleFilter was successfully set using the nsCSSValue.
+static bool
+SetStyleFilterToCSSValue(nsStyleFilter* aStyleFilter,
+ const nsCSSValue& aValue,
+ nsStyleContext* aStyleContext,
+ nsPresContext* aPresContext,
+ RuleNodeCacheConditions& aConditions)
+{
+ nsCSSUnit unit = aValue.GetUnit();
+ if (unit == eCSSUnit_URL) {
+ return aStyleFilter->SetURL(aValue.GetURLStructValue());
+ }
+
+ MOZ_ASSERT(unit == eCSSUnit_Function, "expected a filter function");
+
+ nsCSSValue::Array* filterFunction = aValue.GetArrayValue();
+ nsCSSKeyword functionName =
+ (nsCSSKeyword)filterFunction->Item(0).GetIntValue();
+
+ int32_t type;
+ DebugOnly<bool> foundKeyword =
+ nsCSSProps::FindKeyword(functionName,
+ nsCSSProps::kFilterFunctionKTable,
+ type);
+ MOZ_ASSERT(foundKeyword, "unknown filter type");
+ if (type == NS_STYLE_FILTER_DROP_SHADOW) {
+ RefPtr<nsCSSShadowArray> shadowArray = GetShadowData(
+ filterFunction->Item(1).GetListValue(),
+ aStyleContext,
+ false,
+ aPresContext,
+ aConditions);
+ aStyleFilter->SetDropShadow(shadowArray);
+ return true;
+ }
+
+ int32_t mask = SETCOORD_PERCENT | SETCOORD_FACTOR;
+ if (type == NS_STYLE_FILTER_BLUR) {
+ mask = SETCOORD_LENGTH |
+ SETCOORD_CALC_LENGTH_ONLY |
+ SETCOORD_CALC_CLAMP_NONNEGATIVE;
+ } else if (type == NS_STYLE_FILTER_HUE_ROTATE) {
+ mask = SETCOORD_ANGLE;
+ }
+
+ MOZ_ASSERT(filterFunction->Count() == 2,
+ "all filter functions should have exactly one argument");
+
+ nsCSSValue& arg = filterFunction->Item(1);
+ nsStyleCoord filterParameter;
+ DebugOnly<bool> didSetCoord = SetCoord(arg, filterParameter,
+ nsStyleCoord(), mask,
+ aStyleContext, aPresContext,
+ aConditions);
+ aStyleFilter->SetFilterParameter(filterParameter, type);
+ MOZ_ASSERT(didSetCoord, "unexpected unit");
+ return true;
+}
+
+const void*
+nsRuleNode::ComputeSVGResetData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(SVGReset, svgReset, parentSVGReset)
+
+ // stop-color:
+ const nsCSSValue* stopColorValue = aRuleData->ValueForStopColor();
+ if (eCSSUnit_Initial == stopColorValue->GetUnit() ||
+ eCSSUnit_Unset == stopColorValue->GetUnit()) {
+ svgReset->mStopColor = NS_RGB(0, 0, 0);
+ } else {
+ SetColor(*stopColorValue, parentSVGReset->mStopColor,
+ mPresContext, aContext, svgReset->mStopColor, conditions);
+ }
+
+ // flood-color:
+ const nsCSSValue* floodColorValue = aRuleData->ValueForFloodColor();
+ if (eCSSUnit_Initial == floodColorValue->GetUnit() ||
+ eCSSUnit_Unset == floodColorValue->GetUnit()) {
+ svgReset->mFloodColor = NS_RGB(0, 0, 0);
+ } else {
+ SetColor(*floodColorValue, parentSVGReset->mFloodColor,
+ mPresContext, aContext, svgReset->mFloodColor, conditions);
+ }
+
+ // lighting-color:
+ const nsCSSValue* lightingColorValue = aRuleData->ValueForLightingColor();
+ if (eCSSUnit_Initial == lightingColorValue->GetUnit() ||
+ eCSSUnit_Unset == lightingColorValue->GetUnit()) {
+ svgReset->mLightingColor = NS_RGB(255, 255, 255);
+ } else {
+ SetColor(*lightingColorValue, parentSVGReset->mLightingColor,
+ mPresContext, aContext, svgReset->mLightingColor,
+ conditions);
+ }
+
+ // clip-path: url, <basic-shape> || <geometry-box>, none, inherit
+ const nsCSSValue* clipPathValue = aRuleData->ValueForClipPath();
+ switch (clipPathValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_None:
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ svgReset->mClipPath = StyleClipPath();
+ break;
+ case eCSSUnit_Inherit:
+ conditions.SetUncacheable();
+ svgReset->mClipPath = parentSVGReset->mClipPath;
+ break;
+ case eCSSUnit_URL: {
+ svgReset->mClipPath = StyleClipPath();
+ svgReset->mClipPath.SetURL(clipPathValue->GetURLStructValue());
+ break;
+ }
+ case eCSSUnit_Array: {
+ svgReset->mClipPath = StyleClipPath();
+ SetStyleShapeSourceToCSSValue(&svgReset->mClipPath, clipPathValue, aContext,
+ mPresContext, conditions);
+ break;
+ }
+ default:
+ NS_NOTREACHED("unexpected unit");
+ }
+
+ // stop-opacity:
+ SetFactor(*aRuleData->ValueForStopOpacity(),
+ svgReset->mStopOpacity, conditions,
+ parentSVGReset->mStopOpacity, 1.0f,
+ SETFCT_OPACITY | SETFCT_UNSET_INITIAL);
+
+ // flood-opacity:
+ SetFactor(*aRuleData->ValueForFloodOpacity(),
+ svgReset->mFloodOpacity, conditions,
+ parentSVGReset->mFloodOpacity, 1.0f,
+ SETFCT_OPACITY | SETFCT_UNSET_INITIAL);
+
+ // dominant-baseline: enum, inherit, initial
+ SetValue(*aRuleData->ValueForDominantBaseline(),
+ svgReset->mDominantBaseline,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentSVGReset->mDominantBaseline,
+ NS_STYLE_DOMINANT_BASELINE_AUTO);
+
+ // vector-effect: enum, inherit, initial
+ SetValue(*aRuleData->ValueForVectorEffect(),
+ svgReset->mVectorEffect,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentSVGReset->mVectorEffect,
+ NS_STYLE_VECTOR_EFFECT_NONE);
+
+ // mask-type: enum, inherit, initial
+ SetValue(*aRuleData->ValueForMaskType(),
+ svgReset->mMaskType,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentSVGReset->mMaskType,
+ NS_STYLE_MASK_TYPE_LUMINANCE);
+
+#ifdef MOZ_ENABLE_MASK_AS_SHORTHAND
+ uint32_t maxItemCount = 1;
+ bool rebuild = false;
+
+ // mask-image: none | <url> | <image-list> | <element-reference> | <gradient>
+ nsStyleImage initialImage;
+ SetImageLayerList(aContext, *aRuleData->ValueForMaskImage(),
+ svgReset->mMask.mLayers,
+ parentSVGReset->mMask.mLayers,
+ &nsStyleImageLayers::Layer::mImage,
+ initialImage, parentSVGReset->mMask.mImageCount,
+ svgReset->mMask.mImageCount,
+ maxItemCount, rebuild, conditions);
+ SetImageLayerList(aContext, *aRuleData->ValueForMaskImage(),
+ svgReset->mMask.mLayers,
+ parentSVGReset->mMask.mLayers,
+ &nsStyleImageLayers::Layer::mSourceURI,
+ RefPtr<css::URLValueData>(),
+ parentSVGReset->mMask.mImageCount,
+ svgReset->mMask.mImageCount,
+ maxItemCount, rebuild, conditions);
+
+ // mask-repeat: enum, inherit, initial [pair list]
+ nsStyleImageLayers::Repeat initialRepeat;
+ initialRepeat.SetInitialValues();
+ SetImageLayerPairList(aContext, *aRuleData->ValueForMaskRepeat(),
+ svgReset->mMask.mLayers,
+ parentSVGReset->mMask.mLayers,
+ &nsStyleImageLayers::Layer::mRepeat,
+ initialRepeat, parentSVGReset->mMask.mRepeatCount,
+ svgReset->mMask.mRepeatCount, maxItemCount, rebuild,
+ conditions);
+
+ // mask-clip: enum, inherit, initial [list]
+ SetImageLayerList(aContext, *aRuleData->ValueForMaskClip(),
+ svgReset->mMask.mLayers,
+ parentSVGReset->mMask.mLayers,
+ &nsStyleImageLayers::Layer::mClip,
+ uint8_t(NS_STYLE_IMAGELAYER_CLIP_BORDER),
+ parentSVGReset->mMask.mClipCount,
+ svgReset->mMask.mClipCount, maxItemCount, rebuild,
+ conditions);
+
+ // mask-origin: enum, inherit, initial [list]
+ SetImageLayerList(aContext, *aRuleData->ValueForMaskOrigin(),
+ svgReset->mMask.mLayers,
+ parentSVGReset->mMask.mLayers,
+ &nsStyleImageLayers::Layer::mOrigin,
+ uint8_t(NS_STYLE_IMAGELAYER_ORIGIN_BORDER),
+ parentSVGReset->mMask.mOriginCount,
+ svgReset->mMask.mOriginCount, maxItemCount, rebuild,
+ conditions);
+
+ // mask-position-x/y: enum, length, percent (flags), inherit [list]
+ Position::Coord initialPositionCoord;
+ initialPositionCoord.mPercent = 0.0f;
+ initialPositionCoord.mLength = 0;
+ initialPositionCoord.mHasPercent = true;
+
+ SetImageLayerPositionCoordList(
+ aContext, *aRuleData->ValueForMaskPositionX(),
+ svgReset->mMask.mLayers,
+ parentSVGReset->mMask.mLayers,
+ &Position::mXPosition,
+ initialPositionCoord, parentSVGReset->mMask.mPositionXCount,
+ svgReset->mMask.mPositionXCount, maxItemCount, rebuild,
+ conditions);
+ SetImageLayerPositionCoordList(
+ aContext, *aRuleData->ValueForMaskPositionY(),
+ svgReset->mMask.mLayers,
+ parentSVGReset->mMask.mLayers,
+ &Position::mYPosition,
+ initialPositionCoord, parentSVGReset->mMask.mPositionYCount,
+ svgReset->mMask.mPositionYCount, maxItemCount, rebuild,
+ conditions);
+
+ // mask-size: enum, length, auto, inherit, initial [pair list]
+ nsStyleImageLayers::Size initialSize;
+ initialSize.SetInitialValues();
+ SetImageLayerPairList(aContext, *aRuleData->ValueForMaskSize(),
+ svgReset->mMask.mLayers,
+ parentSVGReset->mMask.mLayers,
+ &nsStyleImageLayers::Layer::mSize,
+ initialSize, parentSVGReset->mMask.mSizeCount,
+ svgReset->mMask.mSizeCount, maxItemCount, rebuild,
+ conditions);
+
+ // mask-mode: enum, inherit, initial [list]
+ SetImageLayerList(aContext, *aRuleData->ValueForMaskMode(),
+ svgReset->mMask.mLayers,
+ parentSVGReset->mMask.mLayers,
+ &nsStyleImageLayers::Layer::mMaskMode,
+ uint8_t(NS_STYLE_MASK_MODE_MATCH_SOURCE),
+ parentSVGReset->mMask.mMaskModeCount,
+ svgReset->mMask.mMaskModeCount, maxItemCount, rebuild, conditions);
+
+ // mask-composite: enum, inherit, initial [list]
+ SetImageLayerList(aContext, *aRuleData->ValueForMaskComposite(),
+ svgReset->mMask.mLayers,
+ parentSVGReset->mMask.mLayers,
+ &nsStyleImageLayers::Layer::mComposite,
+ uint8_t(NS_STYLE_MASK_COMPOSITE_ADD),
+ parentSVGReset->mMask.mCompositeCount,
+ svgReset->mMask.mCompositeCount, maxItemCount, rebuild, conditions);
+
+ if (rebuild) {
+ FillAllBackgroundLists(svgReset->mMask, maxItemCount);
+ }
+#else
+ // mask: none | <url>
+ const nsCSSValue* maskValue = aRuleData->ValueForMask();
+ if (eCSSUnit_URL == maskValue->GetUnit()) {
+ svgReset->mMask.mLayers[0].mSourceURI = maskValue->GetURLStructValue();
+ } else if (eCSSUnit_None == maskValue->GetUnit() ||
+ eCSSUnit_Initial == maskValue->GetUnit() ||
+ eCSSUnit_Unset == maskValue->GetUnit()) {
+ svgReset->mMask.mLayers[0].mSourceURI = nullptr;
+ } else if (eCSSUnit_Inherit == maskValue->GetUnit()) {
+ conditions.SetUncacheable();
+ svgReset->mMask.mLayers[0].mSourceURI =
+ parentSVGReset->mMask.mLayers[0].mSourceURI;
+ }
+#endif
+
+ COMPUTE_END_RESET(SVGReset, svgReset)
+}
+
+const void*
+nsRuleNode::ComputeVariablesData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_INHERITED(Variables, variables, parentVariables)
+
+ MOZ_ASSERT(aRuleData->mVariables,
+ "shouldn't be in ComputeVariablesData if there were no variable "
+ "declarations specified");
+
+ CSSVariableResolver resolver(&variables->mVariables);
+ resolver.Resolve(&parentVariables->mVariables,
+ aRuleData->mVariables);
+ conditions.SetUncacheable();
+
+ COMPUTE_END_INHERITED(Variables, variables)
+}
+
+const void*
+nsRuleNode::ComputeEffectsData(void* aStartStruct,
+ const nsRuleData* aRuleData,
+ nsStyleContext* aContext,
+ nsRuleNode* aHighestNode,
+ const RuleDetail aRuleDetail,
+ const RuleNodeCacheConditions aConditions)
+{
+ COMPUTE_START_RESET(Effects, effects, parentEffects)
+
+ // filter: url, none, inherit
+ const nsCSSValue* filterValue = aRuleData->ValueForFilter();
+ switch (filterValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+ case eCSSUnit_None:
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ effects->mFilters.Clear();
+ break;
+ case eCSSUnit_Inherit:
+ conditions.SetUncacheable();
+ effects->mFilters = parentEffects->mFilters;
+ break;
+ case eCSSUnit_List:
+ case eCSSUnit_ListDep: {
+ effects->mFilters.Clear();
+ const nsCSSValueList* cur = filterValue->GetListValue();
+ while (cur) {
+ nsStyleFilter styleFilter;
+ if (!SetStyleFilterToCSSValue(&styleFilter, cur->mValue, aContext,
+ mPresContext, conditions)) {
+ effects->mFilters.Clear();
+ break;
+ }
+ MOZ_ASSERT(styleFilter.GetType() != NS_STYLE_FILTER_NONE,
+ "filter should be set");
+ effects->mFilters.AppendElement(styleFilter);
+ cur = cur->mNext;
+ }
+ break;
+ }
+ default:
+ NS_NOTREACHED("unexpected unit");
+ }
+
+ // box-shadow: none, list, inherit, initial
+ const nsCSSValue* boxShadowValue = aRuleData->ValueForBoxShadow();
+ switch (boxShadowValue->GetUnit()) {
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_None:
+ effects->mBoxShadow = nullptr;
+ break;
+
+ case eCSSUnit_Inherit:
+ effects->mBoxShadow = parentEffects->mBoxShadow;
+ conditions.SetUncacheable();
+ break;
+
+ case eCSSUnit_List:
+ case eCSSUnit_ListDep:
+ effects->mBoxShadow = GetShadowData(boxShadowValue->GetListValue(),
+ aContext, true, mPresContext, conditions);
+ break;
+
+ default:
+ MOZ_ASSERT(false, "unrecognized shadow unit");
+ }
+
+ // clip property: length, auto, inherit
+ const nsCSSValue* clipValue = aRuleData->ValueForClip();
+ switch (clipValue->GetUnit()) {
+ case eCSSUnit_Inherit:
+ conditions.SetUncacheable();
+ effects->mClipFlags = parentEffects->mClipFlags;
+ effects->mClip = parentEffects->mClip;
+ break;
+
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_Auto:
+ effects->mClipFlags = NS_STYLE_CLIP_AUTO;
+ effects->mClip.SetRect(0,0,0,0);
+ break;
+
+ case eCSSUnit_Null:
+ break;
+
+ case eCSSUnit_Rect: {
+ const nsCSSRect& clipRect = clipValue->GetRectValue();
+
+ effects->mClipFlags = NS_STYLE_CLIP_RECT;
+
+ if (clipRect.mTop.GetUnit() == eCSSUnit_Auto) {
+ effects->mClip.y = 0;
+ effects->mClipFlags |= NS_STYLE_CLIP_TOP_AUTO;
+ }
+ else if (clipRect.mTop.IsLengthUnit()) {
+ effects->mClip.y = CalcLength(clipRect.mTop, aContext,
+ mPresContext, conditions);
+ }
+
+ if (clipRect.mBottom.GetUnit() == eCSSUnit_Auto) {
+ // Setting to NS_MAXSIZE for the 'auto' case ensures that
+ // the clip rect is nonempty. It is important that mClip be
+ // nonempty if the actual clip rect could be nonempty.
+ effects->mClip.height = NS_MAXSIZE;
+ effects->mClipFlags |= NS_STYLE_CLIP_BOTTOM_AUTO;
+ }
+ else if (clipRect.mBottom.IsLengthUnit()) {
+ effects->mClip.height = CalcLength(clipRect.mBottom, aContext,
+ mPresContext, conditions) -
+ effects->mClip.y;
+ }
+
+ if (clipRect.mLeft.GetUnit() == eCSSUnit_Auto) {
+ effects->mClip.x = 0;
+ effects->mClipFlags |= NS_STYLE_CLIP_LEFT_AUTO;
+ }
+ else if (clipRect.mLeft.IsLengthUnit()) {
+ effects->mClip.x = CalcLength(clipRect.mLeft, aContext,
+ mPresContext, conditions);
+ }
+
+ if (clipRect.mRight.GetUnit() == eCSSUnit_Auto) {
+ // Setting to NS_MAXSIZE for the 'auto' case ensures that
+ // the clip rect is nonempty. It is important that mClip be
+ // nonempty if the actual clip rect could be nonempty.
+ effects->mClip.width = NS_MAXSIZE;
+ effects->mClipFlags |= NS_STYLE_CLIP_RIGHT_AUTO;
+ }
+ else if (clipRect.mRight.IsLengthUnit()) {
+ effects->mClip.width = CalcLength(clipRect.mRight, aContext,
+ mPresContext, conditions) -
+ effects->mClip.x;
+ }
+ break;
+ }
+
+ default:
+ MOZ_ASSERT(false, "unrecognized clip unit");
+ }
+
+ // opacity: factor, inherit, initial
+ SetFactor(*aRuleData->ValueForOpacity(), effects->mOpacity, conditions,
+ parentEffects->mOpacity, 1.0f,
+ SETFCT_OPACITY | SETFCT_UNSET_INITIAL);
+
+ // mix-blend-mode: enum, inherit, initial
+ SetValue(*aRuleData->ValueForMixBlendMode(), effects->mMixBlendMode,
+ conditions,
+ SETVAL_ENUMERATED | SETVAL_UNSET_INITIAL,
+ parentEffects->mMixBlendMode, NS_STYLE_BLEND_NORMAL);
+
+ COMPUTE_END_RESET(Effects, effects)
+}
+
+const void*
+nsRuleNode::GetStyleData(nsStyleStructID aSID,
+ nsStyleContext* aContext,
+ bool aComputeData)
+{
+ NS_ASSERTION(IsUsedDirectly(),
+ "if we ever call this on rule nodes that aren't used "
+ "directly, we should adjust handling of mDependentBits "
+ "in some way.");
+ MOZ_ASSERT(!aContext->GetCachedStyleData(aSID),
+ "style context should not have cached data for struct");
+
+ const void *data;
+
+ // Never use cached data for animated style inside a pseudo-element;
+ // see comment on cacheability in AnimValuesStyleRule::MapRuleInfoInto.
+ if (!(HasAnimationData() && ParentHasPseudoElementData(aContext))) {
+ data = mStyleData.GetStyleData(aSID, aContext, aComputeData);
+ if (MOZ_LIKELY(data != nullptr)) {
+ // For inherited structs, mark the struct (which will be set on
+ // the context by our caller) as not being owned by the context.
+ if (!nsCachedStyleData::IsReset(aSID)) {
+ aContext->AddStyleBit(nsCachedStyleData::GetBitForSID(aSID));
+ } else if (HasAnimationData()) {
+ // If we have animation data, the struct should be cached on the style
+ // context so that we can peek the struct.
+ // See comment in AnimValuesStyleRule::MapRuleInfoInto.
+ StoreStyleOnContext(aContext, aSID, const_cast<void*>(data));
+ }
+
+ return data; // We have a fully specified struct. Just return it.
+ }
+ }
+
+ if (MOZ_UNLIKELY(!aComputeData))
+ return nullptr;
+
+ // Nothing is cached. We'll have to delve further and examine our rules.
+ data = WalkRuleTree(aSID, aContext);
+
+ MOZ_ASSERT(data, "should have aborted on out-of-memory");
+ return data;
+}
+
+void
+nsRuleNode::GetDiscretelyAnimatedCSSValue(nsCSSPropertyID aProperty,
+ nsCSSValue* aValue)
+{
+ for (nsRuleNode* node = this; node; node = node->GetParent()) {
+ nsIStyleRule* rule = node->GetRule();
+ if (!rule) {
+ continue;
+ }
+ if (rule->GetDiscretelyAnimatedCSSValue(aProperty, aValue)) {
+ return;
+ }
+ }
+}
+
+/* static */ bool
+nsRuleNode::HasAuthorSpecifiedRules(nsStyleContext* aStyleContext,
+ uint32_t ruleTypeMask,
+ bool aAuthorColorsAllowed)
+{
+#ifdef MOZ_STYLO
+ if (aStyleContext->StyleSource().IsServoComputedValues()) {
+ NS_WARNING("stylo: nsRuleNode::HasAuthorSpecifiedRules not implemented");
+ return true;
+ }
+#endif
+
+ uint32_t inheritBits = 0;
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_BACKGROUND)
+ inheritBits |= NS_STYLE_INHERIT_BIT(Background);
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_BORDER)
+ inheritBits |= NS_STYLE_INHERIT_BIT(Border);
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_PADDING)
+ inheritBits |= NS_STYLE_INHERIT_BIT(Padding);
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_TEXT_SHADOW)
+ inheritBits |= NS_STYLE_INHERIT_BIT(Text);
+
+ // properties in the SIDS, whether or not we care about them
+ size_t nprops = 0,
+ backgroundOffset, borderOffset, paddingOffset, textShadowOffset;
+
+ // We put the reset properties the start of the nsCSSValue array....
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_BACKGROUND) {
+ backgroundOffset = nprops;
+ nprops += nsCSSProps::PropertyCountInStruct(eStyleStruct_Background);
+ }
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_BORDER) {
+ borderOffset = nprops;
+ nprops += nsCSSProps::PropertyCountInStruct(eStyleStruct_Border);
+ }
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_PADDING) {
+ paddingOffset = nprops;
+ nprops += nsCSSProps::PropertyCountInStruct(eStyleStruct_Padding);
+ }
+
+ // ...and the inherited properties at the end of the array.
+ size_t inheritedOffset = nprops;
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_TEXT_SHADOW) {
+ textShadowOffset = nprops;
+ nprops += nsCSSProps::PropertyCountInStruct(eStyleStruct_Text);
+ }
+
+ void* dataStorage = alloca(nprops * sizeof(nsCSSValue));
+ AutoCSSValueArray dataArray(dataStorage, nprops);
+
+ /* We're relying on the use of |aStyleContext| not mutating it! */
+ nsRuleData ruleData(inheritBits, dataArray.get(),
+ aStyleContext->PresContext(), aStyleContext);
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_BACKGROUND) {
+ ruleData.mValueOffsets[eStyleStruct_Background] = backgroundOffset;
+ }
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_BORDER) {
+ ruleData.mValueOffsets[eStyleStruct_Border] = borderOffset;
+ }
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_PADDING) {
+ ruleData.mValueOffsets[eStyleStruct_Padding] = paddingOffset;
+ }
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_TEXT_SHADOW) {
+ ruleData.mValueOffsets[eStyleStruct_Text] = textShadowOffset;
+ }
+
+ static const nsCSSPropertyID backgroundValues[] = {
+ eCSSProperty_background_color,
+ eCSSProperty_background_image,
+ };
+
+ static const nsCSSPropertyID borderValues[] = {
+ eCSSProperty_border_top_color,
+ eCSSProperty_border_top_style,
+ eCSSProperty_border_top_width,
+ eCSSProperty_border_right_color,
+ eCSSProperty_border_right_style,
+ eCSSProperty_border_right_width,
+ eCSSProperty_border_bottom_color,
+ eCSSProperty_border_bottom_style,
+ eCSSProperty_border_bottom_width,
+ eCSSProperty_border_left_color,
+ eCSSProperty_border_left_style,
+ eCSSProperty_border_left_width,
+ eCSSProperty_border_top_left_radius,
+ eCSSProperty_border_top_right_radius,
+ eCSSProperty_border_bottom_right_radius,
+ eCSSProperty_border_bottom_left_radius,
+ };
+
+ static const nsCSSPropertyID paddingValues[] = {
+ eCSSProperty_padding_top,
+ eCSSProperty_padding_right,
+ eCSSProperty_padding_bottom,
+ eCSSProperty_padding_left,
+ };
+
+ static const nsCSSPropertyID textShadowValues[] = {
+ eCSSProperty_text_shadow
+ };
+
+ // Number of properties we care about
+ size_t nValues = 0;
+
+ nsCSSValue* values[MOZ_ARRAY_LENGTH(backgroundValues) +
+ MOZ_ARRAY_LENGTH(borderValues) +
+ MOZ_ARRAY_LENGTH(paddingValues) +
+ MOZ_ARRAY_LENGTH(textShadowValues)];
+
+ nsCSSPropertyID properties[MOZ_ARRAY_LENGTH(backgroundValues) +
+ MOZ_ARRAY_LENGTH(borderValues) +
+ MOZ_ARRAY_LENGTH(paddingValues) +
+ MOZ_ARRAY_LENGTH(textShadowValues)];
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_BACKGROUND) {
+ for (uint32_t i = 0, i_end = ArrayLength(backgroundValues);
+ i < i_end; ++i) {
+ properties[nValues] = backgroundValues[i];
+ values[nValues++] = ruleData.ValueFor(backgroundValues[i]);
+ }
+ }
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_BORDER) {
+ for (uint32_t i = 0, i_end = ArrayLength(borderValues);
+ i < i_end; ++i) {
+ properties[nValues] = borderValues[i];
+ values[nValues++] = ruleData.ValueFor(borderValues[i]);
+ }
+ }
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_PADDING) {
+ for (uint32_t i = 0, i_end = ArrayLength(paddingValues);
+ i < i_end; ++i) {
+ properties[nValues] = paddingValues[i];
+ values[nValues++] = ruleData.ValueFor(paddingValues[i]);
+ }
+ }
+
+ if (ruleTypeMask & NS_AUTHOR_SPECIFIED_TEXT_SHADOW) {
+ for (uint32_t i = 0, i_end = ArrayLength(textShadowValues);
+ i < i_end; ++i) {
+ properties[nValues] = textShadowValues[i];
+ values[nValues++] = ruleData.ValueFor(textShadowValues[i]);
+ }
+ }
+
+ nsStyleContext* styleContext = aStyleContext;
+
+ // We need to be careful not to count styles covered up by user-important or
+ // UA-important declarations. But we do want to catch explicit inherit
+ // styling in those and check our parent style context to see whether we have
+ // user styling for those properties. Note that we don't care here about
+ // inheritance due to lack of a specified value, since all the properties we
+ // care about are reset properties.
+ bool haveExplicitUAInherit;
+ do {
+ haveExplicitUAInherit = false;
+ for (nsRuleNode* ruleNode = styleContext->RuleNode(); ruleNode;
+ ruleNode = ruleNode->GetParent()) {
+ nsIStyleRule *rule = ruleNode->GetRule();
+ if (rule) {
+ ruleData.mLevel = ruleNode->GetLevel();
+ ruleData.mIsImportantRule = ruleNode->IsImportantRule();
+
+ rule->MapRuleInfoInto(&ruleData);
+
+ if (ruleData.mLevel == SheetType::Agent ||
+ ruleData.mLevel == SheetType::User) {
+ // This is a rule whose effect we want to ignore, so if any of
+ // the properties we care about were set, set them to the dummy
+ // value that they'll never otherwise get.
+ for (uint32_t i = 0; i < nValues; ++i) {
+ nsCSSUnit unit = values[i]->GetUnit();
+ if (unit != eCSSUnit_Null &&
+ unit != eCSSUnit_Dummy &&
+ unit != eCSSUnit_DummyInherit) {
+ if (unit == eCSSUnit_Inherit ||
+ (i >= inheritedOffset && unit == eCSSUnit_Unset)) {
+ haveExplicitUAInherit = true;
+ values[i]->SetDummyInheritValue();
+ } else {
+ values[i]->SetDummyValue();
+ }
+ }
+ }
+ } else {
+ // If any of the values we care about was set by the above rule,
+ // we have author style.
+ for (uint32_t i = 0; i < nValues; ++i) {
+ if (values[i]->GetUnit() != eCSSUnit_Null &&
+ values[i]->GetUnit() != eCSSUnit_Dummy && // see above
+ values[i]->GetUnit() != eCSSUnit_DummyInherit) {
+ // If author colors are not allowed, only claim to have
+ // author-specified rules if we're looking at a non-color
+ // property or if we're looking at the background color and it's
+ // set to transparent. Anything else should get set to a dummy
+ // value instead.
+ if (aAuthorColorsAllowed ||
+ !nsCSSProps::PropHasFlags(properties[i],
+ CSS_PROPERTY_IGNORED_WHEN_COLORS_DISABLED) ||
+ (properties[i] == eCSSProperty_background_color &&
+ !values[i]->IsNonTransparentColor())) {
+ return true;
+ }
+
+ values[i]->SetDummyValue();
+ }
+ }
+ }
+ }
+ }
+
+ if (haveExplicitUAInherit) {
+ // reset all the eCSSUnit_Null values to eCSSUnit_Dummy (since they're
+ // not styled by the author, or by anyone else), and then reset all the
+ // eCSSUnit_DummyInherit values to eCSSUnit_Null (so we will be able to
+ // detect them being styled by the author) and move up to our parent
+ // style context.
+ for (uint32_t i = 0; i < nValues; ++i)
+ if (values[i]->GetUnit() == eCSSUnit_Null)
+ values[i]->SetDummyValue();
+ for (uint32_t i = 0; i < nValues; ++i)
+ if (values[i]->GetUnit() == eCSSUnit_DummyInherit)
+ values[i]->Reset();
+ styleContext = styleContext->GetParent();
+ }
+ } while (haveExplicitUAInherit && styleContext);
+
+ return false;
+}
+
+/* static */ void
+nsRuleNode::ComputePropertiesOverridingAnimation(
+ const nsTArray<nsCSSPropertyID>& aProperties,
+ nsStyleContext* aStyleContext,
+ nsCSSPropertyIDSet& aPropertiesOverridden)
+{
+ /*
+ * Set up an nsRuleData with all the structs needed for all of the
+ * properties in aProperties.
+ */
+ uint32_t structBits = 0;
+ size_t nprops = 0;
+ size_t offsets[nsStyleStructID_Length];
+ for (size_t propIdx = 0, propEnd = aProperties.Length();
+ propIdx < propEnd; ++propIdx) {
+ nsCSSPropertyID prop = aProperties[propIdx];
+ nsStyleStructID sid = nsCSSProps::kSIDTable[prop];
+ uint32_t bit = nsCachedStyleData::GetBitForSID(sid);
+ if (!(structBits & bit)) {
+ structBits |= bit;
+ offsets[sid] = nprops;
+ nprops += nsCSSProps::PropertyCountInStruct(sid);
+ }
+ }
+
+ void* dataStorage = alloca(nprops * sizeof(nsCSSValue));
+ AutoCSSValueArray dataArray(dataStorage, nprops);
+
+ // We're relying on the use of |aStyleContext| not mutating it!
+ nsRuleData ruleData(structBits, dataArray.get(),
+ aStyleContext->PresContext(), aStyleContext);
+ for (nsStyleStructID sid = nsStyleStructID(0);
+ sid < nsStyleStructID_Length; sid = nsStyleStructID(sid + 1)) {
+ if (structBits & nsCachedStyleData::GetBitForSID(sid)) {
+ ruleData.mValueOffsets[sid] = offsets[sid];
+ }
+ }
+
+ /*
+ * Actually walk up the rule tree until we're someplace less
+ * specific than animations.
+ */
+ for (nsRuleNode* ruleNode = aStyleContext->RuleNode(); ruleNode;
+ ruleNode = ruleNode->GetParent()) {
+ nsIStyleRule *rule = ruleNode->GetRule();
+ if (rule) {
+ ruleData.mLevel = ruleNode->GetLevel();
+ ruleData.mIsImportantRule = ruleNode->IsImportantRule();
+
+ // Transitions are the only non-!important level overriding
+ // animations in the cascade ordering. They also don't actually
+ // override animations, since transitions are suppressed when both
+ // are present. And since we might not have called
+ // UpdateCascadeResults (which updates when they are suppressed
+ // due to the presence of animations for the same element and
+ // property) for transitions yet (which will make their
+ // MapRuleInfoInto skip the properties that are currently
+ // animating), we should skip them explicitly.
+ if (ruleData.mLevel == SheetType::Transition) {
+ continue;
+ }
+
+ if (!ruleData.mIsImportantRule) {
+ // We're now equal to or less than the animation level; stop.
+ break;
+ }
+
+ rule->MapRuleInfoInto(&ruleData);
+ }
+ }
+
+ /*
+ * Fill in which properties were overridden.
+ */
+ for (size_t propIdx = 0, propEnd = aProperties.Length();
+ propIdx < propEnd; ++propIdx) {
+ nsCSSPropertyID prop = aProperties[propIdx];
+ if (ruleData.ValueFor(prop)->GetUnit() != eCSSUnit_Null) {
+ aPropertiesOverridden.AddProperty(prop);
+ }
+ }
+}
+
+/* static */
+bool
+nsRuleNode::ComputeColor(const nsCSSValue& aValue, nsPresContext* aPresContext,
+ nsStyleContext* aStyleContext, nscolor& aResult)
+{
+ MOZ_ASSERT(aValue.GetUnit() != eCSSUnit_Inherit,
+ "aValue shouldn't have eCSSUnit_Inherit");
+ MOZ_ASSERT(aValue.GetUnit() != eCSSUnit_Initial,
+ "aValue shouldn't have eCSSUnit_Initial");
+ MOZ_ASSERT(aValue.GetUnit() != eCSSUnit_Unset,
+ "aValue shouldn't have eCSSUnit_Unset");
+
+ RuleNodeCacheConditions conditions;
+ bool ok = SetColor(aValue, NS_RGB(0, 0, 0), aPresContext, aStyleContext,
+ aResult, conditions);
+ MOZ_ASSERT(ok || !(aPresContext && aStyleContext));
+ return ok;
+}
+
+/* static */ bool
+nsRuleNode::ParentHasPseudoElementData(nsStyleContext* aContext)
+{
+ nsStyleContext* parent = aContext->GetParent();
+ return parent && parent->HasPseudoElementData();
+}
+
+/* static */ void
+nsRuleNode::StoreStyleOnContext(nsStyleContext* aContext,
+ nsStyleStructID aSID,
+ void* aStruct)
+{
+ aContext->AddStyleBit(nsCachedStyleData::GetBitForSID(aSID));
+ aContext->SetStyle(aSID, aStruct);
+}
+
+#ifdef DEBUG
+bool
+nsRuleNode::ContextHasCachedData(nsStyleContext* aContext,
+ nsStyleStructID aSID)
+{
+ return !!aContext->GetCachedStyleData(aSID);
+}
+#endif