/* -*- 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 SVGFEBase = SVGElement; #define NS_SVG_FE_CID \ { \ 0x60483958, 0xd229, 0x4a77, { \ 0x96, 0xb2, 0x62, 0x3e, 0x69, 0x95, 0x1e, 0x0e \ } \ } /** * Base class for filter primitive elements * Children of those elements e.g. feMergeNode * derive from SVGFEUnstyledElement instead */ class SVGFE : public SVGFEBase { 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 SVGFE(already_AddRefed&& aNodeInfo) : SVGFEBase(std::move(aNodeInfo)) {} virtual ~SVGFE() = default; public: using PrimitiveAttributes = mozilla::gfx::PrimitiveAttributes; 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; } NS_DECLARE_STATIC_IID_ACCESSOR(NS_SVG_FE_CID) // interfaces: NS_DECL_ISUPPORTS_INHERITED // nsIContent interface NS_IMETHOD_(bool) IsAttributeMapped(const nsAtom* aAttribute) const override; // SVGElement interface virtual nsresult Clone(mozilla::dom::NodeInfo*, nsINode** aResult) const override = 0; virtual 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: virtual 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]; }; NS_DEFINE_STATIC_IID_ACCESSOR(SVGFE, NS_SVG_FE_CID) using SVGFEUnstyledElementBase = SVGElement; class SVGFEUnstyledElement : public SVGFEUnstyledElementBase { protected: explicit SVGFEUnstyledElement( already_AddRefed&& aNodeInfo) : SVGFEUnstyledElementBase(std::move(aNodeInfo)) {} public: virtual 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 = SVGFE; 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) virtual bool AttributeAffectsRendering(int32_t aNameSpaceID, nsAtom* aAttribute) const override; virtual SVGAnimatedString& GetResultImageName() override { return mStringAttributes[RESULT]; } virtual void GetSourceImageNames(nsTArray& aSources) override; NS_IMETHOD_(bool) IsAttributeMapped(const nsAtom* aAttribute) const override; protected: virtual bool OperatesOnSRGB(int32_t aInputIndex, bool aInputIsAlreadySRGB) override { return true; } virtual NumberAttributesInfo GetNumberInfo() override; virtual NumberPairAttributesInfo GetNumberPairInfo() override; virtual 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 = SVGFEUnstyledElement; 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_