/* -*- 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/. */ #ifndef DOM_SVG_SVGFILTERS_H_ #define DOM_SVG_SVGFILTERS_H_ #include "mozilla/Attributes.h" #include "mozilla/dom/SVGElement.h" #include "FilterDescription.h" #include "nsImageLoadingContent.h" #include "SVGAnimatedLength.h" #include "SVGAnimatedNumber.h" #include "SVGAnimatedNumberPair.h" #include "SVGAnimatedString.h" namespace mozilla { class SVGFilterInstance; namespace dom { struct SVGStringInfo { SVGStringInfo(const SVGAnimatedString* aString, SVGElement* aElement) : mString(aString), mElement(aElement) {} const SVGAnimatedString* mString; SVGElement* mElement; }; using SVGFilterPrimitiveElementBase = SVGElement; /** * Base class for filter primitive elements * Children of those elements e.g. feMergeNode * derive from SVGFilterPrimitiveChildElement instead */ class SVGFilterPrimitiveElement : public SVGFilterPrimitiveElementBase { friend class mozilla::SVGFilterInstance; protected: using SourceSurface = mozilla::gfx::SourceSurface; using Size = mozilla::gfx::Size; using IntRect = mozilla::gfx::IntRect; using ColorSpace = mozilla::gfx::ColorSpace; using FilterPrimitiveDescription = mozilla::gfx::FilterPrimitiveDescription; explicit SVGFilterPrimitiveElement( already_AddRefed&& aNodeInfo) : SVGFilterPrimitiveElementBase(std::move(aNodeInfo)) {} virtual ~SVGFilterPrimitiveElement() = default; public: using PrimitiveAttributes = mozilla::gfx::PrimitiveAttributes; NS_IMPL_FROMNODE_HELPER(SVGFilterPrimitiveElement, IsSVGFilterPrimitiveElement()) ColorSpace GetInputColorSpace(int32_t aInputIndex, ColorSpace aUnchangedInputColorSpace) { return OperatesOnSRGB(aInputIndex, aUnchangedInputColorSpace == ColorSpace::SRGB) ? ColorSpace::SRGB : ColorSpace::LinearRGB; } // This is only called for filter primitives without inputs. For primitives // with inputs, the output color model is the same as of the first input. ColorSpace GetOutputColorSpace() { return ProducesSRGB() ? ColorSpace::SRGB : ColorSpace::LinearRGB; } // See http://www.w3.org/TR/SVG/filters.html#FilterPrimitiveSubRegion virtual bool SubregionIsUnionOfRegions() { return true; } bool IsSVGFilterPrimitiveElement() const final { return true; } // SVGElement interface nsresult Clone(mozilla::dom::NodeInfo*, nsINode** aResult) const override = 0; bool HasValidDimensions() const override; virtual SVGAnimatedString& GetResultImageName() = 0; // Return a list of all image names used as sources. Default is to // return no sources. virtual void GetSourceImageNames(nsTArray& aSources); virtual FilterPrimitiveDescription GetPrimitiveDescription( SVGFilterInstance* aInstance, const IntRect& aFilterSubregion, const nsTArray& aInputsAreTainted, nsTArray>& aInputImages) = 0; // returns true if changes to the attribute should cause us to // repaint the filter virtual bool AttributeAffectsRendering(int32_t aNameSpaceID, nsAtom* aAttribute) const; // Return whether this filter primitive has tainted output. A filter's // output is tainted if it depends on things that the web page is not // allowed to read from, e.g. the source graphic or cross-origin images. // aReferencePrincipal is the node principal of the filtered frame's element. virtual bool OutputIsTainted(const nsTArray& aInputsAreTainted, nsIPrincipal* aReferencePrincipal); static nsIntRect GetMaxRect() { // Try to avoid overflow errors dealing with this rect. It will // be intersected with some other reasonable-sized rect eventually. return nsIntRect(INT32_MIN / 2, INT32_MIN / 2, INT32_MAX, INT32_MAX); } operator nsISupports*() { return static_cast(this); } // WebIDL already_AddRefed X(); already_AddRefed Y(); already_AddRefed Width(); already_AddRefed Height(); already_AddRefed Result(); protected: virtual bool OperatesOnSRGB(int32_t aInputIndex, bool aInputIsAlreadySRGB) { return StyleIsSetToSRGB(); } // Only called for filter primitives without inputs. virtual bool ProducesSRGB() { return StyleIsSetToSRGB(); } bool StyleIsSetToSRGB(); // SVGElement specializations: LengthAttributesInfo GetLengthInfo() override; Size GetKernelUnitLength(SVGFilterInstance* aInstance, SVGAnimatedNumberPair* aKernelUnitLength); enum { ATTR_X, ATTR_Y, ATTR_WIDTH, ATTR_HEIGHT }; SVGAnimatedLength mLengthAttributes[4]; static LengthInfo sLengthInfo[4]; }; using SVGFilterPrimitiveChildElementBase = SVGElement; class SVGFilterPrimitiveChildElement : public SVGFilterPrimitiveChildElementBase { protected: explicit SVGFilterPrimitiveChildElement( already_AddRefed&& aNodeInfo) : SVGFilterPrimitiveChildElementBase(std::move(aNodeInfo)) {} public: NS_IMPL_FROMNODE_HELPER(SVGFilterPrimitiveChildElement, IsSVGFilterPrimitiveChildElement()) bool IsSVGFilterPrimitiveChildElement() const final { return true; } nsresult Clone(mozilla::dom::NodeInfo*, nsINode** aResult) const override = 0; // returns true if changes to the attribute should cause us to // repaint the filter virtual bool AttributeAffectsRendering(int32_t aNameSpaceID, nsAtom* aAttribute) const = 0; }; //------------------------------------------------------------ using SVGFELightingElementBase = SVGFilterPrimitiveElement; class SVGFELightingElement : public SVGFELightingElementBase { protected: explicit SVGFELightingElement( already_AddRefed&& aNodeInfo) : SVGFELightingElementBase(std::move(aNodeInfo)) {} virtual ~SVGFELightingElement() = default; public: // interfaces: NS_INLINE_DECL_REFCOUNTING_INHERITED(SVGFELightingElement, SVGFELightingElementBase) bool AttributeAffectsRendering(int32_t aNameSpaceID, nsAtom* aAttribute) const override; SVGAnimatedString& GetResultImageName() override { return mStringAttributes[RESULT]; } bool OutputIsTainted(const nsTArray& aInputsAreTainted, nsIPrincipal* aReferencePrincipal) override; void GetSourceImageNames(nsTArray& aSources) override; protected: bool OperatesOnSRGB(int32_t aInputIndex, bool aInputIsAlreadySRGB) override { return true; } NumberAttributesInfo GetNumberInfo() override; NumberPairAttributesInfo GetNumberPairInfo() override; StringAttributesInfo GetStringInfo() override; mozilla::gfx::LightType ComputeLightAttributes( SVGFilterInstance* aInstance, nsTArray& aFloatAttributes); bool AddLightingAttributes( mozilla::gfx::DiffuseLightingAttributes* aAttributes, SVGFilterInstance* aInstance); enum { SURFACE_SCALE, DIFFUSE_CONSTANT, SPECULAR_CONSTANT, SPECULAR_EXPONENT }; SVGAnimatedNumber mNumberAttributes[4]; static NumberInfo sNumberInfo[4]; enum { KERNEL_UNIT_LENGTH }; SVGAnimatedNumberPair mNumberPairAttributes[1]; static NumberPairInfo sNumberPairInfo[1]; enum { RESULT, IN1 }; SVGAnimatedString mStringAttributes[2]; static StringInfo sStringInfo[2]; }; using SVGFELightElementBase = SVGFilterPrimitiveChildElement; class SVGFELightElement : public SVGFELightElementBase { protected: explicit SVGFELightElement( already_AddRefed&& aNodeInfo) : SVGFELightElementBase(std::move(aNodeInfo)) {} public: using PrimitiveAttributes = gfx::PrimitiveAttributes; virtual mozilla::gfx::LightType ComputeLightAttributes( SVGFilterInstance* aInstance, nsTArray& aFloatAttributes) = 0; }; } // namespace dom } // namespace mozilla #endif // DOM_SVG_SVGFILTERS_H_