forked from mirrors/gecko-dev
		
	 bb6f4f363b
			
		
	
	
		bb6f4f363b
		
	
	
	
	
		
			
			See https://www.w3.org/TR/filter-effects-1/#tainted-filter-primitives Differential Revision: https://phabricator.services.mozilla.com/D195996
		
			
				
	
	
		
			438 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			438 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
 | |
| /* vim: set ts=8 sts=2 et sw=2 tw=80: */
 | |
| /* This Source Code Form is subject to the terms of the Mozilla Public
 | |
|  * License, v. 2.0. If a copy of the MPL was not distributed with this
 | |
|  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | |
| 
 | |
| #include "SVGFilters.h"
 | |
| 
 | |
| #include <algorithm>
 | |
| #include "DOMSVGAnimatedNumberList.h"
 | |
| #include "DOMSVGAnimatedLength.h"
 | |
| #include "nsGkAtoms.h"
 | |
| #include "nsCOMPtr.h"
 | |
| #include "nsIFrame.h"
 | |
| #include "nsLayoutUtils.h"
 | |
| #include "SVGAnimatedEnumeration.h"
 | |
| #include "SVGAnimatedNumberPair.h"
 | |
| #include "SVGAnimatedString.h"
 | |
| #include "SVGNumberList.h"
 | |
| #include "mozilla/ArrayUtils.h"
 | |
| #include "mozilla/ComputedStyle.h"
 | |
| #include "mozilla/SVGContentUtils.h"
 | |
| #include "mozilla/SVGFilterInstance.h"
 | |
| #include "mozilla/dom/SVGComponentTransferFunctionElement.h"
 | |
| #include "mozilla/dom/SVGElement.h"
 | |
| #include "mozilla/dom/SVGFEDistantLightElement.h"
 | |
| #include "mozilla/dom/SVGFEFuncAElementBinding.h"
 | |
| #include "mozilla/dom/SVGFEFuncBElementBinding.h"
 | |
| #include "mozilla/dom/SVGFEFuncGElementBinding.h"
 | |
| #include "mozilla/dom/SVGFEFuncRElementBinding.h"
 | |
| #include "mozilla/dom/SVGFEPointLightElement.h"
 | |
| #include "mozilla/dom/SVGFESpotLightElement.h"
 | |
| #include "mozilla/dom/SVGFilterElement.h"
 | |
| #include "mozilla/dom/SVGLengthBinding.h"
 | |
| 
 | |
| #if defined(XP_WIN)
 | |
| // Prevent Windows redefining LoadImage
 | |
| #  undef LoadImage
 | |
| #endif
 | |
| 
 | |
| using namespace mozilla::gfx;
 | |
| 
 | |
| namespace mozilla::dom {
 | |
| 
 | |
| //--------------------Filter Element Base Class-----------------------
 | |
| 
 | |
| SVGElement::LengthInfo SVGFilterPrimitiveElement::sLengthInfo[4] = {
 | |
|     {nsGkAtoms::x, 0, SVGLength_Binding::SVG_LENGTHTYPE_PERCENTAGE,
 | |
|      SVGContentUtils::X},
 | |
|     {nsGkAtoms::y, 0, SVGLength_Binding::SVG_LENGTHTYPE_PERCENTAGE,
 | |
|      SVGContentUtils::Y},
 | |
|     {nsGkAtoms::width, 100, SVGLength_Binding::SVG_LENGTHTYPE_PERCENTAGE,
 | |
|      SVGContentUtils::X},
 | |
|     {nsGkAtoms::height, 100, SVGLength_Binding::SVG_LENGTHTYPE_PERCENTAGE,
 | |
|      SVGContentUtils::Y}};
 | |
| 
 | |
| //----------------------------------------------------------------------
 | |
| // Implementation
 | |
| 
 | |
| void SVGFilterPrimitiveElement::GetSourceImageNames(
 | |
|     nsTArray<SVGStringInfo>& aSources) {}
 | |
| 
 | |
| bool SVGFilterPrimitiveElement::OutputIsTainted(
 | |
|     const nsTArray<bool>& aInputsAreTainted,
 | |
|     nsIPrincipal* aReferencePrincipal) {
 | |
|   // This is the default implementation for OutputIsTainted.
 | |
|   // Our output is tainted if we have at least one tainted input.
 | |
|   return aInputsAreTainted.Contains(true);
 | |
| }
 | |
| 
 | |
| bool SVGFilterPrimitiveElement::AttributeAffectsRendering(
 | |
|     int32_t aNameSpaceID, nsAtom* aAttribute) const {
 | |
|   return aNameSpaceID == kNameSpaceID_None &&
 | |
|          (aAttribute == nsGkAtoms::x || aAttribute == nsGkAtoms::y ||
 | |
|           aAttribute == nsGkAtoms::width || aAttribute == nsGkAtoms::height ||
 | |
|           aAttribute == nsGkAtoms::result);
 | |
| }
 | |
| 
 | |
| already_AddRefed<DOMSVGAnimatedLength> SVGFilterPrimitiveElement::X() {
 | |
|   return mLengthAttributes[ATTR_X].ToDOMAnimatedLength(this);
 | |
| }
 | |
| 
 | |
| already_AddRefed<DOMSVGAnimatedLength> SVGFilterPrimitiveElement::Y() {
 | |
|   return mLengthAttributes[ATTR_Y].ToDOMAnimatedLength(this);
 | |
| }
 | |
| 
 | |
| already_AddRefed<DOMSVGAnimatedLength> SVGFilterPrimitiveElement::Width() {
 | |
|   return mLengthAttributes[ATTR_WIDTH].ToDOMAnimatedLength(this);
 | |
| }
 | |
| 
 | |
| already_AddRefed<DOMSVGAnimatedLength> SVGFilterPrimitiveElement::Height() {
 | |
|   return mLengthAttributes[ATTR_HEIGHT].ToDOMAnimatedLength(this);
 | |
| }
 | |
| 
 | |
| already_AddRefed<DOMSVGAnimatedString> SVGFilterPrimitiveElement::Result() {
 | |
|   return GetResultImageName().ToDOMAnimatedString(this);
 | |
| }
 | |
| 
 | |
| //----------------------------------------------------------------------
 | |
| // SVGElement methods
 | |
| 
 | |
| bool SVGFilterPrimitiveElement::StyleIsSetToSRGB() {
 | |
|   nsIFrame* frame = GetPrimaryFrame();
 | |
|   if (!frame) return false;
 | |
| 
 | |
|   ComputedStyle* style = frame->Style();
 | |
|   return style->StyleSVG()->mColorInterpolationFilters ==
 | |
|          StyleColorInterpolation::Srgb;
 | |
| }
 | |
| 
 | |
| /* virtual */
 | |
| bool SVGFilterPrimitiveElement::HasValidDimensions() const {
 | |
|   return (!mLengthAttributes[ATTR_WIDTH].IsExplicitlySet() ||
 | |
|           mLengthAttributes[ATTR_WIDTH].GetAnimValInSpecifiedUnits() > 0) &&
 | |
|          (!mLengthAttributes[ATTR_HEIGHT].IsExplicitlySet() ||
 | |
|           mLengthAttributes[ATTR_HEIGHT].GetAnimValInSpecifiedUnits() > 0);
 | |
| }
 | |
| 
 | |
| Size SVGFilterPrimitiveElement::GetKernelUnitLength(
 | |
|     SVGFilterInstance* aInstance, SVGAnimatedNumberPair* aKernelUnitLength) {
 | |
|   if (!aKernelUnitLength->IsExplicitlySet()) {
 | |
|     return Size(1, 1);
 | |
|   }
 | |
| 
 | |
|   float kernelX = aInstance->GetPrimitiveNumber(
 | |
|       SVGContentUtils::X, aKernelUnitLength, SVGAnimatedNumberPair::eFirst);
 | |
|   float kernelY = aInstance->GetPrimitiveNumber(
 | |
|       SVGContentUtils::Y, aKernelUnitLength, SVGAnimatedNumberPair::eSecond);
 | |
|   return Size(kernelX, kernelY);
 | |
| }
 | |
| 
 | |
| SVGElement::LengthAttributesInfo SVGFilterPrimitiveElement::GetLengthInfo() {
 | |
|   return LengthAttributesInfo(mLengthAttributes, sLengthInfo,
 | |
|                               ArrayLength(sLengthInfo));
 | |
| }
 | |
| 
 | |
| SVGElement::NumberListInfo
 | |
|     SVGComponentTransferFunctionElement::sNumberListInfo[1] = {
 | |
|         {nsGkAtoms::tableValues}};
 | |
| 
 | |
| SVGElement::NumberInfo SVGComponentTransferFunctionElement::sNumberInfo[5] = {
 | |
|     {nsGkAtoms::slope, 1},
 | |
|     {nsGkAtoms::intercept, 0},
 | |
|     {nsGkAtoms::amplitude, 1},
 | |
|     {nsGkAtoms::exponent, 1},
 | |
|     {nsGkAtoms::offset, 0}};
 | |
| 
 | |
| SVGEnumMapping SVGComponentTransferFunctionElement::sTypeMap[] = {
 | |
|     {nsGkAtoms::identity, SVG_FECOMPONENTTRANSFER_TYPE_IDENTITY},
 | |
|     {nsGkAtoms::table, SVG_FECOMPONENTTRANSFER_TYPE_TABLE},
 | |
|     {nsGkAtoms::discrete, SVG_FECOMPONENTTRANSFER_TYPE_DISCRETE},
 | |
|     {nsGkAtoms::linear, SVG_FECOMPONENTTRANSFER_TYPE_LINEAR},
 | |
|     {nsGkAtoms::gamma, SVG_FECOMPONENTTRANSFER_TYPE_GAMMA},
 | |
|     {nullptr, 0}};
 | |
| 
 | |
| SVGElement::EnumInfo SVGComponentTransferFunctionElement::sEnumInfo[1] = {
 | |
|     {nsGkAtoms::type, sTypeMap, SVG_FECOMPONENTTRANSFER_TYPE_IDENTITY}};
 | |
| 
 | |
| //----------------------------------------------------------------------
 | |
| // nsSVGFilterPrimitiveChildElement methods
 | |
| 
 | |
| bool SVGComponentTransferFunctionElement::AttributeAffectsRendering(
 | |
|     int32_t aNameSpaceID, nsAtom* aAttribute) const {
 | |
|   return aNameSpaceID == kNameSpaceID_None &&
 | |
|          (aAttribute == nsGkAtoms::tableValues ||
 | |
|           aAttribute == nsGkAtoms::slope ||
 | |
|           aAttribute == nsGkAtoms::intercept ||
 | |
|           aAttribute == nsGkAtoms::amplitude ||
 | |
|           aAttribute == nsGkAtoms::exponent ||
 | |
|           aAttribute == nsGkAtoms::offset || aAttribute == nsGkAtoms::type);
 | |
| }
 | |
| 
 | |
| //----------------------------------------------------------------------
 | |
| 
 | |
| already_AddRefed<DOMSVGAnimatedEnumeration>
 | |
| SVGComponentTransferFunctionElement::Type() {
 | |
|   return mEnumAttributes[TYPE].ToDOMAnimatedEnum(this);
 | |
| }
 | |
| 
 | |
| already_AddRefed<DOMSVGAnimatedNumberList>
 | |
| SVGComponentTransferFunctionElement::TableValues() {
 | |
|   return DOMSVGAnimatedNumberList::GetDOMWrapper(
 | |
|       &mNumberListAttributes[TABLEVALUES], this, TABLEVALUES);
 | |
| }
 | |
| 
 | |
| already_AddRefed<DOMSVGAnimatedNumber>
 | |
| SVGComponentTransferFunctionElement::Slope() {
 | |
|   return mNumberAttributes[SLOPE].ToDOMAnimatedNumber(this);
 | |
| }
 | |
| 
 | |
| already_AddRefed<DOMSVGAnimatedNumber>
 | |
| SVGComponentTransferFunctionElement::Intercept() {
 | |
|   return mNumberAttributes[INTERCEPT].ToDOMAnimatedNumber(this);
 | |
| }
 | |
| 
 | |
| already_AddRefed<DOMSVGAnimatedNumber>
 | |
| SVGComponentTransferFunctionElement::Amplitude() {
 | |
|   return mNumberAttributes[AMPLITUDE].ToDOMAnimatedNumber(this);
 | |
| }
 | |
| 
 | |
| already_AddRefed<DOMSVGAnimatedNumber>
 | |
| SVGComponentTransferFunctionElement::Exponent() {
 | |
|   return mNumberAttributes[EXPONENT].ToDOMAnimatedNumber(this);
 | |
| }
 | |
| 
 | |
| already_AddRefed<DOMSVGAnimatedNumber>
 | |
| SVGComponentTransferFunctionElement::Offset() {
 | |
|   return mNumberAttributes[OFFSET].ToDOMAnimatedNumber(this);
 | |
| }
 | |
| 
 | |
| void SVGComponentTransferFunctionElement::ComputeAttributes(
 | |
|     int32_t aChannel, ComponentTransferAttributes& aAttributes) {
 | |
|   uint32_t type = mEnumAttributes[TYPE].GetAnimValue();
 | |
| 
 | |
|   float slope, intercept, amplitude, exponent, offset;
 | |
|   GetAnimatedNumberValues(&slope, &intercept, &litude, &exponent, &offset,
 | |
|                           nullptr);
 | |
| 
 | |
|   const SVGNumberList& tableValues =
 | |
|       mNumberListAttributes[TABLEVALUES].GetAnimValue();
 | |
| 
 | |
|   aAttributes.mTypes[aChannel] = (uint8_t)type;
 | |
|   switch (type) {
 | |
|     case SVG_FECOMPONENTTRANSFER_TYPE_LINEAR: {
 | |
|       aAttributes.mValues[aChannel].SetLength(2);
 | |
|       aAttributes.mValues[aChannel][kComponentTransferSlopeIndex] = slope;
 | |
|       aAttributes.mValues[aChannel][kComponentTransferInterceptIndex] =
 | |
|           intercept;
 | |
|       break;
 | |
|     }
 | |
|     case SVG_FECOMPONENTTRANSFER_TYPE_GAMMA: {
 | |
|       aAttributes.mValues[aChannel].SetLength(3);
 | |
|       aAttributes.mValues[aChannel][kComponentTransferAmplitudeIndex] =
 | |
|           amplitude;
 | |
|       aAttributes.mValues[aChannel][kComponentTransferExponentIndex] = exponent;
 | |
|       aAttributes.mValues[aChannel][kComponentTransferOffsetIndex] = offset;
 | |
|       break;
 | |
|     }
 | |
|     case SVG_FECOMPONENTTRANSFER_TYPE_DISCRETE:
 | |
|     case SVG_FECOMPONENTTRANSFER_TYPE_TABLE: {
 | |
|       if (!tableValues.IsEmpty()) {
 | |
|         aAttributes.mValues[aChannel].AppendElements(&tableValues[0],
 | |
|                                                      tableValues.Length());
 | |
|       }
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| //----------------------------------------------------------------------
 | |
| // SVGElement methods
 | |
| 
 | |
| SVGElement::NumberListAttributesInfo
 | |
| SVGComponentTransferFunctionElement::GetNumberListInfo() {
 | |
|   return NumberListAttributesInfo(mNumberListAttributes, sNumberListInfo,
 | |
|                                   ArrayLength(sNumberListInfo));
 | |
| }
 | |
| 
 | |
| SVGElement::EnumAttributesInfo
 | |
| SVGComponentTransferFunctionElement::GetEnumInfo() {
 | |
|   return EnumAttributesInfo(mEnumAttributes, sEnumInfo, ArrayLength(sEnumInfo));
 | |
| }
 | |
| 
 | |
| SVGElement::NumberAttributesInfo
 | |
| SVGComponentTransferFunctionElement::GetNumberInfo() {
 | |
|   return NumberAttributesInfo(mNumberAttributes, sNumberInfo,
 | |
|                               ArrayLength(sNumberInfo));
 | |
| }
 | |
| 
 | |
| /* virtual */
 | |
| JSObject* SVGFEFuncRElement::WrapNode(JSContext* aCx,
 | |
|                                       JS::Handle<JSObject*> aGivenProto) {
 | |
|   return SVGFEFuncRElement_Binding::Wrap(aCx, this, aGivenProto);
 | |
| }
 | |
| 
 | |
| }  // namespace mozilla::dom
 | |
| 
 | |
| NS_IMPL_NS_NEW_SVG_ELEMENT(FEFuncR)
 | |
| 
 | |
| namespace mozilla::dom {
 | |
| 
 | |
| NS_IMPL_ELEMENT_CLONE_WITH_INIT(SVGFEFuncRElement)
 | |
| 
 | |
| /* virtual */
 | |
| JSObject* SVGFEFuncGElement::WrapNode(JSContext* aCx,
 | |
|                                       JS::Handle<JSObject*> aGivenProto) {
 | |
|   return SVGFEFuncGElement_Binding::Wrap(aCx, this, aGivenProto);
 | |
| }
 | |
| 
 | |
| }  // namespace mozilla::dom
 | |
| 
 | |
| NS_IMPL_NS_NEW_SVG_ELEMENT(FEFuncG)
 | |
| 
 | |
| namespace mozilla::dom {
 | |
| 
 | |
| NS_IMPL_ELEMENT_CLONE_WITH_INIT(SVGFEFuncGElement)
 | |
| 
 | |
| /* virtual */
 | |
| JSObject* SVGFEFuncBElement::WrapNode(JSContext* aCx,
 | |
|                                       JS::Handle<JSObject*> aGivenProto) {
 | |
|   return SVGFEFuncBElement_Binding::Wrap(aCx, this, aGivenProto);
 | |
| }
 | |
| 
 | |
| }  // namespace mozilla::dom
 | |
| 
 | |
| NS_IMPL_NS_NEW_SVG_ELEMENT(FEFuncB)
 | |
| 
 | |
| namespace mozilla::dom {
 | |
| 
 | |
| NS_IMPL_ELEMENT_CLONE_WITH_INIT(SVGFEFuncBElement)
 | |
| 
 | |
| /* virtual */
 | |
| JSObject* SVGFEFuncAElement::WrapNode(JSContext* aCx,
 | |
|                                       JS::Handle<JSObject*> aGivenProto) {
 | |
|   return SVGFEFuncAElement_Binding::Wrap(aCx, this, aGivenProto);
 | |
| }
 | |
| 
 | |
| }  // namespace mozilla::dom
 | |
| 
 | |
| NS_IMPL_NS_NEW_SVG_ELEMENT(FEFuncA)
 | |
| 
 | |
| namespace mozilla::dom {
 | |
| 
 | |
| NS_IMPL_ELEMENT_CLONE_WITH_INIT(SVGFEFuncAElement)
 | |
| 
 | |
| //--------------------------------------------------------------------
 | |
| //
 | |
| SVGElement::NumberInfo SVGFELightingElement::sNumberInfo[4] = {
 | |
|     {nsGkAtoms::surfaceScale, 1},
 | |
|     {nsGkAtoms::diffuseConstant, 1},
 | |
|     {nsGkAtoms::specularConstant, 1},
 | |
|     {nsGkAtoms::specularExponent, 1}};
 | |
| 
 | |
| SVGElement::NumberPairInfo SVGFELightingElement::sNumberPairInfo[1] = {
 | |
|     {nsGkAtoms::kernelUnitLength, 0, 0}};
 | |
| 
 | |
| SVGElement::StringInfo SVGFELightingElement::sStringInfo[2] = {
 | |
|     {nsGkAtoms::result, kNameSpaceID_None, true},
 | |
|     {nsGkAtoms::in, kNameSpaceID_None, true}};
 | |
| 
 | |
| //----------------------------------------------------------------------
 | |
| // Implementation
 | |
| 
 | |
| void SVGFELightingElement::GetSourceImageNames(
 | |
|     nsTArray<SVGStringInfo>& aSources) {
 | |
|   aSources.AppendElement(SVGStringInfo(&mStringAttributes[IN1], this));
 | |
| }
 | |
| 
 | |
| LightType SVGFELightingElement::ComputeLightAttributes(
 | |
|     SVGFilterInstance* aInstance, nsTArray<float>& aFloatAttributes) {
 | |
|   // find specified light
 | |
|   for (nsCOMPtr<nsIContent> child = nsINode::GetFirstChild(); child;
 | |
|        child = child->GetNextSibling()) {
 | |
|     if (child->IsAnyOfSVGElements(nsGkAtoms::feDistantLight,
 | |
|                                   nsGkAtoms::fePointLight,
 | |
|                                   nsGkAtoms::feSpotLight)) {
 | |
|       return static_cast<SVGFELightElement*>(child.get())
 | |
|           ->ComputeLightAttributes(aInstance, aFloatAttributes);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return LightType::None;
 | |
| }
 | |
| 
 | |
| bool SVGFELightingElement::AddLightingAttributes(
 | |
|     mozilla::gfx::DiffuseLightingAttributes* aAttributes,
 | |
|     SVGFilterInstance* aInstance) {
 | |
|   const auto* frame = GetPrimaryFrame();
 | |
|   if (!frame) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   const nsStyleSVGReset* styleSVGReset = frame->Style()->StyleSVGReset();
 | |
|   sRGBColor color(
 | |
|       sRGBColor::FromABGR(styleSVGReset->mLightingColor.CalcColor(frame)));
 | |
|   color.a = 1.f;
 | |
|   float surfaceScale = mNumberAttributes[SURFACE_SCALE].GetAnimValue();
 | |
|   Size kernelUnitLength = GetKernelUnitLength(
 | |
|       aInstance, &mNumberPairAttributes[KERNEL_UNIT_LENGTH]);
 | |
| 
 | |
|   if (kernelUnitLength.width <= 0 || kernelUnitLength.height <= 0) {
 | |
|     // According to spec, A negative or zero value is an error. See link below
 | |
|     // for details.
 | |
|     // https://www.w3.org/TR/SVG/filters.html#feSpecularLightingKernelUnitLengthAttribute
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   aAttributes->mLightType =
 | |
|       ComputeLightAttributes(aInstance, aAttributes->mLightValues);
 | |
|   aAttributes->mSurfaceScale = surfaceScale;
 | |
|   aAttributes->mKernelUnitLength = kernelUnitLength;
 | |
|   aAttributes->mColor = color;
 | |
| 
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| bool SVGFELightingElement::OutputIsTainted(
 | |
|     const nsTArray<bool>& aInputsAreTainted,
 | |
|     nsIPrincipal* aReferencePrincipal) {
 | |
|   if (const auto* frame = GetPrimaryFrame()) {
 | |
|     if (frame->Style()->StyleSVGReset()->mLightingColor.IsCurrentColor()) {
 | |
|       return true;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return SVGFELightingElementBase::OutputIsTainted(aInputsAreTainted,
 | |
|                                                    aReferencePrincipal);
 | |
| }
 | |
| 
 | |
| bool SVGFELightingElement::AttributeAffectsRendering(int32_t aNameSpaceID,
 | |
|                                                      nsAtom* aAttribute) const {
 | |
|   return SVGFELightingElementBase::AttributeAffectsRendering(aNameSpaceID,
 | |
|                                                              aAttribute) ||
 | |
|          (aNameSpaceID == kNameSpaceID_None &&
 | |
|           (aAttribute == nsGkAtoms::in ||
 | |
|            aAttribute == nsGkAtoms::surfaceScale ||
 | |
|            aAttribute == nsGkAtoms::kernelUnitLength));
 | |
| }
 | |
| 
 | |
| //----------------------------------------------------------------------
 | |
| // SVGElement methods
 | |
| 
 | |
| SVGElement::NumberAttributesInfo SVGFELightingElement::GetNumberInfo() {
 | |
|   return NumberAttributesInfo(mNumberAttributes, sNumberInfo,
 | |
|                               ArrayLength(sNumberInfo));
 | |
| }
 | |
| 
 | |
| SVGElement::NumberPairAttributesInfo SVGFELightingElement::GetNumberPairInfo() {
 | |
|   return NumberPairAttributesInfo(mNumberPairAttributes, sNumberPairInfo,
 | |
|                                   ArrayLength(sNumberPairInfo));
 | |
| }
 | |
| 
 | |
| SVGElement::StringAttributesInfo SVGFELightingElement::GetStringInfo() {
 | |
|   return StringAttributesInfo(mStringAttributes, sStringInfo,
 | |
|                               ArrayLength(sStringInfo));
 | |
| }
 | |
| 
 | |
| }  // namespace mozilla::dom
 |