summaryrefslogtreecommitdiffstats
path: root/layout/style/nsCSSValue.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'layout/style/nsCSSValue.cpp')
-rw-r--r--layout/style/nsCSSValue.cpp3237
1 files changed, 3237 insertions, 0 deletions
diff --git a/layout/style/nsCSSValue.cpp b/layout/style/nsCSSValue.cpp
new file mode 100644
index 000000000..baf5b7897
--- /dev/null
+++ b/layout/style/nsCSSValue.cpp
@@ -0,0 +1,3237 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set ts=2 et sw=2 tw=80: */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+/* representation of simple property values within CSS declarations */
+
+#include "nsCSSValue.h"
+
+#include "mozilla/StyleSheetInlines.h"
+#include "mozilla/Likely.h"
+#include "mozilla/MemoryReporting.h"
+#include "mozilla/Move.h"
+#include "mozilla/css/ImageLoader.h"
+#include "CSSCalc.h"
+#include "gfxFontConstants.h"
+#include "imgIRequest.h"
+#include "imgRequestProxy.h"
+#include "nsIDocument.h"
+#include "nsIPrincipal.h"
+#include "nsCSSProps.h"
+#include "nsNetUtil.h"
+#include "nsPresContext.h"
+#include "nsStyleUtil.h"
+#include "nsDeviceContext.h"
+#include "nsStyleSet.h"
+#include "nsContentUtils.h"
+
+using namespace mozilla;
+
+static bool
+IsLocalRefURL(nsStringBuffer* aString)
+{
+ // Find the first non-"C0 controls + space" character.
+ char16_t* current = static_cast<char16_t*>(aString->Data());
+ for (; *current != '\0'; current++) {
+ if (*current > 0x20) {
+ // if the first non-"C0 controls + space" character is '#', this is a
+ // local-ref URL.
+ return *current == '#';
+ }
+ }
+
+ return false;
+}
+
+nsCSSValue::nsCSSValue(int32_t aValue, nsCSSUnit aUnit)
+ : mUnit(aUnit)
+{
+ MOZ_ASSERT(aUnit == eCSSUnit_Integer || aUnit == eCSSUnit_Enumerated ||
+ aUnit == eCSSUnit_EnumColor,
+ "not an int value");
+ if (aUnit == eCSSUnit_Integer || aUnit == eCSSUnit_Enumerated ||
+ aUnit == eCSSUnit_EnumColor) {
+ mValue.mInt = aValue;
+ }
+ else {
+ mUnit = eCSSUnit_Null;
+ mValue.mInt = 0;
+ }
+}
+
+nsCSSValue::nsCSSValue(float aValue, nsCSSUnit aUnit)
+ : mUnit(aUnit)
+{
+ MOZ_ASSERT(eCSSUnit_Percent <= aUnit, "not a float value");
+ if (eCSSUnit_Percent <= aUnit) {
+ mValue.mFloat = aValue;
+ MOZ_ASSERT(!mozilla::IsNaN(mValue.mFloat));
+ }
+ else {
+ mUnit = eCSSUnit_Null;
+ mValue.mInt = 0;
+ }
+}
+
+nsCSSValue::nsCSSValue(const nsString& aValue, nsCSSUnit aUnit)
+ : mUnit(aUnit)
+{
+ MOZ_ASSERT(UnitHasStringValue(), "not a string value");
+ if (UnitHasStringValue()) {
+ mValue.mString = BufferFromString(aValue).take();
+ }
+ else {
+ mUnit = eCSSUnit_Null;
+ mValue.mInt = 0;
+ }
+}
+
+nsCSSValue::nsCSSValue(nsCSSValue::Array* aValue, nsCSSUnit aUnit)
+ : mUnit(aUnit)
+{
+ MOZ_ASSERT(UnitHasArrayValue(), "bad unit");
+ mValue.mArray = aValue;
+ mValue.mArray->AddRef();
+}
+
+nsCSSValue::nsCSSValue(mozilla::css::URLValue* aValue)
+ : mUnit(eCSSUnit_URL)
+{
+ mValue.mURL = aValue;
+ mValue.mURL->AddRef();
+}
+
+nsCSSValue::nsCSSValue(mozilla::css::ImageValue* aValue)
+ : mUnit(eCSSUnit_Image)
+{
+ mValue.mImage = aValue;
+ mValue.mImage->AddRef();
+}
+
+nsCSSValue::nsCSSValue(nsCSSValueGradient* aValue)
+ : mUnit(eCSSUnit_Gradient)
+{
+ mValue.mGradient = aValue;
+ mValue.mGradient->AddRef();
+}
+
+nsCSSValue::nsCSSValue(nsCSSValueTokenStream* aValue)
+ : mUnit(eCSSUnit_TokenStream)
+{
+ mValue.mTokenStream = aValue;
+ mValue.mTokenStream->AddRef();
+}
+
+nsCSSValue::nsCSSValue(mozilla::css::GridTemplateAreasValue* aValue)
+ : mUnit(eCSSUnit_GridTemplateAreas)
+{
+ mValue.mGridTemplateAreas = aValue;
+ mValue.mGridTemplateAreas->AddRef();
+}
+
+nsCSSValue::nsCSSValue(css::FontFamilyListRefCnt* aValue)
+ : mUnit(eCSSUnit_FontFamilyList)
+{
+ mValue.mFontFamilyList = aValue;
+ mValue.mFontFamilyList->AddRef();
+}
+
+nsCSSValue::nsCSSValue(const nsCSSValue& aCopy)
+ : mUnit(aCopy.mUnit)
+{
+ if (mUnit <= eCSSUnit_DummyInherit) {
+ // nothing to do, but put this important case first
+ }
+ else if (eCSSUnit_Percent <= mUnit) {
+ mValue.mFloat = aCopy.mValue.mFloat;
+ MOZ_ASSERT(!mozilla::IsNaN(mValue.mFloat));
+ }
+ else if (UnitHasStringValue()) {
+ mValue.mString = aCopy.mValue.mString;
+ mValue.mString->AddRef();
+ }
+ else if (eCSSUnit_Integer <= mUnit && mUnit <= eCSSUnit_EnumColor) {
+ mValue.mInt = aCopy.mValue.mInt;
+ }
+ else if (IsIntegerColorUnit()) {
+ mValue.mColor = aCopy.mValue.mColor;
+ }
+ else if (IsFloatColorUnit()) {
+ mValue.mFloatColor = aCopy.mValue.mFloatColor;
+ mValue.mFloatColor->AddRef();
+ }
+ else if (eCSSUnit_ComplexColor == mUnit) {
+ mValue.mComplexColor = aCopy.mValue.mComplexColor;
+ mValue.mComplexColor->AddRef();
+ }
+ else if (UnitHasArrayValue()) {
+ mValue.mArray = aCopy.mValue.mArray;
+ mValue.mArray->AddRef();
+ }
+ else if (eCSSUnit_URL == mUnit) {
+ mValue.mURL = aCopy.mValue.mURL;
+ mValue.mURL->AddRef();
+ }
+ else if (eCSSUnit_Image == mUnit) {
+ mValue.mImage = aCopy.mValue.mImage;
+ mValue.mImage->AddRef();
+ }
+ else if (eCSSUnit_Gradient == mUnit) {
+ mValue.mGradient = aCopy.mValue.mGradient;
+ mValue.mGradient->AddRef();
+ }
+ else if (eCSSUnit_TokenStream == mUnit) {
+ mValue.mTokenStream = aCopy.mValue.mTokenStream;
+ mValue.mTokenStream->AddRef();
+ }
+ else if (eCSSUnit_Pair == mUnit) {
+ mValue.mPair = aCopy.mValue.mPair;
+ mValue.mPair->AddRef();
+ }
+ else if (eCSSUnit_Triplet == mUnit) {
+ mValue.mTriplet = aCopy.mValue.mTriplet;
+ mValue.mTriplet->AddRef();
+ }
+ else if (eCSSUnit_Rect == mUnit) {
+ mValue.mRect = aCopy.mValue.mRect;
+ mValue.mRect->AddRef();
+ }
+ else if (eCSSUnit_List == mUnit) {
+ mValue.mList = aCopy.mValue.mList;
+ mValue.mList->AddRef();
+ }
+ else if (eCSSUnit_ListDep == mUnit) {
+ mValue.mListDependent = aCopy.mValue.mListDependent;
+ }
+ else if (eCSSUnit_SharedList == mUnit) {
+ mValue.mSharedList = aCopy.mValue.mSharedList;
+ mValue.mSharedList->AddRef();
+ }
+ else if (eCSSUnit_PairList == mUnit) {
+ mValue.mPairList = aCopy.mValue.mPairList;
+ mValue.mPairList->AddRef();
+ }
+ else if (eCSSUnit_PairListDep == mUnit) {
+ mValue.mPairListDependent = aCopy.mValue.mPairListDependent;
+ }
+ else if (eCSSUnit_GridTemplateAreas == mUnit) {
+ mValue.mGridTemplateAreas = aCopy.mValue.mGridTemplateAreas;
+ mValue.mGridTemplateAreas->AddRef();
+ }
+ else if (eCSSUnit_FontFamilyList == mUnit) {
+ mValue.mFontFamilyList = aCopy.mValue.mFontFamilyList;
+ mValue.mFontFamilyList->AddRef();
+ }
+ else {
+ MOZ_ASSERT(false, "unknown unit");
+ }
+}
+
+nsCSSValue& nsCSSValue::operator=(const nsCSSValue& aCopy)
+{
+ if (this != &aCopy) {
+ Reset();
+ new (this) nsCSSValue(aCopy);
+ }
+ return *this;
+}
+
+nsCSSValue&
+nsCSSValue::operator=(nsCSSValue&& aOther)
+{
+ MOZ_ASSERT(this != &aOther, "Self assigment with rvalue reference");
+
+ Reset();
+ mUnit = aOther.mUnit;
+ mValue = aOther.mValue;
+ aOther.mUnit = eCSSUnit_Null;
+
+ return *this;
+}
+
+bool nsCSSValue::operator==(const nsCSSValue& aOther) const
+{
+ MOZ_ASSERT(mUnit != eCSSUnit_ListDep &&
+ aOther.mUnit != eCSSUnit_ListDep &&
+ mUnit != eCSSUnit_PairListDep &&
+ aOther.mUnit != eCSSUnit_PairListDep,
+ "don't use operator== with dependent lists");
+
+ if (mUnit == aOther.mUnit) {
+ if (mUnit <= eCSSUnit_DummyInherit) {
+ return true;
+ }
+ else if (UnitHasStringValue()) {
+ return (NS_strcmp(GetBufferValue(mValue.mString),
+ GetBufferValue(aOther.mValue.mString)) == 0);
+ }
+ else if ((eCSSUnit_Integer <= mUnit) && (mUnit <= eCSSUnit_EnumColor)) {
+ return mValue.mInt == aOther.mValue.mInt;
+ }
+ else if (IsIntegerColorUnit()) {
+ return mValue.mColor == aOther.mValue.mColor;
+ }
+ else if (IsFloatColorUnit()) {
+ return *mValue.mFloatColor == *aOther.mValue.mFloatColor;
+ }
+ else if (eCSSUnit_ComplexColor == mUnit) {
+ return *mValue.mComplexColor == *aOther.mValue.mComplexColor;
+ }
+ else if (UnitHasArrayValue()) {
+ return *mValue.mArray == *aOther.mValue.mArray;
+ }
+ else if (eCSSUnit_URL == mUnit) {
+ return mValue.mURL->Equals(*aOther.mValue.mURL);
+ }
+ else if (eCSSUnit_Image == mUnit) {
+ return mValue.mImage->Equals(*aOther.mValue.mImage);
+ }
+ else if (eCSSUnit_Gradient == mUnit) {
+ return *mValue.mGradient == *aOther.mValue.mGradient;
+ }
+ else if (eCSSUnit_TokenStream == mUnit) {
+ return *mValue.mTokenStream == *aOther.mValue.mTokenStream;
+ }
+ else if (eCSSUnit_Pair == mUnit) {
+ return *mValue.mPair == *aOther.mValue.mPair;
+ }
+ else if (eCSSUnit_Triplet == mUnit) {
+ return *mValue.mTriplet == *aOther.mValue.mTriplet;
+ }
+ else if (eCSSUnit_Rect == mUnit) {
+ return *mValue.mRect == *aOther.mValue.mRect;
+ }
+ else if (eCSSUnit_List == mUnit) {
+ return nsCSSValueList::Equal(mValue.mList, aOther.mValue.mList);
+ }
+ else if (eCSSUnit_SharedList == mUnit) {
+ return *mValue.mSharedList == *aOther.mValue.mSharedList;
+ }
+ else if (eCSSUnit_PairList == mUnit) {
+ return nsCSSValuePairList::Equal(mValue.mPairList,
+ aOther.mValue.mPairList);
+ }
+ else if (eCSSUnit_GridTemplateAreas == mUnit) {
+ return *mValue.mGridTemplateAreas == *aOther.mValue.mGridTemplateAreas;
+ }
+ else if (eCSSUnit_FontFamilyList == mUnit) {
+ return *mValue.mFontFamilyList == *aOther.mValue.mFontFamilyList;
+ }
+ else {
+ return mValue.mFloat == aOther.mValue.mFloat;
+ }
+ }
+ return false;
+}
+
+double
+nsCSSValue::GetAngleValueInRadians() const
+{
+ double angle = GetFloatValue();
+
+ switch (GetUnit()) {
+ case eCSSUnit_Radian: return angle;
+ case eCSSUnit_Turn: return angle * 2 * M_PI;
+ case eCSSUnit_Degree: return angle * M_PI / 180.0;
+ case eCSSUnit_Grad: return angle * M_PI / 200.0;
+
+ default:
+ MOZ_ASSERT(false, "unrecognized angular unit");
+ return 0.0;
+ }
+}
+
+double
+nsCSSValue::GetAngleValueInDegrees() const
+{
+ double angle = GetFloatValue();
+
+ switch (GetUnit()) {
+ case eCSSUnit_Degree: return angle;
+ case eCSSUnit_Grad: return angle * 0.9; // grad / 400 * 360
+ case eCSSUnit_Radian: return angle * 180.0 / M_PI; // rad / 2pi * 360
+ case eCSSUnit_Turn: return angle * 360.0;
+
+ default:
+ MOZ_ASSERT(false, "unrecognized angular unit");
+ return 0.0;
+ }
+}
+
+imgRequestProxy* nsCSSValue::GetImageValue(nsIDocument* aDocument) const
+{
+ MOZ_ASSERT(mUnit == eCSSUnit_Image, "not an Image value");
+ return mValue.mImage->mRequests.GetWeak(aDocument);
+}
+
+already_AddRefed<imgRequestProxy>
+nsCSSValue::GetPossiblyStaticImageValue(nsIDocument* aDocument,
+ nsPresContext* aPresContext) const
+{
+ imgRequestProxy* req = GetImageValue(aDocument);
+ if (aPresContext->IsDynamic()) {
+ return do_AddRef(req);
+ }
+ return nsContentUtils::GetStaticRequest(req);
+}
+
+nscoord nsCSSValue::GetFixedLength(nsPresContext* aPresContext) const
+{
+ MOZ_ASSERT(mUnit == eCSSUnit_PhysicalMillimeter,
+ "not a fixed length unit");
+
+ float inches = mValue.mFloat / MM_PER_INCH_FLOAT;
+ return NSToCoordFloorClamped(inches *
+ float(aPresContext->DeviceContext()->AppUnitsPerPhysicalInch()));
+}
+
+nscoord nsCSSValue::GetPixelLength() const
+{
+ MOZ_ASSERT(IsPixelLengthUnit(), "not a fixed length unit");
+
+ double scaleFactor;
+ switch (mUnit) {
+ case eCSSUnit_Pixel: return nsPresContext::CSSPixelsToAppUnits(mValue.mFloat);
+ case eCSSUnit_Pica: scaleFactor = 16.0; break;
+ case eCSSUnit_Point: scaleFactor = 4/3.0; break;
+ case eCSSUnit_Inch: scaleFactor = 96.0; break;
+ case eCSSUnit_Millimeter: scaleFactor = 96/25.4; break;
+ case eCSSUnit_Centimeter: scaleFactor = 96/2.54; break;
+ case eCSSUnit_Quarter: scaleFactor = 96/101.6; break;
+ default:
+ NS_ERROR("should never get here");
+ return 0;
+ }
+ return nsPresContext::CSSPixelsToAppUnits(float(mValue.mFloat*scaleFactor));
+}
+
+void nsCSSValue::DoReset()
+{
+ if (UnitHasStringValue()) {
+ mValue.mString->Release();
+ } else if (IsFloatColorUnit()) {
+ mValue.mFloatColor->Release();
+ } else if (eCSSUnit_ComplexColor == mUnit) {
+ mValue.mComplexColor->Release();
+ } else if (UnitHasArrayValue()) {
+ mValue.mArray->Release();
+ } else if (eCSSUnit_URL == mUnit) {
+ mValue.mURL->Release();
+ } else if (eCSSUnit_Image == mUnit) {
+ mValue.mImage->Release();
+ } else if (eCSSUnit_Gradient == mUnit) {
+ mValue.mGradient->Release();
+ } else if (eCSSUnit_TokenStream == mUnit) {
+ mValue.mTokenStream->Release();
+ } else if (eCSSUnit_Pair == mUnit) {
+ mValue.mPair->Release();
+ } else if (eCSSUnit_Triplet == mUnit) {
+ mValue.mTriplet->Release();
+ } else if (eCSSUnit_Rect == mUnit) {
+ mValue.mRect->Release();
+ } else if (eCSSUnit_List == mUnit) {
+ mValue.mList->Release();
+ } else if (eCSSUnit_SharedList == mUnit) {
+ mValue.mSharedList->Release();
+ } else if (eCSSUnit_PairList == mUnit) {
+ mValue.mPairList->Release();
+ } else if (eCSSUnit_GridTemplateAreas == mUnit) {
+ mValue.mGridTemplateAreas->Release();
+ } else if (eCSSUnit_FontFamilyList == mUnit) {
+ mValue.mFontFamilyList->Release();
+ }
+ mUnit = eCSSUnit_Null;
+}
+
+void nsCSSValue::SetIntValue(int32_t aValue, nsCSSUnit aUnit)
+{
+ MOZ_ASSERT(aUnit == eCSSUnit_Integer || aUnit == eCSSUnit_Enumerated ||
+ aUnit == eCSSUnit_EnumColor,
+ "not an int value");
+ Reset();
+ if (aUnit == eCSSUnit_Integer || aUnit == eCSSUnit_Enumerated ||
+ aUnit == eCSSUnit_EnumColor) {
+ mUnit = aUnit;
+ mValue.mInt = aValue;
+ }
+}
+
+void nsCSSValue::SetPercentValue(float aValue)
+{
+ Reset();
+ mUnit = eCSSUnit_Percent;
+ mValue.mFloat = aValue;
+ MOZ_ASSERT(!mozilla::IsNaN(mValue.mFloat));
+}
+
+void nsCSSValue::SetFloatValue(float aValue, nsCSSUnit aUnit)
+{
+ MOZ_ASSERT(IsFloatUnit(aUnit), "not a float value");
+ Reset();
+ if (IsFloatUnit(aUnit)) {
+ mUnit = aUnit;
+ mValue.mFloat = aValue;
+ MOZ_ASSERT(!mozilla::IsNaN(mValue.mFloat));
+ }
+}
+
+void nsCSSValue::SetStringValue(const nsString& aValue,
+ nsCSSUnit aUnit)
+{
+ Reset();
+ mUnit = aUnit;
+ MOZ_ASSERT(UnitHasStringValue(), "not a string unit");
+ if (UnitHasStringValue()) {
+ mValue.mString = BufferFromString(aValue).take();
+ } else
+ mUnit = eCSSUnit_Null;
+}
+
+void nsCSSValue::SetColorValue(nscolor aValue)
+{
+ SetIntegerColorValue(aValue, eCSSUnit_RGBAColor);
+}
+
+
+
+void nsCSSValue::SetIntegerColorValue(nscolor aValue, nsCSSUnit aUnit)
+{
+ Reset();
+ mUnit = aUnit;
+ MOZ_ASSERT(IsIntegerColorUnit(), "bad unit");
+ mValue.mColor = aValue;
+}
+
+void nsCSSValue::SetIntegerCoordValue(nscoord aValue)
+{
+ SetFloatValue(nsPresContext::AppUnitsToFloatCSSPixels(aValue),
+ eCSSUnit_Pixel);
+}
+
+void nsCSSValue::SetFloatColorValue(float aComponent1,
+ float aComponent2,
+ float aComponent3,
+ float aAlpha,
+ nsCSSUnit aUnit)
+{
+ Reset();
+ mUnit = aUnit;
+ MOZ_ASSERT(IsFloatColorUnit(), "bad unit");
+ mValue.mFloatColor =
+ new nsCSSValueFloatColor(aComponent1, aComponent2, aComponent3, aAlpha);
+ mValue.mFloatColor->AddRef();
+}
+
+void
+nsCSSValue::SetRGBAColorValue(const RGBAColorData& aValue)
+{
+ SetFloatColorValue(aValue.mR, aValue.mG, aValue.mB,
+ aValue.mA, eCSSUnit_PercentageRGBAColor);
+}
+
+void
+nsCSSValue::SetComplexColorValue(already_AddRefed<ComplexColorValue> aValue)
+{
+ Reset();
+ mUnit = eCSSUnit_ComplexColor;
+ mValue.mComplexColor = aValue.take();
+}
+
+void nsCSSValue::SetArrayValue(nsCSSValue::Array* aValue, nsCSSUnit aUnit)
+{
+ Reset();
+ mUnit = aUnit;
+ MOZ_ASSERT(UnitHasArrayValue(), "bad unit");
+ mValue.mArray = aValue;
+ mValue.mArray->AddRef();
+}
+
+void nsCSSValue::SetURLValue(mozilla::css::URLValue* aValue)
+{
+ Reset();
+ mUnit = eCSSUnit_URL;
+ mValue.mURL = aValue;
+ mValue.mURL->AddRef();
+}
+
+void nsCSSValue::SetImageValue(mozilla::css::ImageValue* aValue)
+{
+ Reset();
+ mUnit = eCSSUnit_Image;
+ mValue.mImage = aValue;
+ mValue.mImage->AddRef();
+}
+
+void nsCSSValue::SetGradientValue(nsCSSValueGradient* aValue)
+{
+ Reset();
+ mUnit = eCSSUnit_Gradient;
+ mValue.mGradient = aValue;
+ mValue.mGradient->AddRef();
+}
+
+void nsCSSValue::SetTokenStreamValue(nsCSSValueTokenStream* aValue)
+{
+ Reset();
+ mUnit = eCSSUnit_TokenStream;
+ mValue.mTokenStream = aValue;
+ mValue.mTokenStream->AddRef();
+}
+
+void nsCSSValue::SetGridTemplateAreas(mozilla::css::GridTemplateAreasValue* aValue)
+{
+ Reset();
+ mUnit = eCSSUnit_GridTemplateAreas;
+ mValue.mGridTemplateAreas = aValue;
+ mValue.mGridTemplateAreas->AddRef();
+}
+
+void nsCSSValue::SetFontFamilyListValue(css::FontFamilyListRefCnt* aValue)
+{
+ Reset();
+ mUnit = eCSSUnit_FontFamilyList;
+ mValue.mFontFamilyList = aValue;
+ mValue.mFontFamilyList->AddRef();
+}
+
+void nsCSSValue::SetPairValue(const nsCSSValuePair* aValue)
+{
+ // pairs should not be used for null/inherit/initial values
+ MOZ_ASSERT(aValue &&
+ aValue->mXValue.GetUnit() != eCSSUnit_Null &&
+ aValue->mYValue.GetUnit() != eCSSUnit_Null &&
+ aValue->mXValue.GetUnit() != eCSSUnit_Inherit &&
+ aValue->mYValue.GetUnit() != eCSSUnit_Inherit &&
+ aValue->mXValue.GetUnit() != eCSSUnit_Initial &&
+ aValue->mYValue.GetUnit() != eCSSUnit_Initial &&
+ aValue->mXValue.GetUnit() != eCSSUnit_Unset &&
+ aValue->mYValue.GetUnit() != eCSSUnit_Unset,
+ "missing or inappropriate pair value");
+ Reset();
+ mUnit = eCSSUnit_Pair;
+ mValue.mPair = new nsCSSValuePair_heap(aValue->mXValue, aValue->mYValue);
+ mValue.mPair->AddRef();
+}
+
+void nsCSSValue::SetPairValue(const nsCSSValue& xValue,
+ const nsCSSValue& yValue)
+{
+ MOZ_ASSERT(xValue.GetUnit() != eCSSUnit_Null &&
+ yValue.GetUnit() != eCSSUnit_Null &&
+ xValue.GetUnit() != eCSSUnit_Inherit &&
+ yValue.GetUnit() != eCSSUnit_Inherit &&
+ xValue.GetUnit() != eCSSUnit_Initial &&
+ yValue.GetUnit() != eCSSUnit_Initial &&
+ xValue.GetUnit() != eCSSUnit_Unset &&
+ yValue.GetUnit() != eCSSUnit_Unset,
+ "inappropriate pair value");
+ Reset();
+ mUnit = eCSSUnit_Pair;
+ mValue.mPair = new nsCSSValuePair_heap(xValue, yValue);
+ mValue.mPair->AddRef();
+}
+
+void nsCSSValue::SetTripletValue(const nsCSSValueTriplet* aValue)
+{
+ // triplet should not be used for null/inherit/initial values
+ MOZ_ASSERT(aValue &&
+ aValue->mXValue.GetUnit() != eCSSUnit_Null &&
+ aValue->mYValue.GetUnit() != eCSSUnit_Null &&
+ aValue->mZValue.GetUnit() != eCSSUnit_Null &&
+ aValue->mXValue.GetUnit() != eCSSUnit_Inherit &&
+ aValue->mYValue.GetUnit() != eCSSUnit_Inherit &&
+ aValue->mZValue.GetUnit() != eCSSUnit_Inherit &&
+ aValue->mXValue.GetUnit() != eCSSUnit_Initial &&
+ aValue->mYValue.GetUnit() != eCSSUnit_Initial &&
+ aValue->mZValue.GetUnit() != eCSSUnit_Initial &&
+ aValue->mXValue.GetUnit() != eCSSUnit_Unset &&
+ aValue->mYValue.GetUnit() != eCSSUnit_Unset &&
+ aValue->mZValue.GetUnit() != eCSSUnit_Unset,
+ "missing or inappropriate triplet value");
+ Reset();
+ mUnit = eCSSUnit_Triplet;
+ mValue.mTriplet = new nsCSSValueTriplet_heap(aValue->mXValue, aValue->mYValue, aValue->mZValue);
+ mValue.mTriplet->AddRef();
+}
+
+void nsCSSValue::SetTripletValue(const nsCSSValue& xValue,
+ const nsCSSValue& yValue,
+ const nsCSSValue& zValue)
+{
+ // Only allow Null for the z component
+ MOZ_ASSERT(xValue.GetUnit() != eCSSUnit_Null &&
+ yValue.GetUnit() != eCSSUnit_Null &&
+ xValue.GetUnit() != eCSSUnit_Inherit &&
+ yValue.GetUnit() != eCSSUnit_Inherit &&
+ zValue.GetUnit() != eCSSUnit_Inherit &&
+ xValue.GetUnit() != eCSSUnit_Initial &&
+ yValue.GetUnit() != eCSSUnit_Initial &&
+ zValue.GetUnit() != eCSSUnit_Initial &&
+ xValue.GetUnit() != eCSSUnit_Unset &&
+ yValue.GetUnit() != eCSSUnit_Unset &&
+ zValue.GetUnit() != eCSSUnit_Unset,
+ "inappropriate triplet value");
+ Reset();
+ mUnit = eCSSUnit_Triplet;
+ mValue.mTriplet = new nsCSSValueTriplet_heap(xValue, yValue, zValue);
+ mValue.mTriplet->AddRef();
+}
+
+nsCSSRect& nsCSSValue::SetRectValue()
+{
+ Reset();
+ mUnit = eCSSUnit_Rect;
+ mValue.mRect = new nsCSSRect_heap;
+ mValue.mRect->AddRef();
+ return *mValue.mRect;
+}
+
+nsCSSValueList* nsCSSValue::SetListValue()
+{
+ Reset();
+ mUnit = eCSSUnit_List;
+ mValue.mList = new nsCSSValueList_heap;
+ mValue.mList->AddRef();
+ return mValue.mList;
+}
+
+void nsCSSValue::SetSharedListValue(nsCSSValueSharedList* aList)
+{
+ Reset();
+ mUnit = eCSSUnit_SharedList;
+ mValue.mSharedList = aList;
+ mValue.mSharedList->AddRef();
+}
+
+void nsCSSValue::SetDependentListValue(nsCSSValueList* aList)
+{
+ Reset();
+ if (aList) {
+ mUnit = eCSSUnit_ListDep;
+ mValue.mListDependent = aList;
+ }
+}
+
+void
+nsCSSValue::AdoptListValue(UniquePtr<nsCSSValueList> aValue)
+{
+ // We have to copy the first element since for owned lists the first
+ // element should be an nsCSSValueList_heap object.
+ SetListValue();
+ mValue.mList->mValue = Move(aValue->mValue);
+ mValue.mList->mNext = aValue->mNext;
+ aValue->mNext = nullptr;
+ aValue.reset();
+}
+
+nsCSSValuePairList* nsCSSValue::SetPairListValue()
+{
+ Reset();
+ mUnit = eCSSUnit_PairList;
+ mValue.mPairList = new nsCSSValuePairList_heap;
+ mValue.mPairList->AddRef();
+ return mValue.mPairList;
+}
+
+void nsCSSValue::SetDependentPairListValue(nsCSSValuePairList* aList)
+{
+ Reset();
+ if (aList) {
+ mUnit = eCSSUnit_PairListDep;
+ mValue.mPairListDependent = aList;
+ }
+}
+
+void
+nsCSSValue::AdoptPairListValue(UniquePtr<nsCSSValuePairList> aValue)
+{
+ // We have to copy the first element, since for owned pair lists, the first
+ // element should be an nsCSSValuePairList_heap object.
+ SetPairListValue();
+ mValue.mPairList->mXValue = Move(aValue->mXValue);
+ mValue.mPairList->mYValue = Move(aValue->mYValue);
+ mValue.mPairList->mNext = aValue->mNext;
+ aValue->mNext = nullptr;
+ aValue.reset();
+}
+
+void nsCSSValue::SetAutoValue()
+{
+ Reset();
+ mUnit = eCSSUnit_Auto;
+}
+
+void nsCSSValue::SetInheritValue()
+{
+ Reset();
+ mUnit = eCSSUnit_Inherit;
+}
+
+void nsCSSValue::SetInitialValue()
+{
+ Reset();
+ mUnit = eCSSUnit_Initial;
+}
+
+void nsCSSValue::SetUnsetValue()
+{
+ Reset();
+ mUnit = eCSSUnit_Unset;
+}
+
+void nsCSSValue::SetNoneValue()
+{
+ Reset();
+ mUnit = eCSSUnit_None;
+}
+
+void nsCSSValue::SetAllValue()
+{
+ Reset();
+ mUnit = eCSSUnit_All;
+}
+
+void nsCSSValue::SetNormalValue()
+{
+ Reset();
+ mUnit = eCSSUnit_Normal;
+}
+
+void nsCSSValue::SetSystemFontValue()
+{
+ Reset();
+ mUnit = eCSSUnit_System_Font;
+}
+
+void nsCSSValue::SetDummyValue()
+{
+ Reset();
+ mUnit = eCSSUnit_Dummy;
+}
+
+void nsCSSValue::SetDummyInheritValue()
+{
+ Reset();
+ mUnit = eCSSUnit_DummyInherit;
+}
+
+void nsCSSValue::SetCalcValue(const nsStyleCoord::CalcValue* aCalc)
+{
+ RefPtr<nsCSSValue::Array> arr = nsCSSValue::Array::Create(1);
+ if (!aCalc->mHasPercent) {
+ arr->Item(0).SetIntegerCoordValue(aCalc->mLength);
+ } else {
+ nsCSSValue::Array *arr2 = nsCSSValue::Array::Create(2);
+ arr->Item(0).SetArrayValue(arr2, eCSSUnit_Calc_Plus);
+ arr2->Item(0).SetIntegerCoordValue(aCalc->mLength);
+ arr2->Item(1).SetPercentValue(aCalc->mPercent);
+ }
+
+ SetArrayValue(arr, eCSSUnit_Calc);
+}
+
+void nsCSSValue::StartImageLoad(nsIDocument* aDocument) const
+{
+ MOZ_ASSERT(eCSSUnit_URL == mUnit, "Not a URL value!");
+ mozilla::css::ImageValue* image =
+ new mozilla::css::ImageValue(mValue.mURL->GetURI(),
+ mValue.mURL->mString,
+ mValue.mURL->mBaseURI,
+ mValue.mURL->mReferrer,
+ mValue.mURL->mOriginPrincipal,
+ aDocument);
+
+ nsCSSValue* writable = const_cast<nsCSSValue*>(this);
+ writable->SetImageValue(image);
+}
+
+nscolor nsCSSValue::GetColorValue() const
+{
+ MOZ_ASSERT(IsNumericColorUnit(), "not a color value");
+ if (IsFloatColorUnit()) {
+ return mValue.mFloatColor->GetColorValue(mUnit);
+ }
+ return mValue.mColor;
+}
+
+bool nsCSSValue::IsNonTransparentColor() const
+{
+ // We have the value in the form it was specified in at this point, so we
+ // have to look for both the keyword 'transparent' and its equivalent in
+ // rgba notation.
+ nsDependentString buf;
+ return
+ (IsIntegerColorUnit() && NS_GET_A(GetColorValue()) > 0) ||
+ (IsFloatColorUnit() && mValue.mFloatColor->IsNonTransparentColor()) ||
+ (mUnit == eCSSUnit_Ident &&
+ !nsGkAtoms::transparent->Equals(GetStringValue(buf))) ||
+ (mUnit == eCSSUnit_EnumColor);
+}
+
+nsCSSValue::Array*
+nsCSSValue::InitFunction(nsCSSKeyword aFunctionId, uint32_t aNumArgs)
+{
+ RefPtr<nsCSSValue::Array> func = Array::Create(aNumArgs + 1);
+ func->Item(0).SetIntValue(aFunctionId, eCSSUnit_Enumerated);
+ SetArrayValue(func, eCSSUnit_Function);
+ return func;
+}
+
+bool
+nsCSSValue::EqualsFunction(nsCSSKeyword aFunctionId) const
+{
+ if (mUnit != eCSSUnit_Function) {
+ return false;
+ }
+
+ nsCSSValue::Array* func = mValue.mArray;
+ MOZ_ASSERT(func && func->Count() >= 1 &&
+ func->Item(0).GetUnit() == eCSSUnit_Enumerated,
+ "illegally structured function value");
+
+ nsCSSKeyword thisFunctionId = func->Item(0).GetKeywordValue();
+ return thisFunctionId == aFunctionId;
+}
+
+// static
+already_AddRefed<nsStringBuffer>
+nsCSSValue::BufferFromString(const nsString& aValue)
+{
+ RefPtr<nsStringBuffer> buffer = nsStringBuffer::FromString(aValue);
+ if (buffer) {
+ return buffer.forget();
+ }
+
+ nsString::size_type length = aValue.Length();
+
+ // NOTE: Alloc prouduces a new, already-addref'd (refcnt = 1) buffer.
+ // NOTE: String buffer allocation is currently fallible.
+ size_t sz = (length + 1) * sizeof(char16_t);
+ buffer = nsStringBuffer::Alloc(sz);
+ if (MOZ_UNLIKELY(!buffer)) {
+ NS_ABORT_OOM(sz);
+ }
+
+ char16_t* data = static_cast<char16_t*>(buffer->Data());
+ nsCharTraits<char16_t>::copy(data, aValue.get(), length);
+ // Null-terminate.
+ data[length] = 0;
+ return buffer.forget();
+}
+
+namespace {
+
+struct CSSValueSerializeCalcOps {
+ CSSValueSerializeCalcOps(nsCSSPropertyID aProperty, nsAString& aResult,
+ nsCSSValue::Serialization aSerialization)
+ : mProperty(aProperty),
+ mResult(aResult),
+ mValueSerialization(aSerialization)
+ {
+ }
+
+ typedef nsCSSValue input_type;
+ typedef nsCSSValue::Array input_array_type;
+
+ static nsCSSUnit GetUnit(const input_type& aValue) {
+ return aValue.GetUnit();
+ }
+
+ void Append(const char* aString)
+ {
+ mResult.AppendASCII(aString);
+ }
+
+ void AppendLeafValue(const input_type& aValue)
+ {
+ MOZ_ASSERT(aValue.GetUnit() == eCSSUnit_Percent ||
+ aValue.IsLengthUnit() ||
+ aValue.GetUnit() == eCSSUnit_Number,
+ "unexpected unit");
+ aValue.AppendToString(mProperty, mResult, mValueSerialization);
+ }
+
+ void AppendNumber(const input_type& aValue)
+ {
+ MOZ_ASSERT(aValue.GetUnit() == eCSSUnit_Number, "unexpected unit");
+ aValue.AppendToString(mProperty, mResult, mValueSerialization);
+ }
+
+private:
+ nsCSSPropertyID mProperty;
+ nsAString &mResult;
+ nsCSSValue::Serialization mValueSerialization;
+};
+
+} // namespace
+
+void
+nsCSSValue::AppendPolygonToString(nsCSSPropertyID aProperty, nsAString& aResult,
+ Serialization aSerialization) const
+{
+ const nsCSSValue::Array* array = GetArrayValue();
+ MOZ_ASSERT(array->Count() > 1 && array->Count() <= 3,
+ "Polygons must have name and at least one more value.");
+ // When the array has 2 elements, the item on index 1 is the coordinate
+ // pair list.
+ // When the array has 3 elements, the item on index 1 is a fill-rule
+ // and item on index 2 is the coordinate pair list.
+ size_t index = 1;
+ if (array->Count() == 3) {
+ const nsCSSValue& fillRuleValue = array->Item(index);
+ MOZ_ASSERT(fillRuleValue.GetUnit() == eCSSUnit_Enumerated,
+ "Expected polygon fill rule.");
+ int32_t fillRule = fillRuleValue.GetIntValue();
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(fillRule,
+ nsCSSProps::kFillRuleKTable),
+ aResult);
+ aResult.AppendLiteral(", ");
+ ++index;
+ }
+ array->Item(index).AppendToString(aProperty, aResult, aSerialization);
+}
+
+inline void
+nsCSSValue::AppendPositionCoordinateToString(
+ const nsCSSValue& aValue, nsCSSPropertyID aProperty,
+ nsAString& aResult, Serialization aSerialization) const
+{
+ if (aValue.GetUnit() == eCSSUnit_Enumerated) {
+ int32_t intValue = aValue.GetIntValue();
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(intValue,
+ nsCSSProps::kShapeRadiusKTable), aResult);
+ } else {
+ aValue.AppendToString(aProperty, aResult, aSerialization);
+ }
+}
+
+void
+nsCSSValue::AppendCircleOrEllipseToString(nsCSSKeyword aFunctionId,
+ nsCSSPropertyID aProperty,
+ nsAString& aResult,
+ Serialization aSerialization) const
+{
+ const nsCSSValue::Array* array = GetArrayValue();
+ size_t count = aFunctionId == eCSSKeyword_circle ? 2 : 3;
+ MOZ_ASSERT(array->Count() == count + 1, "wrong number of arguments");
+
+ bool hasRadii = array->Item(1).GetUnit() != eCSSUnit_Null;
+
+ // closest-side is the default, so we don't need to
+ // output it if all values are closest-side.
+ if (array->Item(1).GetUnit() == eCSSUnit_Enumerated &&
+ array->Item(1).GetIntValue() == NS_RADIUS_CLOSEST_SIDE &&
+ (aFunctionId == eCSSKeyword_circle ||
+ (array->Item(2).GetUnit() == eCSSUnit_Enumerated &&
+ array->Item(2).GetIntValue() == NS_RADIUS_CLOSEST_SIDE))) {
+ hasRadii = false;
+ } else {
+ AppendPositionCoordinateToString(array->Item(1), aProperty,
+ aResult, aSerialization);
+
+ if (hasRadii && aFunctionId == eCSSKeyword_ellipse) {
+ aResult.Append(' ');
+ AppendPositionCoordinateToString(array->Item(2), aProperty,
+ aResult, aSerialization);
+ }
+ }
+
+ if (hasRadii) {
+ aResult.Append(' ');
+ }
+
+ // Any position specified?
+ if (array->Item(count).GetUnit() != eCSSUnit_Array) {
+ MOZ_ASSERT(array->Item(count).GetUnit() == eCSSUnit_Null,
+ "unexpected value");
+ // We only serialize to the 2 or 4 value form
+ // |circle()| is valid, but should be expanded
+ // to |circle(at 50% 50%)|
+ aResult.AppendLiteral("at 50% 50%");
+ return;
+ }
+
+ aResult.AppendLiteral("at ");
+ array->Item(count).AppendBasicShapePositionToString(aResult, aSerialization);
+}
+
+// https://drafts.csswg.org/css-shapes/#basic-shape-serialization
+// basic-shape asks us to omit a lot of redundant things whilst serializing
+// position values. Other specs are not clear about this
+// (https://github.com/w3c/csswg-drafts/issues/368), so for now we special-case
+// basic shapes only
+void
+nsCSSValue::AppendBasicShapePositionToString(nsAString& aResult,
+ Serialization aSerialization) const
+{
+ const nsCSSValue::Array* array = GetArrayValue();
+ // We always parse these into an array of four elements
+ MOZ_ASSERT(array->Count() == 4,
+ "basic-shape position value doesn't have enough elements");
+
+ const nsCSSValue &xEdge = array->Item(0);
+ const nsCSSValue &xOffset = array->Item(1);
+ const nsCSSValue &yEdge = array->Item(2);
+ const nsCSSValue &yOffset = array->Item(3);
+
+ MOZ_ASSERT(xEdge.GetUnit() == eCSSUnit_Enumerated &&
+ yEdge.GetUnit() == eCSSUnit_Enumerated &&
+ xOffset.IsLengthPercentCalcUnit() &&
+ yOffset.IsLengthPercentCalcUnit() &&
+ xEdge.GetIntValue() != NS_STYLE_IMAGELAYER_POSITION_CENTER &&
+ yEdge.GetIntValue() != NS_STYLE_IMAGELAYER_POSITION_CENTER,
+ "Ensure invariants from ParsePositionValueBasicShape "
+ "haven't been modified");
+ if (xEdge.GetIntValue() == NS_STYLE_IMAGELAYER_POSITION_LEFT &&
+ yEdge.GetIntValue() == NS_STYLE_IMAGELAYER_POSITION_TOP) {
+ // We can omit these defaults
+ xOffset.AppendToString(eCSSProperty_UNKNOWN, aResult, aSerialization);
+ aResult.Append(' ');
+ yOffset.AppendToString(eCSSProperty_UNKNOWN, aResult, aSerialization);
+ } else {
+ // We only serialize to the two or four valued form
+ xEdge.AppendToString(eCSSProperty_object_position, aResult, aSerialization);
+ aResult.Append(' ');
+ xOffset.AppendToString(eCSSProperty_UNKNOWN, aResult, aSerialization);
+ aResult.Append(' ');
+ yEdge.AppendToString(eCSSProperty_object_position, aResult, aSerialization);
+ aResult.Append(' ');
+ yOffset.AppendToString(eCSSProperty_UNKNOWN, aResult, aSerialization);
+ }
+}
+
+// Helper to append |aString| with the shorthand sides notation used in e.g.
+// 'padding'. |aProperties| and |aValues| are expected to have 4 elements.
+/*static*/ void
+nsCSSValue::AppendSidesShorthandToString(const nsCSSPropertyID aProperties[],
+ const nsCSSValue* aValues[],
+ nsAString& aString,
+ nsCSSValue::Serialization
+ aSerialization)
+{
+ const nsCSSValue& value1 = *aValues[0];
+ const nsCSSValue& value2 = *aValues[1];
+ const nsCSSValue& value3 = *aValues[2];
+ const nsCSSValue& value4 = *aValues[3];
+
+ MOZ_ASSERT(value1.GetUnit() != eCSSUnit_Null, "null value 1");
+ value1.AppendToString(aProperties[0], aString, aSerialization);
+ if (value1 != value2 || value1 != value3 || value1 != value4) {
+ aString.Append(char16_t(' '));
+ MOZ_ASSERT(value2.GetUnit() != eCSSUnit_Null, "null value 2");
+ value2.AppendToString(aProperties[1], aString, aSerialization);
+ if (value1 != value3 || value2 != value4) {
+ aString.Append(char16_t(' '));
+ MOZ_ASSERT(value3.GetUnit() != eCSSUnit_Null, "null value 3");
+ value3.AppendToString(aProperties[2], aString, aSerialization);
+ if (value2 != value4) {
+ aString.Append(char16_t(' '));
+ MOZ_ASSERT(value4.GetUnit() != eCSSUnit_Null, "null value 4");
+ value4.AppendToString(aProperties[3], aString, aSerialization);
+ }
+ }
+ }
+}
+
+/*static*/ void
+nsCSSValue::AppendBasicShapeRadiusToString(const nsCSSPropertyID aProperties[],
+ const nsCSSValue* aValues[],
+ nsAString& aResult,
+ Serialization aSerialization)
+{
+ bool needY = false;
+ const nsCSSValue* xVals[4];
+ const nsCSSValue* yVals[4];
+ for (int i = 0; i < 4; i++) {
+ if (aValues[i]->GetUnit() == eCSSUnit_Pair) {
+ needY = true;
+ xVals[i] = &aValues[i]->GetPairValue().mXValue;
+ yVals[i] = &aValues[i]->GetPairValue().mYValue;
+ } else {
+ xVals[i] = yVals[i] = aValues[i];
+ }
+ }
+
+ AppendSidesShorthandToString(aProperties, xVals, aResult, aSerialization);
+ if (needY) {
+ aResult.AppendLiteral(" / ");
+ AppendSidesShorthandToString(aProperties, yVals, aResult, aSerialization);
+ }
+}
+
+void
+nsCSSValue::AppendInsetToString(nsCSSPropertyID aProperty, nsAString& aResult,
+ Serialization aSerialization) const
+{
+ const nsCSSValue::Array* array = GetArrayValue();
+ MOZ_ASSERT(array->Count() == 6,
+ "inset function has wrong number of arguments");
+ if (array->Item(1).GetUnit() != eCSSUnit_Null) {
+ array->Item(1).AppendToString(aProperty, aResult, aSerialization);
+ if (array->Item(2).GetUnit() != eCSSUnit_Null) {
+ aResult.Append(' ');
+ array->Item(2).AppendToString(aProperty, aResult, aSerialization);
+ if (array->Item(3).GetUnit() != eCSSUnit_Null) {
+ aResult.Append(' ');
+ array->Item(3).AppendToString(aProperty, aResult, aSerialization);
+ if (array->Item(4).GetUnit() != eCSSUnit_Null) {
+ aResult.Append(' ');
+ array->Item(4).AppendToString(aProperty, aResult, aSerialization);
+ }
+ }
+ }
+ }
+
+ if (array->Item(5).GetUnit() == eCSSUnit_Array) {
+ const nsCSSPropertyID* subprops =
+ nsCSSProps::SubpropertyEntryFor(eCSSProperty_border_radius);
+ const nsCSSValue::Array* radius = array->Item(5).GetArrayValue();
+ MOZ_ASSERT(radius->Count() == 4, "expected 4 radii values");
+ const nsCSSValue* vals[4] = {
+ &(radius->Item(0)),
+ &(radius->Item(1)),
+ &(radius->Item(2)),
+ &(radius->Item(3))
+ };
+ aResult.AppendLiteral(" round ");
+ AppendBasicShapeRadiusToString(subprops, vals, aResult,
+ aSerialization);
+ } else {
+ MOZ_ASSERT(array->Item(5).GetUnit() == eCSSUnit_Null,
+ "unexpected value");
+ }
+}
+
+/* static */ void
+nsCSSValue::AppendAlignJustifyValueToString(int32_t aValue, nsAString& aResult)
+{
+ auto legacy = aValue & NS_STYLE_ALIGN_LEGACY;
+ if (legacy) {
+ aValue &= ~legacy;
+ aResult.AppendLiteral("legacy ");
+ }
+ auto overflowPos = aValue & (NS_STYLE_ALIGN_SAFE | NS_STYLE_ALIGN_UNSAFE);
+ aValue &= ~overflowPos;
+ MOZ_ASSERT(!(aValue & NS_STYLE_ALIGN_FLAG_BITS),
+ "unknown bits in align/justify value");
+ MOZ_ASSERT((aValue != NS_STYLE_ALIGN_AUTO &&
+ aValue != NS_STYLE_ALIGN_NORMAL &&
+ aValue != NS_STYLE_ALIGN_BASELINE &&
+ aValue != NS_STYLE_ALIGN_LAST_BASELINE) ||
+ (!legacy && !overflowPos),
+ "auto/normal/baseline/'last baseline' never have any flags");
+ MOZ_ASSERT(legacy == 0 || overflowPos == 0,
+ "'legacy' together with <overflow-position>");
+ if (aValue == NS_STYLE_ALIGN_LAST_BASELINE) {
+ aResult.AppendLiteral("last ");
+ aValue = NS_STYLE_ALIGN_BASELINE;
+ }
+ const auto& kwtable(nsCSSProps::kAlignAllKeywords);
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(aValue, kwtable), aResult);
+ // Don't serialize the 'unsafe' keyword; it's the default.
+ if (MOZ_UNLIKELY(overflowPos == NS_STYLE_ALIGN_SAFE)) {
+ aResult.Append(' ');
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(overflowPos, kwtable),
+ aResult);
+ }
+}
+
+void
+nsCSSValue::AppendToString(nsCSSPropertyID aProperty, nsAString& aResult,
+ Serialization aSerialization) const
+{
+ // eCSSProperty_UNKNOWN gets used for some recursive calls below.
+ MOZ_ASSERT((0 <= aProperty &&
+ aProperty <= eCSSProperty_COUNT_no_shorthands) ||
+ aProperty == eCSSProperty_UNKNOWN ||
+ aProperty == eCSSProperty_DOM,
+ "property ID out of range");
+
+ nsCSSUnit unit = GetUnit();
+ if (unit == eCSSUnit_Null) {
+ return;
+ }
+
+ if (eCSSUnit_String <= unit && unit <= eCSSUnit_Attr) {
+ if (unit == eCSSUnit_Attr) {
+ aResult.AppendLiteral("attr(");
+ }
+ nsAutoString buffer;
+ GetStringValue(buffer);
+ if (unit == eCSSUnit_String) {
+ nsStyleUtil::AppendEscapedCSSString(buffer, aResult);
+ } else {
+ nsStyleUtil::AppendEscapedCSSIdent(buffer, aResult);
+ }
+ }
+ else if (eCSSUnit_Array <= unit && unit <= eCSSUnit_Symbols) {
+ switch (unit) {
+ case eCSSUnit_Counter: aResult.AppendLiteral("counter("); break;
+ case eCSSUnit_Counters: aResult.AppendLiteral("counters("); break;
+ case eCSSUnit_Cubic_Bezier: aResult.AppendLiteral("cubic-bezier("); break;
+ case eCSSUnit_Steps: aResult.AppendLiteral("steps("); break;
+ case eCSSUnit_Symbols: aResult.AppendLiteral("symbols("); break;
+ default: break;
+ }
+
+ nsCSSValue::Array *array = GetArrayValue();
+ bool mark = false;
+ for (size_t i = 0, i_end = array->Count(); i < i_end; ++i) {
+ if (mark && array->Item(i).GetUnit() != eCSSUnit_Null) {
+ if ((unit == eCSSUnit_Array &&
+ eCSSProperty_transition_timing_function != aProperty) ||
+ unit == eCSSUnit_Symbols)
+ aResult.Append(' ');
+ else if (unit != eCSSUnit_Steps)
+ aResult.AppendLiteral(", ");
+ }
+ if (unit == eCSSUnit_Steps && i == 1) {
+ MOZ_ASSERT(array->Item(i).GetUnit() == eCSSUnit_Enumerated,
+ "unexpected value");
+ int32_t side = array->Item(i).GetIntValue();
+ MOZ_ASSERT(side == NS_STYLE_TRANSITION_TIMING_FUNCTION_STEP_START ||
+ side == NS_STYLE_TRANSITION_TIMING_FUNCTION_STEP_END ||
+ side == -1,
+ "unexpected value");
+ if (side == NS_STYLE_TRANSITION_TIMING_FUNCTION_STEP_START) {
+ aResult.AppendLiteral(", start");
+ } else if (side == NS_STYLE_TRANSITION_TIMING_FUNCTION_STEP_END) {
+ aResult.AppendLiteral(", end");
+ }
+ continue;
+ }
+ if (unit == eCSSUnit_Symbols && i == 0) {
+ MOZ_ASSERT(array->Item(i).GetUnit() == eCSSUnit_Enumerated,
+ "unexpected value");
+ int32_t system = array->Item(i).GetIntValue();
+ if (system != NS_STYLE_COUNTER_SYSTEM_SYMBOLIC) {
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(
+ system, nsCSSProps::kCounterSystemKTable), aResult);
+ mark = true;
+ }
+ continue;
+ }
+ nsCSSPropertyID prop =
+ ((eCSSUnit_Counter <= unit && unit <= eCSSUnit_Counters) &&
+ i == array->Count() - 1)
+ ? eCSSProperty_list_style_type : aProperty;
+ if (array->Item(i).GetUnit() != eCSSUnit_Null) {
+ array->Item(i).AppendToString(prop, aResult, aSerialization);
+ mark = true;
+ }
+ }
+ if (eCSSUnit_Array == unit &&
+ aProperty == eCSSProperty_transition_timing_function) {
+ aResult.Append(')');
+ }
+ }
+ /* Although Function is backed by an Array, we'll handle it separately
+ * because it's a bit quirky.
+ */
+ else if (eCSSUnit_Function == unit) {
+ const nsCSSValue::Array* array = GetArrayValue();
+ MOZ_ASSERT(array->Count() >= 1,
+ "Functions must have at least one element for the name.");
+
+ const nsCSSValue& functionName = array->Item(0);
+ MOZ_ASSERT(functionName.GetUnit() == eCSSUnit_Enumerated,
+ "Functions must have an enumerated name.");
+ // The first argument is always of nsCSSKeyword type.
+ const nsCSSKeyword functionId = functionName.GetKeywordValue();
+
+ // minmax(auto, <flex>) is equivalent to (and is our internal representation
+ // of) <flex>, and both are serialized as <flex>
+ if (functionId == eCSSKeyword_minmax &&
+ array->Count() == 3 &&
+ array->Item(1).GetUnit() == eCSSUnit_Auto &&
+ array->Item(2).GetUnit() == eCSSUnit_FlexFraction) {
+ array->Item(2).AppendToString(aProperty, aResult, aSerialization);
+ MOZ_ASSERT(aProperty == eCSSProperty_grid_template_columns ||
+ aProperty == eCSSProperty_grid_template_rows ||
+ aProperty == eCSSProperty_grid_auto_columns ||
+ aProperty == eCSSProperty_grid_auto_rows);
+ return;
+ }
+
+ /* Append the function name. */
+ NS_ConvertASCIItoUTF16 ident(nsCSSKeywords::GetStringValue(functionId));
+ // Bug 721136: Normalize the identifier to lowercase, except that things
+ // like scaleX should have the last character capitalized. This matches
+ // what other browsers do.
+ switch (functionId) {
+ case eCSSKeyword_rotatex:
+ case eCSSKeyword_scalex:
+ case eCSSKeyword_skewx:
+ case eCSSKeyword_translatex:
+ ident.Replace(ident.Length() - 1, 1, char16_t('X'));
+ break;
+
+ case eCSSKeyword_rotatey:
+ case eCSSKeyword_scaley:
+ case eCSSKeyword_skewy:
+ case eCSSKeyword_translatey:
+ ident.Replace(ident.Length() - 1, 1, char16_t('Y'));
+ break;
+
+ case eCSSKeyword_rotatez:
+ case eCSSKeyword_scalez:
+ case eCSSKeyword_translatez:
+ ident.Replace(ident.Length() - 1, 1, char16_t('Z'));
+ break;
+
+ default:
+ break;
+ }
+ nsStyleUtil::AppendEscapedCSSIdent(ident, aResult);
+ aResult.Append('(');
+
+ switch (functionId) {
+ case eCSSKeyword_polygon:
+ AppendPolygonToString(aProperty, aResult, aSerialization);
+ break;
+
+ case eCSSKeyword_circle:
+ case eCSSKeyword_ellipse:
+ AppendCircleOrEllipseToString(functionId, aProperty, aResult,
+ aSerialization);
+ break;
+
+ case eCSSKeyword_inset:
+ AppendInsetToString(aProperty, aResult, aSerialization);
+ break;
+
+ default: {
+ // Now, step through the function contents, writing each of
+ // them as we go.
+ for (size_t index = 1; index < array->Count(); ++index) {
+ array->Item(index).AppendToString(aProperty, aResult,
+ aSerialization);
+
+ /* If we're not at the final element, append a comma. */
+ if (index + 1 != array->Count())
+ aResult.AppendLiteral(", ");
+ }
+ }
+ }
+
+ /* Finally, append the closing parenthesis. */
+ aResult.Append(')');
+ }
+ else if (IsCalcUnit()) {
+ MOZ_ASSERT(GetUnit() == eCSSUnit_Calc, "unexpected unit");
+ CSSValueSerializeCalcOps ops(aProperty, aResult, aSerialization);
+ css::SerializeCalc(*this, ops);
+ }
+ else if (eCSSUnit_Integer == unit) {
+ aResult.AppendInt(GetIntValue(), 10);
+ }
+ else if (eCSSUnit_Enumerated == unit) {
+ int32_t intValue = GetIntValue();
+ switch(aProperty) {
+
+
+ case eCSSProperty_text_combine_upright:
+ if (intValue <= NS_STYLE_TEXT_COMBINE_UPRIGHT_ALL) {
+ AppendASCIItoUTF16(nsCSSProps::LookupPropertyValue(aProperty, intValue),
+ aResult);
+ } else if (intValue == NS_STYLE_TEXT_COMBINE_UPRIGHT_DIGITS_2) {
+ aResult.AppendLiteral("digits 2");
+ } else if (intValue == NS_STYLE_TEXT_COMBINE_UPRIGHT_DIGITS_3) {
+ aResult.AppendLiteral("digits 3");
+ } else {
+ aResult.AppendLiteral("digits 4");
+ }
+ break;
+
+ case eCSSProperty_text_decoration_line:
+ if (NS_STYLE_TEXT_DECORATION_LINE_NONE == intValue) {
+ AppendASCIItoUTF16(nsCSSProps::LookupPropertyValue(aProperty, intValue),
+ aResult);
+ } else {
+ // Ignore the "override all" internal value.
+ // (It doesn't have a string representation.)
+ intValue &= ~NS_STYLE_TEXT_DECORATION_LINE_OVERRIDE_ALL;
+ nsStyleUtil::AppendBitmaskCSSValue(
+ aProperty, intValue,
+ NS_STYLE_TEXT_DECORATION_LINE_UNDERLINE,
+ NS_STYLE_TEXT_DECORATION_LINE_PREF_ANCHORS,
+ aResult);
+ }
+ break;
+
+ case eCSSProperty_paint_order:
+ static_assert
+ (NS_STYLE_PAINT_ORDER_BITWIDTH * NS_STYLE_PAINT_ORDER_LAST_VALUE <= 8,
+ "SVGStyleStruct::mPaintOrder and the following cast not big enough");
+ nsStyleUtil::AppendPaintOrderValue(static_cast<uint8_t>(GetIntValue()),
+ aResult);
+ break;
+
+ case eCSSProperty_font_synthesis:
+ nsStyleUtil::AppendBitmaskCSSValue(aProperty, intValue,
+ NS_FONT_SYNTHESIS_WEIGHT,
+ NS_FONT_SYNTHESIS_STYLE,
+ aResult);
+ break;
+
+ case eCSSProperty_font_variant_east_asian:
+ nsStyleUtil::AppendBitmaskCSSValue(aProperty, intValue,
+ NS_FONT_VARIANT_EAST_ASIAN_JIS78,
+ NS_FONT_VARIANT_EAST_ASIAN_RUBY,
+ aResult);
+ break;
+
+ case eCSSProperty_font_variant_ligatures:
+ nsStyleUtil::AppendBitmaskCSSValue(aProperty, intValue,
+ NS_FONT_VARIANT_LIGATURES_NONE,
+ NS_FONT_VARIANT_LIGATURES_NO_CONTEXTUAL,
+ aResult);
+ break;
+
+ case eCSSProperty_font_variant_numeric:
+ nsStyleUtil::AppendBitmaskCSSValue(aProperty, intValue,
+ NS_FONT_VARIANT_NUMERIC_LINING,
+ NS_FONT_VARIANT_NUMERIC_ORDINAL,
+ aResult);
+ break;
+
+ case eCSSProperty_grid_auto_flow:
+ nsStyleUtil::AppendBitmaskCSSValue(aProperty, intValue,
+ NS_STYLE_GRID_AUTO_FLOW_ROW,
+ NS_STYLE_GRID_AUTO_FLOW_DENSE,
+ aResult);
+ break;
+
+ case eCSSProperty_grid_column_start:
+ case eCSSProperty_grid_column_end:
+ case eCSSProperty_grid_row_start:
+ case eCSSProperty_grid_row_end:
+ // "span" is the only enumerated-unit value for these properties
+ aResult.AppendLiteral("span");
+ break;
+
+ case eCSSProperty_touch_action:
+ nsStyleUtil::AppendBitmaskCSSValue(aProperty, intValue,
+ NS_STYLE_TOUCH_ACTION_NONE,
+ NS_STYLE_TOUCH_ACTION_MANIPULATION,
+ aResult);
+ break;
+
+ case eCSSProperty_clip_path:
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(intValue,
+ nsCSSProps::kClipPathGeometryBoxKTable),
+ aResult);
+ break;
+
+ case eCSSProperty_shape_outside:
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(intValue,
+ nsCSSProps::kShapeOutsideShapeBoxKTable),
+ aResult);
+ break;
+
+ case eCSSProperty_contain:
+ if (intValue & NS_STYLE_CONTAIN_STRICT) {
+ NS_ASSERTION(intValue == (NS_STYLE_CONTAIN_STRICT | NS_STYLE_CONTAIN_ALL_BITS),
+ "contain: strict should imply contain: layout style paint");
+ // Only output strict.
+ intValue = NS_STYLE_CONTAIN_STRICT;
+ }
+ nsStyleUtil::AppendBitmaskCSSValue(aProperty,
+ intValue,
+ NS_STYLE_CONTAIN_STRICT,
+ NS_STYLE_CONTAIN_PAINT,
+ aResult);
+ break;
+
+ case eCSSProperty_align_content:
+ case eCSSProperty_justify_content: {
+ AppendAlignJustifyValueToString(intValue & NS_STYLE_ALIGN_ALL_BITS, aResult);
+ auto fallback = intValue >> NS_STYLE_ALIGN_ALL_SHIFT;
+ if (fallback) {
+ MOZ_ASSERT(nsCSSProps::ValueToKeywordEnum(fallback & ~NS_STYLE_ALIGN_FLAG_BITS,
+ nsCSSProps::kAlignSelfPosition)
+ != eCSSKeyword_UNKNOWN, "unknown fallback value");
+ aResult.Append(' ');
+ AppendAlignJustifyValueToString(fallback, aResult);
+ }
+ break;
+ }
+
+ case eCSSProperty_align_items:
+ case eCSSProperty_align_self:
+ case eCSSProperty_justify_items:
+ case eCSSProperty_justify_self:
+ AppendAlignJustifyValueToString(intValue, aResult);
+ break;
+
+ case eCSSProperty_text_emphasis_position: {
+ nsStyleUtil::AppendBitmaskCSSValue(aProperty, intValue,
+ NS_STYLE_TEXT_EMPHASIS_POSITION_OVER,
+ NS_STYLE_TEXT_EMPHASIS_POSITION_RIGHT,
+ aResult);
+ break;
+ }
+
+ case eCSSProperty_text_emphasis_style: {
+ auto fill = intValue & NS_STYLE_TEXT_EMPHASIS_STYLE_FILL_MASK;
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(
+ fill, nsCSSProps::kTextEmphasisStyleFillKTable), aResult);
+ aResult.Append(' ');
+ auto shape = intValue & NS_STYLE_TEXT_EMPHASIS_STYLE_SHAPE_MASK;
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(
+ shape, nsCSSProps::kTextEmphasisStyleShapeKTable), aResult);
+ break;
+ }
+
+ default:
+ const nsAFlatCString& name = nsCSSProps::LookupPropertyValue(aProperty, intValue);
+ AppendASCIItoUTF16(name, aResult);
+ break;
+ }
+ }
+ else if (eCSSUnit_EnumColor == unit) {
+ // we can lookup the property in the ColorTable and then
+ // get a string mapping the name
+ nsAutoCString str;
+ if (nsCSSProps::GetColorName(GetIntValue(), str)){
+ AppendASCIItoUTF16(str, aResult);
+ } else {
+ MOZ_ASSERT(false, "bad color value");
+ }
+ }
+ else if (IsNumericColorUnit(unit)) {
+ if (aSerialization == eNormalized ||
+ unit == eCSSUnit_RGBColor ||
+ unit == eCSSUnit_RGBAColor) {
+ nscolor color = GetColorValue();
+ if (aSerialization == eNormalized &&
+ color == NS_RGBA(0, 0, 0, 0)) {
+ // Use the strictest match for 'transparent' so we do correct
+ // round-tripping of all other rgba() values.
+ aResult.AppendLiteral("transparent");
+ } else {
+ // For brevity, we omit the alpha component if it's equal to 255 (full
+ // opaque). Also, we use "rgba" rather than "rgb" when the color includes
+ // the non-opaque alpha value, for backwards-compat (even though they're
+ // aliases as of css-color-4).
+ // e.g.:
+ // rgba(1, 2, 3, 1.0) => rgb(1, 2, 3)
+ // rgba(1, 2, 3, 0.5) => rgba(1, 2, 3, 0.5)
+ uint8_t a = NS_GET_A(color);
+ bool showAlpha = (a != 255);
+
+ if (showAlpha) {
+ aResult.AppendLiteral("rgba(");
+ } else {
+ aResult.AppendLiteral("rgb(");
+ }
+
+ NS_NAMED_LITERAL_STRING(comma, ", ");
+
+ aResult.AppendInt(NS_GET_R(color), 10);
+ aResult.Append(comma);
+ aResult.AppendInt(NS_GET_G(color), 10);
+ aResult.Append(comma);
+ aResult.AppendInt(NS_GET_B(color), 10);
+ if (showAlpha) {
+ aResult.Append(comma);
+ aResult.AppendFloat(nsStyleUtil::ColorComponentToFloat(a));
+ }
+ aResult.Append(char16_t(')'));
+ }
+ } else if (eCSSUnit_HexColor == unit ||
+ eCSSUnit_HexColorAlpha == unit) {
+ nscolor color = GetColorValue();
+ aResult.Append('#');
+ aResult.AppendPrintf("%02x", NS_GET_R(color));
+ aResult.AppendPrintf("%02x", NS_GET_G(color));
+ aResult.AppendPrintf("%02x", NS_GET_B(color));
+ if (eCSSUnit_HexColorAlpha == unit) {
+ aResult.AppendPrintf("%02x", NS_GET_A(color));
+ }
+ } else if (eCSSUnit_ShortHexColor == unit ||
+ eCSSUnit_ShortHexColorAlpha == unit) {
+ nscolor color = GetColorValue();
+ aResult.Append('#');
+ aResult.AppendInt(NS_GET_R(color) / 0x11, 16);
+ aResult.AppendInt(NS_GET_G(color) / 0x11, 16);
+ aResult.AppendInt(NS_GET_B(color) / 0x11, 16);
+ if (eCSSUnit_ShortHexColorAlpha == unit) {
+ aResult.AppendInt(NS_GET_A(color) / 0x11, 16);
+ }
+ } else {
+ MOZ_ASSERT(IsFloatColorUnit());
+ mValue.mFloatColor->AppendToString(unit, aResult);
+ }
+ }
+ else if (eCSSUnit_ComplexColor == unit) {
+ StyleComplexColor color = GetStyleComplexColorValue();
+ nsCSSValue serializable;
+ if (color.IsCurrentColor()) {
+ serializable.SetIntValue(NS_COLOR_CURRENTCOLOR, eCSSUnit_EnumColor);
+ } else if (color.IsNumericColor()) {
+ serializable.SetColorValue(color.mColor);
+ } else {
+ MOZ_ASSERT_UNREACHABLE("Cannot serialize a complex color");
+ }
+ serializable.AppendToString(aProperty, aResult, aSerialization);
+ }
+ else if (eCSSUnit_URL == unit || eCSSUnit_Image == unit) {
+ aResult.AppendLiteral("url(");
+ nsStyleUtil::AppendEscapedCSSString(
+ nsDependentString(GetOriginalURLValue()), aResult);
+ aResult.Append(')');
+ }
+ else if (eCSSUnit_Element == unit) {
+ aResult.AppendLiteral("-moz-element(#");
+ nsAutoString tmpStr;
+ GetStringValue(tmpStr);
+ nsStyleUtil::AppendEscapedCSSIdent(tmpStr, aResult);
+ aResult.Append(')');
+ }
+ else if (eCSSUnit_Percent == unit) {
+ aResult.AppendFloat(GetPercentValue() * 100.0f);
+ }
+ else if (eCSSUnit_Percent < unit) { // length unit
+ aResult.AppendFloat(GetFloatValue());
+ }
+ else if (eCSSUnit_Gradient == unit) {
+ nsCSSValueGradient* gradient = GetGradientValue();
+
+ if (gradient->mIsLegacySyntax) {
+ aResult.AppendLiteral("-moz-");
+ }
+ if (gradient->mIsRepeating) {
+ aResult.AppendLiteral("repeating-");
+ }
+ if (gradient->mIsRadial) {
+ aResult.AppendLiteral("radial-gradient(");
+ } else {
+ aResult.AppendLiteral("linear-gradient(");
+ }
+
+ bool needSep = false;
+ if (gradient->mIsRadial && !gradient->mIsLegacySyntax) {
+ if (!gradient->mIsExplicitSize) {
+ if (gradient->GetRadialShape().GetUnit() != eCSSUnit_None) {
+ MOZ_ASSERT(gradient->GetRadialShape().GetUnit() ==
+ eCSSUnit_Enumerated,
+ "bad unit for radial gradient shape");
+ int32_t intValue = gradient->GetRadialShape().GetIntValue();
+ MOZ_ASSERT(intValue != NS_STYLE_GRADIENT_SHAPE_LINEAR,
+ "radial gradient with linear shape?!");
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(intValue,
+ nsCSSProps::kRadialGradientShapeKTable),
+ aResult);
+ needSep = true;
+ }
+
+ if (gradient->GetRadialSize().GetUnit() != eCSSUnit_None) {
+ if (needSep) {
+ aResult.Append(' ');
+ }
+ MOZ_ASSERT(gradient->GetRadialSize().GetUnit() == eCSSUnit_Enumerated,
+ "bad unit for radial gradient size");
+ int32_t intValue = gradient->GetRadialSize().GetIntValue();
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(intValue,
+ nsCSSProps::kRadialGradientSizeKTable),
+ aResult);
+ needSep = true;
+ }
+ } else {
+ MOZ_ASSERT(gradient->GetRadiusX().GetUnit() != eCSSUnit_None,
+ "bad unit for radial gradient explicit size");
+ gradient->GetRadiusX().AppendToString(aProperty, aResult,
+ aSerialization);
+ if (gradient->GetRadiusY().GetUnit() != eCSSUnit_None) {
+ aResult.Append(' ');
+ gradient->GetRadiusY().AppendToString(aProperty, aResult,
+ aSerialization);
+ }
+ needSep = true;
+ }
+ }
+ if (!gradient->mIsRadial && !gradient->mIsLegacySyntax) {
+ if (gradient->mBgPos.mXValue.GetUnit() != eCSSUnit_None ||
+ gradient->mBgPos.mYValue.GetUnit() != eCSSUnit_None) {
+ MOZ_ASSERT(gradient->mAngle.GetUnit() == eCSSUnit_None);
+ MOZ_ASSERT(gradient->mBgPos.mXValue.GetUnit() == eCSSUnit_Enumerated &&
+ gradient->mBgPos.mYValue.GetUnit() == eCSSUnit_Enumerated,
+ "unexpected unit");
+ aResult.AppendLiteral("to");
+ if (!(gradient->mBgPos.mXValue.GetIntValue() & NS_STYLE_IMAGELAYER_POSITION_CENTER)) {
+ aResult.Append(' ');
+ gradient->mBgPos.mXValue.AppendToString(eCSSProperty_background_position_x,
+ aResult, aSerialization);
+ }
+ if (!(gradient->mBgPos.mYValue.GetIntValue() & NS_STYLE_IMAGELAYER_POSITION_CENTER)) {
+ aResult.Append(' ');
+ gradient->mBgPos.mYValue.AppendToString(eCSSProperty_background_position_y,
+ aResult, aSerialization);
+ }
+ needSep = true;
+ } else if (gradient->mAngle.GetUnit() != eCSSUnit_None) {
+ gradient->mAngle.AppendToString(aProperty, aResult, aSerialization);
+ needSep = true;
+ }
+ } else if (gradient->mBgPos.mXValue.GetUnit() != eCSSUnit_None ||
+ gradient->mBgPos.mYValue.GetUnit() != eCSSUnit_None ||
+ gradient->mAngle.GetUnit() != eCSSUnit_None) {
+ if (needSep) {
+ aResult.Append(' ');
+ }
+ if (gradient->mIsRadial && !gradient->mIsLegacySyntax) {
+ aResult.AppendLiteral("at ");
+ }
+ if (gradient->mBgPos.mXValue.GetUnit() != eCSSUnit_None) {
+ gradient->mBgPos.mXValue.AppendToString(eCSSProperty_background_position_x,
+ aResult, aSerialization);
+ aResult.Append(' ');
+ }
+ if (gradient->mBgPos.mYValue.GetUnit() != eCSSUnit_None) {
+ gradient->mBgPos.mYValue.AppendToString(eCSSProperty_background_position_y,
+ aResult, aSerialization);
+ aResult.Append(' ');
+ }
+ if (gradient->mAngle.GetUnit() != eCSSUnit_None) {
+ MOZ_ASSERT(gradient->mIsLegacySyntax,
+ "angle is allowed only for legacy syntax");
+ gradient->mAngle.AppendToString(aProperty, aResult, aSerialization);
+ }
+ needSep = true;
+ }
+
+ if (gradient->mIsRadial && gradient->mIsLegacySyntax &&
+ (gradient->GetRadialShape().GetUnit() != eCSSUnit_None ||
+ gradient->GetRadialSize().GetUnit() != eCSSUnit_None)) {
+ MOZ_ASSERT(!gradient->mIsExplicitSize);
+ if (needSep) {
+ aResult.AppendLiteral(", ");
+ }
+ if (gradient->GetRadialShape().GetUnit() != eCSSUnit_None) {
+ MOZ_ASSERT(gradient->GetRadialShape().GetUnit() == eCSSUnit_Enumerated,
+ "bad unit for radial gradient shape");
+ int32_t intValue = gradient->GetRadialShape().GetIntValue();
+ MOZ_ASSERT(intValue != NS_STYLE_GRADIENT_SHAPE_LINEAR,
+ "radial gradient with linear shape?!");
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(intValue,
+ nsCSSProps::kRadialGradientShapeKTable),
+ aResult);
+ aResult.Append(' ');
+ }
+
+ if (gradient->GetRadialSize().GetUnit() != eCSSUnit_None) {
+ MOZ_ASSERT(gradient->GetRadialSize().GetUnit() == eCSSUnit_Enumerated,
+ "bad unit for radial gradient size");
+ int32_t intValue = gradient->GetRadialSize().GetIntValue();
+ AppendASCIItoUTF16(nsCSSProps::ValueToKeyword(intValue,
+ nsCSSProps::kRadialGradientSizeKTable),
+ aResult);
+ }
+ needSep = true;
+ }
+ if (needSep) {
+ aResult.AppendLiteral(", ");
+ }
+
+ for (uint32_t i = 0 ;;) {
+ bool isInterpolationHint = gradient->mStops[i].mIsInterpolationHint;
+ if (!isInterpolationHint) {
+ gradient->mStops[i].mColor.AppendToString(aProperty, aResult,
+ aSerialization);
+ }
+ if (gradient->mStops[i].mLocation.GetUnit() != eCSSUnit_None) {
+ if (!isInterpolationHint) {
+ aResult.Append(' ');
+ }
+ gradient->mStops[i].mLocation.AppendToString(aProperty, aResult,
+ aSerialization);
+ }
+ if (++i == gradient->mStops.Length()) {
+ break;
+ }
+ aResult.AppendLiteral(", ");
+ }
+
+ aResult.Append(')');
+ } else if (eCSSUnit_TokenStream == unit) {
+ nsCSSPropertyID shorthand = mValue.mTokenStream->mShorthandPropertyID;
+ if (shorthand == eCSSProperty_UNKNOWN ||
+ nsCSSProps::PropHasFlags(shorthand, CSS_PROPERTY_IS_ALIAS) ||
+ aProperty == eCSSProperty__x_system_font) {
+ // We treat serialization of aliases like '-moz-transform' as a special
+ // case, since it really wants to be serialized as if it were a longhand
+ // even though it is implemented as a shorthand. We also need to
+ // serialize -x-system-font's token stream value, even though the
+ // value is set through the font shorthand. This serialization
+ // of -x-system-font is needed when we need to output the
+ // 'font' shorthand followed by a number of overriding font
+ // longhand components.
+ aResult.Append(mValue.mTokenStream->mTokenStream);
+ }
+ } else if (eCSSUnit_Pair == unit) {
+ if (eCSSProperty_font_variant_alternates == aProperty) {
+ int32_t intValue = GetPairValue().mXValue.GetIntValue();
+ nsAutoString out;
+
+ // simple, enumerated values
+ nsStyleUtil::AppendBitmaskCSSValue(aProperty,
+ intValue & NS_FONT_VARIANT_ALTERNATES_ENUMERATED_MASK,
+ NS_FONT_VARIANT_ALTERNATES_HISTORICAL,
+ NS_FONT_VARIANT_ALTERNATES_HISTORICAL,
+ out);
+
+ // functional values
+ const nsCSSValueList *list = GetPairValue().mYValue.GetListValue();
+ AutoTArray<gfxAlternateValue,8> altValues;
+
+ nsStyleUtil::ComputeFunctionalAlternates(list, altValues);
+ nsStyleUtil::SerializeFunctionalAlternates(altValues, out);
+ aResult.Append(out);
+ } else {
+ GetPairValue().AppendToString(aProperty, aResult, aSerialization);
+ }
+ } else if (eCSSUnit_Triplet == unit) {
+ GetTripletValue().AppendToString(aProperty, aResult, aSerialization);
+ } else if (eCSSUnit_Rect == unit) {
+ GetRectValue().AppendToString(aProperty, aResult, aSerialization);
+ } else if (eCSSUnit_List == unit || eCSSUnit_ListDep == unit) {
+ GetListValue()->AppendToString(aProperty, aResult, aSerialization);
+ } else if (eCSSUnit_SharedList == unit) {
+ GetSharedListValue()->AppendToString(aProperty, aResult, aSerialization);
+ } else if (eCSSUnit_PairList == unit || eCSSUnit_PairListDep == unit) {
+ switch (aProperty) {
+ case eCSSProperty_font_feature_settings:
+ nsStyleUtil::AppendFontFeatureSettings(*this, aResult);
+ break;
+ default:
+ GetPairListValue()->AppendToString(aProperty, aResult, aSerialization);
+ break;
+ }
+ } else if (eCSSUnit_GridTemplateAreas == unit) {
+ const mozilla::css::GridTemplateAreasValue* areas = GetGridTemplateAreas();
+ MOZ_ASSERT(!areas->mTemplates.IsEmpty(),
+ "Unexpected empty array in GridTemplateAreasValue");
+ nsStyleUtil::AppendEscapedCSSString(areas->mTemplates[0], aResult);
+ for (uint32_t i = 1; i < areas->mTemplates.Length(); i++) {
+ aResult.Append(char16_t(' '));
+ nsStyleUtil::AppendEscapedCSSString(areas->mTemplates[i], aResult);
+ }
+ } else if (eCSSUnit_FontFamilyList == unit) {
+ nsStyleUtil::AppendEscapedCSSFontFamilyList(*mValue.mFontFamilyList,
+ aResult);
+ }
+
+ switch (unit) {
+ case eCSSUnit_Null: break;
+ case eCSSUnit_Auto: aResult.AppendLiteral("auto"); break;
+ case eCSSUnit_Inherit: aResult.AppendLiteral("inherit"); break;
+ case eCSSUnit_Initial: aResult.AppendLiteral("initial"); break;
+ case eCSSUnit_Unset: aResult.AppendLiteral("unset"); break;
+ case eCSSUnit_None: aResult.AppendLiteral("none"); break;
+ case eCSSUnit_Normal: aResult.AppendLiteral("normal"); break;
+ case eCSSUnit_System_Font: aResult.AppendLiteral("-moz-use-system-font"); break;
+ case eCSSUnit_All: aResult.AppendLiteral("all"); break;
+ case eCSSUnit_Dummy:
+ case eCSSUnit_DummyInherit:
+ MOZ_ASSERT(false, "should never serialize");
+ break;
+
+ case eCSSUnit_FontFamilyList: break;
+ case eCSSUnit_String: break;
+ case eCSSUnit_Ident: break;
+ case eCSSUnit_URL: break;
+ case eCSSUnit_Image: break;
+ case eCSSUnit_Element: break;
+ case eCSSUnit_Array: break;
+ case eCSSUnit_Attr:
+ case eCSSUnit_Cubic_Bezier:
+ case eCSSUnit_Steps:
+ case eCSSUnit_Symbols:
+ case eCSSUnit_Counter:
+ case eCSSUnit_Counters: aResult.Append(char16_t(')')); break;
+ case eCSSUnit_Local_Font: break;
+ case eCSSUnit_Font_Format: break;
+ case eCSSUnit_Function: break;
+ case eCSSUnit_Calc: break;
+ case eCSSUnit_Calc_Plus: break;
+ case eCSSUnit_Calc_Minus: break;
+ case eCSSUnit_Calc_Times_L: break;
+ case eCSSUnit_Calc_Times_R: break;
+ case eCSSUnit_Calc_Divided: break;
+ case eCSSUnit_Integer: break;
+ case eCSSUnit_Enumerated: break;
+ case eCSSUnit_EnumColor: break;
+ case eCSSUnit_RGBColor: break;
+ case eCSSUnit_RGBAColor: break;
+ case eCSSUnit_HexColor: break;
+ case eCSSUnit_ShortHexColor: break;
+ case eCSSUnit_HexColorAlpha: break;
+ case eCSSUnit_ShortHexColorAlpha: break;
+ case eCSSUnit_PercentageRGBColor: break;
+ case eCSSUnit_PercentageRGBAColor: break;
+ case eCSSUnit_HSLColor: break;
+ case eCSSUnit_HSLAColor: break;
+ case eCSSUnit_ComplexColor: break;
+ case eCSSUnit_Percent: aResult.Append(char16_t('%')); break;
+ case eCSSUnit_Number: break;
+ case eCSSUnit_Gradient: break;
+ case eCSSUnit_TokenStream: break;
+ case eCSSUnit_Pair: break;
+ case eCSSUnit_Triplet: break;
+ case eCSSUnit_Rect: break;
+ case eCSSUnit_List: break;
+ case eCSSUnit_ListDep: break;
+ case eCSSUnit_SharedList: break;
+ case eCSSUnit_PairList: break;
+ case eCSSUnit_PairListDep: break;
+ case eCSSUnit_GridTemplateAreas: break;
+
+ case eCSSUnit_Inch: aResult.AppendLiteral("in"); break;
+ case eCSSUnit_Millimeter: aResult.AppendLiteral("mm"); break;
+ case eCSSUnit_PhysicalMillimeter: aResult.AppendLiteral("mozmm"); break;
+ case eCSSUnit_Centimeter: aResult.AppendLiteral("cm"); break;
+ case eCSSUnit_Point: aResult.AppendLiteral("pt"); break;
+ case eCSSUnit_Pica: aResult.AppendLiteral("pc"); break;
+ case eCSSUnit_Quarter: aResult.AppendLiteral("q"); break;
+
+ case eCSSUnit_ViewportWidth: aResult.AppendLiteral("vw"); break;
+ case eCSSUnit_ViewportHeight: aResult.AppendLiteral("vh"); break;
+ case eCSSUnit_ViewportMin: aResult.AppendLiteral("vmin"); break;
+ case eCSSUnit_ViewportMax: aResult.AppendLiteral("vmax"); break;
+
+ case eCSSUnit_EM: aResult.AppendLiteral("em"); break;
+ case eCSSUnit_XHeight: aResult.AppendLiteral("ex"); break;
+ case eCSSUnit_Char: aResult.AppendLiteral("ch"); break;
+ case eCSSUnit_RootEM: aResult.AppendLiteral("rem"); break;
+
+ case eCSSUnit_Pixel: aResult.AppendLiteral("px"); break;
+
+ case eCSSUnit_Degree: aResult.AppendLiteral("deg"); break;
+ case eCSSUnit_Grad: aResult.AppendLiteral("grad"); break;
+ case eCSSUnit_Radian: aResult.AppendLiteral("rad"); break;
+ case eCSSUnit_Turn: aResult.AppendLiteral("turn"); break;
+
+ case eCSSUnit_Hertz: aResult.AppendLiteral("Hz"); break;
+ case eCSSUnit_Kilohertz: aResult.AppendLiteral("kHz"); break;
+
+ case eCSSUnit_Seconds: aResult.Append(char16_t('s')); break;
+ case eCSSUnit_Milliseconds: aResult.AppendLiteral("ms"); break;
+
+ case eCSSUnit_FlexFraction: aResult.AppendLiteral("fr"); break;
+ }
+}
+
+size_t
+nsCSSValue::SizeOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ size_t n = 0;
+
+ switch (GetUnit()) {
+ // No value: nothing extra to measure.
+ case eCSSUnit_Null:
+ case eCSSUnit_Auto:
+ case eCSSUnit_Inherit:
+ case eCSSUnit_Initial:
+ case eCSSUnit_Unset:
+ case eCSSUnit_None:
+ case eCSSUnit_Normal:
+ case eCSSUnit_System_Font:
+ case eCSSUnit_All:
+ case eCSSUnit_Dummy:
+ case eCSSUnit_DummyInherit:
+ break;
+
+ // String
+ case eCSSUnit_String:
+ case eCSSUnit_Ident:
+ case eCSSUnit_Attr:
+ case eCSSUnit_Local_Font:
+ case eCSSUnit_Font_Format:
+ case eCSSUnit_Element:
+ n += mValue.mString->SizeOfIncludingThisIfUnshared(aMallocSizeOf);
+ break;
+
+ // Array
+ case eCSSUnit_Array:
+ case eCSSUnit_Counter:
+ case eCSSUnit_Counters:
+ case eCSSUnit_Cubic_Bezier:
+ case eCSSUnit_Steps:
+ case eCSSUnit_Symbols:
+ case eCSSUnit_Function:
+ case eCSSUnit_Calc:
+ case eCSSUnit_Calc_Plus:
+ case eCSSUnit_Calc_Minus:
+ case eCSSUnit_Calc_Times_L:
+ case eCSSUnit_Calc_Times_R:
+ case eCSSUnit_Calc_Divided:
+ break;
+
+ // URL
+ case eCSSUnit_URL:
+ n += mValue.mURL->SizeOfIncludingThis(aMallocSizeOf);
+ break;
+
+ // Image
+ case eCSSUnit_Image:
+ // Not yet measured. Measurement may be added later if DMD finds it
+ // worthwhile.
+ break;
+
+ // Gradient
+ case eCSSUnit_Gradient:
+ n += mValue.mGradient->SizeOfIncludingThis(aMallocSizeOf);
+ break;
+
+ // TokenStream
+ case eCSSUnit_TokenStream:
+ n += mValue.mTokenStream->SizeOfIncludingThis(aMallocSizeOf);
+ break;
+
+ // Pair
+ case eCSSUnit_Pair:
+ n += mValue.mPair->SizeOfIncludingThis(aMallocSizeOf);
+ break;
+
+ // Triplet
+ case eCSSUnit_Triplet:
+ n += mValue.mTriplet->SizeOfIncludingThis(aMallocSizeOf);
+ break;
+
+ // Rect
+ case eCSSUnit_Rect:
+ n += mValue.mRect->SizeOfIncludingThis(aMallocSizeOf);
+ break;
+
+ // List
+ case eCSSUnit_List:
+ n += mValue.mList->SizeOfIncludingThis(aMallocSizeOf);
+ break;
+
+ // ListDep: not measured because it's non-owning.
+ case eCSSUnit_ListDep:
+ break;
+
+ // SharedList
+ case eCSSUnit_SharedList:
+ // Makes more sense not to measure, since it most cases the list
+ // will be shared.
+ break;
+
+ // PairList
+ case eCSSUnit_PairList:
+ n += mValue.mPairList->SizeOfIncludingThis(aMallocSizeOf);
+ break;
+
+ // PairListDep: not measured because it's non-owning.
+ case eCSSUnit_PairListDep:
+ break;
+
+ // GridTemplateAreas
+ case eCSSUnit_GridTemplateAreas:
+ n += mValue.mGridTemplateAreas->SizeOfIncludingThis(aMallocSizeOf);
+ break;
+
+ case eCSSUnit_FontFamilyList:
+ n += mValue.mFontFamilyList->SizeOfIncludingThis(aMallocSizeOf);
+ break;
+
+ // Int: nothing extra to measure.
+ case eCSSUnit_Integer:
+ case eCSSUnit_Enumerated:
+ case eCSSUnit_EnumColor:
+ break;
+
+ // Integer Color: nothing extra to measure.
+ case eCSSUnit_RGBColor:
+ case eCSSUnit_RGBAColor:
+ case eCSSUnit_HexColor:
+ case eCSSUnit_ShortHexColor:
+ case eCSSUnit_HexColorAlpha:
+ case eCSSUnit_ShortHexColorAlpha:
+ break;
+
+ // Float Color
+ case eCSSUnit_PercentageRGBColor:
+ case eCSSUnit_PercentageRGBAColor:
+ case eCSSUnit_HSLColor:
+ case eCSSUnit_HSLAColor:
+ n += mValue.mFloatColor->SizeOfIncludingThis(aMallocSizeOf);
+ break;
+
+ // Complex Color
+ case eCSSUnit_ComplexColor:
+ n += mValue.mComplexColor->SizeOfIncludingThis(aMallocSizeOf);
+ break;
+
+ // Float: nothing extra to measure.
+ case eCSSUnit_Percent:
+ case eCSSUnit_Number:
+ case eCSSUnit_PhysicalMillimeter:
+ case eCSSUnit_ViewportWidth:
+ case eCSSUnit_ViewportHeight:
+ case eCSSUnit_ViewportMin:
+ case eCSSUnit_ViewportMax:
+ case eCSSUnit_EM:
+ case eCSSUnit_XHeight:
+ case eCSSUnit_Char:
+ case eCSSUnit_RootEM:
+ case eCSSUnit_Point:
+ case eCSSUnit_Inch:
+ case eCSSUnit_Millimeter:
+ case eCSSUnit_Centimeter:
+ case eCSSUnit_Pica:
+ case eCSSUnit_Pixel:
+ case eCSSUnit_Quarter:
+ case eCSSUnit_Degree:
+ case eCSSUnit_Grad:
+ case eCSSUnit_Turn:
+ case eCSSUnit_Radian:
+ case eCSSUnit_Hertz:
+ case eCSSUnit_Kilohertz:
+ case eCSSUnit_Seconds:
+ case eCSSUnit_Milliseconds:
+ case eCSSUnit_FlexFraction:
+ break;
+
+ default:
+ MOZ_ASSERT(false, "bad nsCSSUnit");
+ break;
+ }
+
+ return n;
+}
+
+// --- nsCSSValueList -----------------
+
+nsCSSValueList::~nsCSSValueList()
+{
+ MOZ_COUNT_DTOR(nsCSSValueList);
+ NS_CSS_DELETE_LIST_MEMBER(nsCSSValueList, this, mNext);
+}
+
+nsCSSValueList*
+nsCSSValueList::Clone() const
+{
+ nsCSSValueList* result = new nsCSSValueList(*this);
+ nsCSSValueList* dest = result;
+ const nsCSSValueList* src = this->mNext;
+ while (src) {
+ dest->mNext = new nsCSSValueList(*src);
+ dest = dest->mNext;
+ src = src->mNext;
+ }
+
+ MOZ_ASSERT(result, "shouldn't return null; supposed to be infallible");
+ return result;
+}
+
+void
+nsCSSValueList::CloneInto(nsCSSValueList* aList) const
+{
+ NS_ASSERTION(!aList->mNext, "Must be an empty list!");
+ aList->mValue = mValue;
+ aList->mNext = mNext ? mNext->Clone() : nullptr;
+}
+
+static void
+AppendValueListToString(const nsCSSValueList* val,
+ nsCSSPropertyID aProperty, nsAString& aResult,
+ nsCSSValue::Serialization aSerialization)
+{
+ for (;;) {
+ val->mValue.AppendToString(aProperty, aResult, aSerialization);
+ val = val->mNext;
+ if (!val)
+ break;
+
+ if (nsCSSProps::PropHasFlags(aProperty,
+ CSS_PROPERTY_VALUE_LIST_USES_COMMAS))
+ aResult.Append(char16_t(','));
+ aResult.Append(char16_t(' '));
+ }
+}
+
+static void
+AppendGridTemplateToString(const nsCSSValueList* val,
+ nsCSSPropertyID aProperty, nsAString& aResult,
+ nsCSSValue::Serialization aSerialization)
+{
+ // This is called for the "list" that's the top-level value of the property.
+ bool isSubgrid = false;
+ for (;;) {
+ bool addSpaceSeparator = true;
+ nsCSSUnit unit = val->mValue.GetUnit();
+
+ if (unit == eCSSUnit_Enumerated &&
+ val->mValue.GetIntValue() == NS_STYLE_GRID_TEMPLATE_SUBGRID) {
+ MOZ_ASSERT(!isSubgrid, "saw subgrid once already");
+ isSubgrid = true;
+ aResult.AppendLiteral("subgrid");
+
+ } else if (unit == eCSSUnit_Pair) {
+ // This is a repeat 'auto-fill' / 'auto-fit'.
+ const nsCSSValuePair& pair = val->mValue.GetPairValue();
+ switch (pair.mXValue.GetIntValue()) {
+ case NS_STYLE_GRID_REPEAT_AUTO_FILL:
+ aResult.AppendLiteral("repeat(auto-fill, ");
+ break;
+ case NS_STYLE_GRID_REPEAT_AUTO_FIT:
+ aResult.AppendLiteral("repeat(auto-fit, ");
+ break;
+ default:
+ MOZ_ASSERT_UNREACHABLE("unexpected enum value");
+ }
+ const nsCSSValueList* repeatList = pair.mYValue.GetListValue();
+ if (repeatList->mValue.GetUnit() != eCSSUnit_Null) {
+ aResult.Append('[');
+ AppendValueListToString(repeatList->mValue.GetListValue(), aProperty,
+ aResult, aSerialization);
+ aResult.Append(']');
+ if (!isSubgrid) {
+ aResult.Append(' ');
+ }
+ } else if (isSubgrid) {
+ aResult.AppendLiteral("[]");
+ }
+ if (!isSubgrid) {
+ repeatList = repeatList->mNext;
+ repeatList->mValue.AppendToString(aProperty, aResult, aSerialization);
+ repeatList = repeatList->mNext;
+ if (repeatList->mValue.GetUnit() != eCSSUnit_Null) {
+ aResult.AppendLiteral(" [");
+ AppendValueListToString(repeatList->mValue.GetListValue(), aProperty,
+ aResult, aSerialization);
+ aResult.Append(']');
+ }
+ }
+ aResult.Append(')');
+
+ } else if (unit == eCSSUnit_Null) {
+ // Empty or omitted <line-names>.
+ if (isSubgrid) {
+ aResult.AppendLiteral("[]");
+ } else {
+ // Serializes to nothing.
+ addSpaceSeparator = false; // Avoid a double space.
+ }
+
+ } else if (unit == eCSSUnit_List || unit == eCSSUnit_ListDep) {
+ // Non-empty <line-names>
+ aResult.Append('[');
+ AppendValueListToString(val->mValue.GetListValue(), aProperty,
+ aResult, aSerialization);
+ aResult.Append(']');
+
+ } else {
+ // <track-size>
+ val->mValue.AppendToString(aProperty, aResult, aSerialization);
+ if (!isSubgrid &&
+ val->mNext &&
+ val->mNext->mValue.GetUnit() == eCSSUnit_Null &&
+ !val->mNext->mNext) {
+ // Break out of the loop early to avoid a trailing space.
+ break;
+ }
+ }
+
+ val = val->mNext;
+ if (!val) {
+ break;
+ }
+
+ if (addSpaceSeparator) {
+ aResult.Append(char16_t(' '));
+ }
+ }
+}
+
+void
+nsCSSValueList::AppendToString(nsCSSPropertyID aProperty, nsAString& aResult,
+ nsCSSValue::Serialization aSerialization) const
+{
+ if (aProperty == eCSSProperty_grid_template_columns ||
+ aProperty == eCSSProperty_grid_template_rows) {
+ AppendGridTemplateToString(this, aProperty, aResult, aSerialization);
+ } else {
+ AppendValueListToString(this, aProperty, aResult, aSerialization);
+ }
+}
+
+/* static */ bool
+nsCSSValueList::Equal(const nsCSSValueList* aList1,
+ const nsCSSValueList* aList2)
+{
+ if (aList1 == aList2) {
+ return true;
+ }
+
+ const nsCSSValueList *p1 = aList1, *p2 = aList2;
+ for ( ; p1 && p2; p1 = p1->mNext, p2 = p2->mNext) {
+ if (p1->mValue != p2->mValue)
+ return false;
+ }
+ return !p1 && !p2; // true if same length, false otherwise
+}
+
+size_t
+nsCSSValueList::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ size_t n = 0;
+ const nsCSSValueList* v = this;
+ while (v) {
+ n += aMallocSizeOf(v);
+ n += v->mValue.SizeOfExcludingThis(aMallocSizeOf);
+ v = v->mNext;
+ }
+ return n;
+}
+
+size_t
+nsCSSValueList_heap::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ // Only measure it if it's unshared, to avoid double-counting.
+ size_t n = 0;
+ if (mRefCnt <= 1) {
+ n += aMallocSizeOf(this);
+ n += mValue.SizeOfExcludingThis(aMallocSizeOf);
+ n += mNext ? mNext->SizeOfIncludingThis(aMallocSizeOf) : 0;
+ }
+ return n;
+}
+
+// --- nsCSSValueSharedList -----------------
+
+nsCSSValueSharedList::~nsCSSValueSharedList()
+{
+ MOZ_COUNT_DTOR(nsCSSValueSharedList);
+ if (mHead) {
+ NS_CSS_DELETE_LIST_MEMBER(nsCSSValueList, mHead, mNext);
+ delete mHead;
+ }
+}
+
+void
+nsCSSValueSharedList::AppendToString(nsCSSPropertyID aProperty, nsAString& aResult,
+ nsCSSValue::Serialization aSerialization) const
+{
+ if (mHead) {
+ mHead->AppendToString(aProperty, aResult, aSerialization);
+ }
+}
+
+bool
+nsCSSValueSharedList::operator==(const nsCSSValueSharedList& aOther) const
+{
+ return nsCSSValueList::Equal(mHead, aOther.mHead);
+}
+
+size_t
+nsCSSValueSharedList::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ // Only measure it if it's unshared, to avoid double-counting.
+ size_t n = 0;
+ if (mRefCnt <= 1) {
+ n += aMallocSizeOf(this);
+ n += mHead->SizeOfIncludingThis(aMallocSizeOf);
+ }
+ return n;
+}
+
+// --- nsCSSRect -----------------
+
+nsCSSRect::nsCSSRect(void)
+{
+ MOZ_COUNT_CTOR(nsCSSRect);
+}
+
+nsCSSRect::nsCSSRect(const nsCSSRect& aCopy)
+ : mTop(aCopy.mTop),
+ mRight(aCopy.mRight),
+ mBottom(aCopy.mBottom),
+ mLeft(aCopy.mLeft)
+{
+ MOZ_COUNT_CTOR(nsCSSRect);
+}
+
+nsCSSRect::~nsCSSRect()
+{
+ MOZ_COUNT_DTOR(nsCSSRect);
+}
+
+void
+nsCSSRect::AppendToString(nsCSSPropertyID aProperty, nsAString& aResult,
+ nsCSSValue::Serialization aSerialization) const
+{
+ MOZ_ASSERT(mTop.GetUnit() != eCSSUnit_Null &&
+ mTop.GetUnit() != eCSSUnit_Inherit &&
+ mTop.GetUnit() != eCSSUnit_Initial &&
+ mTop.GetUnit() != eCSSUnit_Unset,
+ "parser should have used a bare value");
+
+ if (eCSSProperty_border_image_slice == aProperty ||
+ eCSSProperty_border_image_width == aProperty ||
+ eCSSProperty_border_image_outset == aProperty ||
+ eCSSProperty_DOM == aProperty) {
+ NS_NAMED_LITERAL_STRING(space, " ");
+
+ mTop.AppendToString(aProperty, aResult, aSerialization);
+ aResult.Append(space);
+ mRight.AppendToString(aProperty, aResult, aSerialization);
+ aResult.Append(space);
+ mBottom.AppendToString(aProperty, aResult, aSerialization);
+ aResult.Append(space);
+ mLeft.AppendToString(aProperty, aResult, aSerialization);
+ } else {
+ NS_NAMED_LITERAL_STRING(comma, ", ");
+
+ aResult.AppendLiteral("rect(");
+ mTop.AppendToString(aProperty, aResult, aSerialization);
+ aResult.Append(comma);
+ mRight.AppendToString(aProperty, aResult, aSerialization);
+ aResult.Append(comma);
+ mBottom.AppendToString(aProperty, aResult, aSerialization);
+ aResult.Append(comma);
+ mLeft.AppendToString(aProperty, aResult, aSerialization);
+ aResult.Append(char16_t(')'));
+ }
+}
+
+void nsCSSRect::SetAllSidesTo(const nsCSSValue& aValue)
+{
+ mTop = aValue;
+ mRight = aValue;
+ mBottom = aValue;
+ mLeft = aValue;
+}
+
+size_t
+nsCSSRect_heap::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ // Only measure it if it's unshared, to avoid double-counting.
+ size_t n = 0;
+ if (mRefCnt <= 1) {
+ n += aMallocSizeOf(this);
+ n += mTop .SizeOfExcludingThis(aMallocSizeOf);
+ n += mRight .SizeOfExcludingThis(aMallocSizeOf);
+ n += mBottom.SizeOfExcludingThis(aMallocSizeOf);
+ n += mLeft .SizeOfExcludingThis(aMallocSizeOf);
+ }
+ return n;
+}
+
+static_assert(NS_SIDE_TOP == 0 && NS_SIDE_RIGHT == 1 &&
+ NS_SIDE_BOTTOM == 2 && NS_SIDE_LEFT == 3,
+ "box side constants not top/right/bottom/left == 0/1/2/3");
+
+/* static */ const nsCSSRect::side_type nsCSSRect::sides[4] = {
+ &nsCSSRect::mTop,
+ &nsCSSRect::mRight,
+ &nsCSSRect::mBottom,
+ &nsCSSRect::mLeft,
+};
+
+// --- nsCSSValuePair -----------------
+
+void
+nsCSSValuePair::AppendToString(nsCSSPropertyID aProperty,
+ nsAString& aResult,
+ nsCSSValue::Serialization aSerialization) const
+{
+ mXValue.AppendToString(aProperty, aResult, aSerialization);
+ if (mYValue.GetUnit() != eCSSUnit_Null) {
+ aResult.Append(char16_t(' '));
+ mYValue.AppendToString(aProperty, aResult, aSerialization);
+ }
+}
+
+size_t
+nsCSSValuePair::SizeOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ size_t n = 0;
+ n += mXValue.SizeOfExcludingThis(aMallocSizeOf);
+ n += mYValue.SizeOfExcludingThis(aMallocSizeOf);
+ return n;
+}
+
+size_t
+nsCSSValuePair_heap::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ // Only measure it if it's unshared, to avoid double-counting.
+ size_t n = 0;
+ if (mRefCnt <= 1) {
+ n += aMallocSizeOf(this);
+ n += mXValue.SizeOfExcludingThis(aMallocSizeOf);
+ n += mYValue.SizeOfExcludingThis(aMallocSizeOf);
+ }
+ return n;
+}
+
+// --- nsCSSValueTriplet -----------------
+
+void
+nsCSSValueTriplet::AppendToString(nsCSSPropertyID aProperty,
+ nsAString& aResult,
+ nsCSSValue::Serialization aSerialization) const
+{
+ mXValue.AppendToString(aProperty, aResult, aSerialization);
+ if (mYValue.GetUnit() != eCSSUnit_Null) {
+ aResult.Append(char16_t(' '));
+ mYValue.AppendToString(aProperty, aResult, aSerialization);
+ if (mZValue.GetUnit() != eCSSUnit_Null) {
+ aResult.Append(char16_t(' '));
+ mZValue.AppendToString(aProperty, aResult, aSerialization);
+ }
+ }
+}
+
+size_t
+nsCSSValueTriplet_heap::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ // Only measure it if it's unshared, to avoid double-counting.
+ size_t n = 0;
+ if (mRefCnt <= 1) {
+ n += aMallocSizeOf(this);
+ n += mXValue.SizeOfExcludingThis(aMallocSizeOf);
+ n += mYValue.SizeOfExcludingThis(aMallocSizeOf);
+ n += mZValue.SizeOfExcludingThis(aMallocSizeOf);
+ }
+ return n;
+}
+
+// --- nsCSSValuePairList -----------------
+
+nsCSSValuePairList::~nsCSSValuePairList()
+{
+ MOZ_COUNT_DTOR(nsCSSValuePairList);
+ NS_CSS_DELETE_LIST_MEMBER(nsCSSValuePairList, this, mNext);
+}
+
+nsCSSValuePairList*
+nsCSSValuePairList::Clone() const
+{
+ nsCSSValuePairList* result = new nsCSSValuePairList(*this);
+ nsCSSValuePairList* dest = result;
+ const nsCSSValuePairList* src = this->mNext;
+ while (src) {
+ dest->mNext = new nsCSSValuePairList(*src);
+ dest = dest->mNext;
+ src = src->mNext;
+ }
+
+ MOZ_ASSERT(result, "shouldn't return null; supposed to be infallible");
+ return result;
+}
+
+void
+nsCSSValuePairList::AppendToString(nsCSSPropertyID aProperty,
+ nsAString& aResult,
+ nsCSSValue::Serialization aSerialization) const
+{
+ const nsCSSValuePairList* item = this;
+ for (;;) {
+ MOZ_ASSERT(item->mXValue.GetUnit() != eCSSUnit_Null,
+ "unexpected null unit");
+ item->mXValue.AppendToString(aProperty, aResult, aSerialization);
+ if (item->mXValue.GetUnit() != eCSSUnit_Inherit &&
+ item->mXValue.GetUnit() != eCSSUnit_Initial &&
+ item->mXValue.GetUnit() != eCSSUnit_Unset &&
+ item->mYValue.GetUnit() != eCSSUnit_Null) {
+ aResult.Append(char16_t(' '));
+ item->mYValue.AppendToString(aProperty, aResult, aSerialization);
+ }
+ item = item->mNext;
+ if (!item)
+ break;
+
+ if (nsCSSProps::PropHasFlags(aProperty,
+ CSS_PROPERTY_VALUE_LIST_USES_COMMAS) ||
+ aProperty == eCSSProperty_clip_path ||
+ aProperty == eCSSProperty_shape_outside)
+ aResult.Append(char16_t(','));
+ aResult.Append(char16_t(' '));
+ }
+}
+
+/* static */ bool
+nsCSSValuePairList::Equal(const nsCSSValuePairList* aList1,
+ const nsCSSValuePairList* aList2)
+{
+ if (aList1 == aList2) {
+ return true;
+ }
+
+ const nsCSSValuePairList *p1 = aList1, *p2 = aList2;
+ for ( ; p1 && p2; p1 = p1->mNext, p2 = p2->mNext) {
+ if (p1->mXValue != p2->mXValue ||
+ p1->mYValue != p2->mYValue)
+ return false;
+ }
+ return !p1 && !p2; // true if same length, false otherwise
+}
+
+size_t
+nsCSSValuePairList::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ size_t n = 0;
+ const nsCSSValuePairList* v = this;
+ while (v) {
+ n += aMallocSizeOf(v);
+ n += v->mXValue.SizeOfExcludingThis(aMallocSizeOf);
+ n += v->mYValue.SizeOfExcludingThis(aMallocSizeOf);
+ v = v->mNext;
+ }
+ return n;
+}
+
+size_t
+nsCSSValuePairList_heap::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ // Only measure it if it's unshared, to avoid double-counting.
+ size_t n = 0;
+ if (mRefCnt <= 1) {
+ n += aMallocSizeOf(this);
+ n += mXValue.SizeOfExcludingThis(aMallocSizeOf);
+ n += mYValue.SizeOfExcludingThis(aMallocSizeOf);
+ n += mNext ? mNext->SizeOfIncludingThis(aMallocSizeOf) : 0;
+ }
+ return n;
+}
+
+size_t
+nsCSSValue::Array::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ size_t n = aMallocSizeOf(this);
+ for (size_t i = 0; i < mCount; i++) {
+ n += mArray[i].SizeOfExcludingThis(aMallocSizeOf);
+ }
+ return n;
+}
+
+css::URLValueData::URLValueData(already_AddRefed<PtrHolder<nsIURI>> aURI,
+ nsStringBuffer* aString,
+ already_AddRefed<PtrHolder<nsIURI>> aBaseURI,
+ already_AddRefed<PtrHolder<nsIURI>> aReferrer,
+ already_AddRefed<PtrHolder<nsIPrincipal>>
+ aOriginPrincipal)
+ : mURI(Move(aURI))
+ , mBaseURI(Move(aBaseURI))
+ , mString(aString)
+ , mReferrer(Move(aReferrer))
+ , mOriginPrincipal(Move(aOriginPrincipal))
+ , mURIResolved(true)
+ , mIsLocalRef(IsLocalRefURL(aString))
+{
+ MOZ_ASSERT(mString);
+ MOZ_ASSERT(mBaseURI);
+ MOZ_ASSERT(mOriginPrincipal);
+}
+
+css::URLValueData::URLValueData(nsStringBuffer* aString,
+ already_AddRefed<PtrHolder<nsIURI>> aBaseURI,
+ already_AddRefed<PtrHolder<nsIURI>> aReferrer,
+ already_AddRefed<PtrHolder<nsIPrincipal>>
+ aOriginPrincipal)
+ : mBaseURI(Move(aBaseURI))
+ , mString(aString)
+ , mReferrer(Move(aReferrer))
+ , mOriginPrincipal(Move(aOriginPrincipal))
+ , mURIResolved(false)
+ , mIsLocalRef(IsLocalRefURL(aString))
+{
+ MOZ_ASSERT(aString);
+ MOZ_ASSERT(mBaseURI);
+ MOZ_ASSERT(mOriginPrincipal);
+}
+
+bool
+css::URLValueData::Equals(const URLValueData& aOther) const
+{
+ MOZ_ASSERT(NS_IsMainThread());
+
+ bool eq;
+ // Cast away const so we can call nsIPrincipal::Equals.
+ auto& self = *const_cast<URLValueData*>(this);
+ auto& other = const_cast<URLValueData&>(aOther);
+ return NS_strcmp(nsCSSValue::GetBufferValue(mString),
+ nsCSSValue::GetBufferValue(aOther.mString)) == 0 &&
+ (GetURI() == aOther.GetURI() || // handles null == null
+ (mURI && aOther.mURI &&
+ NS_SUCCEEDED(mURI->Equals(aOther.mURI, &eq)) &&
+ eq)) &&
+ (mBaseURI == aOther.mBaseURI ||
+ (NS_SUCCEEDED(self.mBaseURI.get()->Equals(other.mBaseURI.get(), &eq)) &&
+ eq)) &&
+ (mOriginPrincipal == aOther.mOriginPrincipal ||
+ self.mOriginPrincipal.get()->Equals(other.mOriginPrincipal.get())) &&
+ mIsLocalRef == aOther.mIsLocalRef;
+}
+
+bool
+css::URLValueData::DefinitelyEqualURIs(const URLValueData& aOther) const
+{
+ return mBaseURI == aOther.mBaseURI &&
+ (mString == aOther.mString ||
+ NS_strcmp(nsCSSValue::GetBufferValue(mString),
+ nsCSSValue::GetBufferValue(aOther.mString)) == 0);
+}
+
+bool
+css::URLValueData::DefinitelyEqualURIsAndPrincipal(
+ const URLValueData& aOther) const
+{
+ return mOriginPrincipal == aOther.mOriginPrincipal &&
+ DefinitelyEqualURIs(aOther);
+}
+
+nsIURI*
+css::URLValueData::GetURI() const
+{
+ MOZ_ASSERT(NS_IsMainThread());
+
+ if (!mURIResolved) {
+ MOZ_ASSERT(!mURI);
+ nsCOMPtr<nsIURI> newURI;
+ NS_NewURI(getter_AddRefs(newURI),
+ NS_ConvertUTF16toUTF8(nsCSSValue::GetBufferValue(mString)),
+ nullptr, const_cast<nsIURI*>(mBaseURI.get()));
+ mURI = new PtrHolder<nsIURI>(newURI.forget());
+ mURIResolved = true;
+ }
+
+ return mURI;
+}
+
+already_AddRefed<nsIURI>
+css::URLValueData::ResolveLocalRef(nsIURI* aURI) const
+{
+ nsCOMPtr<nsIURI> result = GetURI();
+
+ if (result && mIsLocalRef) {
+ nsCString ref;
+ mURI->GetRef(ref);
+
+ aURI->Clone(getter_AddRefs(result));
+ result->SetRef(ref);
+ }
+
+ return result.forget();
+}
+
+already_AddRefed<nsIURI>
+css::URLValueData::ResolveLocalRef(nsIContent* aContent) const
+{
+ nsCOMPtr<nsIURI> url = aContent->GetBaseURI();
+ return ResolveLocalRef(url);
+}
+
+void
+css::URLValueData::GetSourceString(nsString& aRef) const
+{
+ nsIURI* uri = GetURI();
+ if (!uri) {
+ aRef.Truncate();
+ return;
+ }
+
+ nsCString cref;
+ if (mIsLocalRef) {
+ // XXXheycam It's possible we can just return mString in this case, since
+ // it should be the "#fragment" string the URLValueData was created with.
+ uri->GetRef(cref);
+ cref.Insert('#', 0);
+ } else {
+ // It's not entirely clear how to best handle failure here. Ensuring the
+ // string is empty seems safest.
+ nsresult rv = uri->GetSpec(cref);
+ if (NS_FAILED(rv)) {
+ cref.Truncate();
+ }
+ }
+
+ aRef = NS_ConvertUTF8toUTF16(cref);
+}
+
+bool
+css::URLValueData::EqualsExceptRef(nsIURI* aURI) const
+{
+ nsIURI* uri = GetURI();
+ if (!uri) {
+ return false;
+ }
+
+ bool ret = false;
+ uri->EqualsExceptRef(aURI, &ret);
+ return ret;
+}
+
+size_t
+css::URLValueData::SizeOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ size_t n = 0;
+ n += mString->SizeOfIncludingThisIfUnshared(aMallocSizeOf);
+
+ // Measurement of the following members may be added later if DMD finds it
+ // is worthwhile:
+ // - mURI
+ // - mReferrer
+ // - mOriginPrincipal
+ return n;
+}
+
+URLValue::URLValue(nsStringBuffer* aString, nsIURI* aBaseURI, nsIURI* aReferrer,
+ nsIPrincipal* aOriginPrincipal)
+ : URLValueData(aString,
+ do_AddRef(new PtrHolder<nsIURI>(aBaseURI)),
+ do_AddRef(new PtrHolder<nsIURI>(aReferrer)),
+ do_AddRef(new PtrHolder<nsIPrincipal>(aOriginPrincipal)))
+{
+ MOZ_ASSERT(NS_IsMainThread());
+}
+
+URLValue::URLValue(nsIURI* aURI, nsStringBuffer* aString, nsIURI* aBaseURI,
+ nsIURI* aReferrer, nsIPrincipal* aOriginPrincipal)
+ : URLValueData(do_AddRef(new PtrHolder<nsIURI>(aURI)),
+ aString,
+ do_AddRef(new PtrHolder<nsIURI>(aBaseURI)),
+ do_AddRef(new PtrHolder<nsIURI>(aReferrer)),
+ do_AddRef(new PtrHolder<nsIPrincipal>(aOriginPrincipal)))
+{
+ MOZ_ASSERT(NS_IsMainThread());
+}
+
+size_t
+css::URLValue::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ // Only measure it if it's unshared, to avoid double-counting.
+ size_t n = 0;
+ if (mRefCnt <= 1) {
+ n += aMallocSizeOf(this);
+ n += URLValueData::SizeOfExcludingThis(aMallocSizeOf);
+ }
+ return n;
+}
+
+css::ImageValue::ImageValue(nsIURI* aURI, nsStringBuffer* aString,
+ nsIURI* aBaseURI, nsIURI* aReferrer,
+ nsIPrincipal* aOriginPrincipal,
+ nsIDocument* aDocument)
+ : URLValueData(do_AddRef(new PtrHolder<nsIURI>(aURI)),
+ aString,
+ do_AddRef(new PtrHolder<nsIURI>(aBaseURI, false)),
+ do_AddRef(new PtrHolder<nsIURI>(aReferrer)),
+ do_AddRef(new PtrHolder<nsIPrincipal>(aOriginPrincipal)))
+{
+ Initialize(aDocument);
+}
+
+css::ImageValue::ImageValue(
+ nsStringBuffer* aString,
+ already_AddRefed<PtrHolder<nsIURI>> aBaseURI,
+ already_AddRefed<PtrHolder<nsIURI>> aReferrer,
+ already_AddRefed<PtrHolder<nsIPrincipal>> aOriginPrincipal)
+ : URLValueData(aString, Move(aBaseURI), Move(aReferrer),
+ Move(aOriginPrincipal))
+{
+}
+
+void
+css::ImageValue::Initialize(nsIDocument* aDocument)
+{
+ MOZ_ASSERT(NS_IsMainThread());
+ MOZ_ASSERT(!mInitialized);
+
+ // NB: If aDocument is not the original document, we may not be able to load
+ // images from aDocument. Instead we do the image load from the original doc
+ // and clone it to aDocument.
+ nsIDocument* loadingDoc = aDocument->GetOriginalDocument();
+ if (!loadingDoc) {
+ loadingDoc = aDocument;
+ }
+
+ loadingDoc->StyleImageLoader()->LoadImage(GetURI(), mOriginPrincipal,
+ mReferrer, this);
+
+ if (loadingDoc != aDocument) {
+ aDocument->StyleImageLoader()->MaybeRegisterCSSImage(this);
+ }
+
+#ifdef DEBUG
+ mInitialized = true;
+#endif
+}
+
+css::ImageValue::~ImageValue()
+{
+ MOZ_ASSERT(NS_IsMainThread());
+
+ for (auto iter = mRequests.Iter(); !iter.Done(); iter.Next()) {
+ nsIDocument* doc = iter.Key();
+ RefPtr<imgRequestProxy>& proxy = iter.Data();
+
+ if (doc) {
+ doc->StyleImageLoader()->DeregisterCSSImage(this);
+ }
+
+ if (proxy) {
+ proxy->CancelAndForgetObserver(NS_BINDING_ABORTED);
+ }
+
+ iter.Remove();
+ }
+}
+
+size_t
+css::ComplexColorValue::SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const
+{
+ // Only measure it if it's unshared, to avoid double-counting.
+ size_t n = 0;
+ if (mRefCnt <= 1) {
+ n += aMallocSizeOf(this);
+ }
+ return n;
+}
+
+nsCSSValueGradientStop::nsCSSValueGradientStop()
+ : mLocation(eCSSUnit_None),
+ mColor(eCSSUnit_Null),
+ mIsInterpolationHint(false)
+{
+ MOZ_COUNT_CTOR(nsCSSValueGradientStop);
+}
+
+nsCSSValueGradientStop::nsCSSValueGradientStop(const nsCSSValueGradientStop& aOther)
+ : mLocation(aOther.mLocation),
+ mColor(aOther.mColor),
+ mIsInterpolationHint(aOther.mIsInterpolationHint)
+{
+ MOZ_COUNT_CTOR(nsCSSValueGradientStop);
+}
+
+nsCSSValueGradientStop::~nsCSSValueGradientStop()
+{
+ MOZ_COUNT_DTOR(nsCSSValueGradientStop);
+}
+
+size_t
+nsCSSValueGradientStop::SizeOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ size_t n = 0;
+ n += mLocation.SizeOfExcludingThis(aMallocSizeOf);
+ n += mColor .SizeOfExcludingThis(aMallocSizeOf);
+ return n;
+}
+
+nsCSSValueGradient::nsCSSValueGradient(bool aIsRadial,
+ bool aIsRepeating)
+ : mIsRadial(aIsRadial),
+ mIsRepeating(aIsRepeating),
+ mIsLegacySyntax(false),
+ mIsExplicitSize(false),
+ mBgPos(eCSSUnit_None),
+ mAngle(eCSSUnit_None)
+{
+ mRadialValues[0].SetNoneValue();
+ mRadialValues[1].SetNoneValue();
+}
+
+size_t
+nsCSSValueGradient::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ // Only measure it if it's unshared, to avoid double-counting.
+ size_t n = 0;
+ if (mRefCnt <= 1) {
+ n += aMallocSizeOf(this);
+ n += mBgPos.SizeOfExcludingThis(aMallocSizeOf);
+ n += mAngle.SizeOfExcludingThis(aMallocSizeOf);
+ n += mRadialValues[0].SizeOfExcludingThis(aMallocSizeOf);
+ n += mRadialValues[1].SizeOfExcludingThis(aMallocSizeOf);
+ n += mStops.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ for (uint32_t i = 0; i < mStops.Length(); i++) {
+ n += mStops[i].SizeOfExcludingThis(aMallocSizeOf);
+ }
+ }
+ return n;
+}
+
+// --- nsCSSValueTokenStream ------------
+
+nsCSSValueTokenStream::nsCSSValueTokenStream()
+ : mPropertyID(eCSSProperty_UNKNOWN)
+ , mShorthandPropertyID(eCSSProperty_UNKNOWN)
+ , mLevel(SheetType::Count)
+{
+ MOZ_COUNT_CTOR(nsCSSValueTokenStream);
+}
+
+nsCSSValueTokenStream::~nsCSSValueTokenStream()
+{
+ MOZ_COUNT_DTOR(nsCSSValueTokenStream);
+}
+
+size_t
+nsCSSValueTokenStream::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ // Only measure it if it's unshared, to avoid double-counting.
+ size_t n = 0;
+ if (mRefCnt <= 1) {
+ n += aMallocSizeOf(this);
+ n += mTokenStream.SizeOfExcludingThisIfUnshared(aMallocSizeOf);
+ }
+ return n;
+}
+
+// --- nsCSSValueFloatColor -------------
+
+bool
+nsCSSValueFloatColor::operator==(nsCSSValueFloatColor& aOther) const
+{
+ return mComponent1 == aOther.mComponent1 &&
+ mComponent2 == aOther.mComponent2 &&
+ mComponent3 == aOther.mComponent3 &&
+ mAlpha == aOther.mAlpha;
+}
+
+nscolor
+nsCSSValueFloatColor::GetColorValue(nsCSSUnit aUnit) const
+{
+ MOZ_ASSERT(nsCSSValue::IsFloatColorUnit(aUnit), "unexpected unit");
+
+ // We should clamp each component value since eCSSUnit_PercentageRGBColor
+ // and eCSSUnit_PercentageRGBAColor may store values greater than 1.0.
+ if (aUnit == eCSSUnit_PercentageRGBColor ||
+ aUnit == eCSSUnit_PercentageRGBAColor) {
+ return NS_RGBA(
+ // We need to clamp before multiplying by 255.0f to avoid overflow.
+ NSToIntRound(mozilla::clamped(mComponent1, 0.0f, 1.0f) * 255.0f),
+ NSToIntRound(mozilla::clamped(mComponent2, 0.0f, 1.0f) * 255.0f),
+ NSToIntRound(mozilla::clamped(mComponent3, 0.0f, 1.0f) * 255.0f),
+ NSToIntRound(mozilla::clamped(mAlpha, 0.0f, 1.0f) * 255.0f));
+ }
+
+ // HSL color
+ MOZ_ASSERT(aUnit == eCSSUnit_HSLColor ||
+ aUnit == eCSSUnit_HSLAColor);
+ nscolor hsl = NS_HSL2RGB(mComponent1, mComponent2, mComponent3);
+ return NS_RGBA(NS_GET_R(hsl),
+ NS_GET_G(hsl),
+ NS_GET_B(hsl),
+ NSToIntRound(mAlpha * 255.0f));
+}
+
+bool
+nsCSSValueFloatColor::IsNonTransparentColor() const
+{
+ return mAlpha > 0.0f;
+}
+
+void
+nsCSSValueFloatColor::AppendToString(nsCSSUnit aUnit, nsAString& aResult) const
+{
+ // Similar to the rgb()/rgba() case in nsCSSValue::AppendToString. We omit the
+ // alpha component if it's equal to 1.0f (full opaque). Also, we try to
+ // preserve the author-specified function name, unless it's rgba()/hsla() and
+ // we're omitting the alpha component - then we use rgb()/hsl().
+ MOZ_ASSERT(nsCSSValue::IsFloatColorUnit(aUnit), "unexpected unit");
+
+ bool showAlpha = (mAlpha != 1.0f);
+ bool isHSL = (aUnit == eCSSUnit_HSLColor ||
+ aUnit == eCSSUnit_HSLAColor);
+
+ if (isHSL) {
+ aResult.AppendLiteral("hsl");
+ } else {
+ aResult.AppendLiteral("rgb");
+ }
+ if (showAlpha && (aUnit == eCSSUnit_HSLAColor || aUnit == eCSSUnit_PercentageRGBAColor)) {
+ aResult.AppendLiteral("a(");
+ } else {
+ aResult.Append('(');
+ }
+ if (isHSL) {
+ aResult.AppendFloat(mComponent1 * 360.0f);
+ aResult.AppendLiteral(", ");
+ } else {
+ aResult.AppendFloat(mComponent1 * 100.0f);
+ aResult.AppendLiteral("%, ");
+ }
+ aResult.AppendFloat(mComponent2 * 100.0f);
+ aResult.AppendLiteral("%, ");
+ aResult.AppendFloat(mComponent3 * 100.0f);
+ if (showAlpha) {
+ aResult.AppendLiteral("%, ");
+ aResult.AppendFloat(mAlpha);
+ aResult.Append(')');
+ } else {
+ aResult.AppendLiteral("%)");
+ }
+}
+
+size_t
+nsCSSValueFloatColor::SizeOfIncludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ // Only measure it if it's unshared, to avoid double-counting.
+ size_t n = 0;
+ if (mRefCnt <= 1) {
+ n += aMallocSizeOf(this);
+ }
+ return n;
+}
+
+// --- nsCSSCornerSizes -----------------
+
+nsCSSCornerSizes::nsCSSCornerSizes(void)
+{
+ MOZ_COUNT_CTOR(nsCSSCornerSizes);
+}
+
+nsCSSCornerSizes::nsCSSCornerSizes(const nsCSSCornerSizes& aCopy)
+ : mTopLeft(aCopy.mTopLeft),
+ mTopRight(aCopy.mTopRight),
+ mBottomRight(aCopy.mBottomRight),
+ mBottomLeft(aCopy.mBottomLeft)
+{
+ MOZ_COUNT_CTOR(nsCSSCornerSizes);
+}
+
+nsCSSCornerSizes::~nsCSSCornerSizes()
+{
+ MOZ_COUNT_DTOR(nsCSSCornerSizes);
+}
+
+void
+nsCSSCornerSizes::Reset()
+{
+ NS_FOR_CSS_FULL_CORNERS(corner) {
+ this->GetCorner(corner).Reset();
+ }
+}
+
+static_assert(NS_CORNER_TOP_LEFT == 0 && NS_CORNER_TOP_RIGHT == 1 &&
+ NS_CORNER_BOTTOM_RIGHT == 2 && NS_CORNER_BOTTOM_LEFT == 3,
+ "box corner constants not tl/tr/br/bl == 0/1/2/3");
+
+/* static */ const nsCSSCornerSizes::corner_type
+nsCSSCornerSizes::corners[4] = {
+ &nsCSSCornerSizes::mTopLeft,
+ &nsCSSCornerSizes::mTopRight,
+ &nsCSSCornerSizes::mBottomRight,
+ &nsCSSCornerSizes::mBottomLeft,
+};
+
+size_t
+mozilla::css::GridTemplateAreasValue::SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
+{
+ // Only measure it if it's unshared, to avoid double-counting.
+ size_t n = 0;
+ if (mRefCnt <= 1) {
+ n += aMallocSizeOf(this);
+ n += mNamedAreas.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ n += mTemplates.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ }
+ return n;
+}