зеркало из https://github.com/mozilla/gecko-dev.git
316 строки
11 KiB
C++
316 строки
11 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/. */
|
|
|
|
#ifndef mozilla_dom_SVGSVGElement_h
|
|
#define mozilla_dom_SVGSVGElement_h
|
|
|
|
#include "SVGAnimatedEnumeration.h"
|
|
#include "SVGViewportElement.h"
|
|
|
|
nsresult NS_NewSVGSVGElement(
|
|
nsIContent** aResult, already_AddRefed<mozilla::dom::NodeInfo>&& aNodeInfo,
|
|
mozilla::dom::FromParser aFromParser);
|
|
|
|
namespace mozilla {
|
|
class AutoSVGViewHandler;
|
|
class SMILTimeContainer;
|
|
class SVGFragmentIdentifier;
|
|
class EventChainPreVisitor;
|
|
|
|
namespace dom {
|
|
class DOMSVGAngle;
|
|
class DOMSVGLength;
|
|
class DOMSVGNumber;
|
|
class SVGMatrix;
|
|
class SVGRect;
|
|
class SVGSVGElement;
|
|
|
|
// Stores svgView arguments of SVG fragment identifiers.
|
|
class SVGView {
|
|
public:
|
|
SVGView();
|
|
|
|
SVGAnimatedEnumeration mZoomAndPan;
|
|
SVGAnimatedViewBox mViewBox;
|
|
SVGAnimatedPreserveAspectRatio mPreserveAspectRatio;
|
|
UniquePtr<SVGAnimatedTransformList> mTransforms;
|
|
};
|
|
|
|
class DOMSVGTranslatePoint final : public nsISVGPoint {
|
|
public:
|
|
DOMSVGTranslatePoint(SVGPoint* aPt, SVGSVGElement* aElement)
|
|
: nsISVGPoint(aPt, true), mElement(aElement) {}
|
|
|
|
explicit DOMSVGTranslatePoint(DOMSVGTranslatePoint* aPt)
|
|
: nsISVGPoint(&aPt->mPt, true), mElement(aPt->mElement) {}
|
|
|
|
NS_DECL_ISUPPORTS_INHERITED
|
|
NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(DOMSVGTranslatePoint, nsISVGPoint)
|
|
|
|
virtual DOMSVGPoint* Copy() override;
|
|
|
|
// WebIDL
|
|
virtual float X() override { return mPt.GetX(); }
|
|
virtual float Y() override { return mPt.GetY(); }
|
|
virtual void SetX(float aValue, ErrorResult& rv) override;
|
|
virtual void SetY(float aValue, ErrorResult& rv) override;
|
|
virtual already_AddRefed<nsISVGPoint> MatrixTransform(
|
|
SVGMatrix& matrix) override;
|
|
|
|
virtual nsISupports* GetParentObject() override;
|
|
|
|
RefPtr<SVGSVGElement> mElement;
|
|
|
|
private:
|
|
~DOMSVGTranslatePoint() = default;
|
|
};
|
|
|
|
typedef SVGViewportElement SVGSVGElementBase;
|
|
|
|
class SVGSVGElement final : public SVGSVGElementBase {
|
|
friend class ::nsSVGOuterSVGFrame;
|
|
friend class mozilla::SVGFragmentIdentifier;
|
|
friend class mozilla::AutoSVGViewHandler;
|
|
friend class mozilla::AutoPreserveAspectRatioOverride;
|
|
friend class mozilla::dom::SVGView;
|
|
|
|
protected:
|
|
SVGSVGElement(already_AddRefed<mozilla::dom::NodeInfo>&& aNodeInfo,
|
|
FromParser aFromParser);
|
|
virtual JSObject* WrapNode(JSContext* aCx,
|
|
JS::Handle<JSObject*> aGivenProto) override;
|
|
|
|
friend nsresult(::NS_NewSVGSVGElement(
|
|
nsIContent** aResult,
|
|
already_AddRefed<mozilla::dom::NodeInfo>&& aNodeInfo,
|
|
mozilla::dom::FromParser aFromParser));
|
|
|
|
~SVGSVGElement() = default;
|
|
|
|
public:
|
|
// interfaces:
|
|
NS_DECL_ISUPPORTS_INHERITED
|
|
NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(SVGSVGElement, SVGSVGElementBase)
|
|
|
|
/**
|
|
* For use by zoom controls to allow currentScale, currentTranslate.x and
|
|
* currentTranslate.y to be set by a single operation that dispatches a
|
|
* single SVGZoom event (instead of one SVGZoom and two SVGScroll events).
|
|
*
|
|
* XXX SVGZoomEvent is no more, is this needed?
|
|
*/
|
|
MOZ_CAN_RUN_SCRIPT_BOUNDARY void SetCurrentScaleTranslate(float s, float x,
|
|
float y);
|
|
|
|
// nsIContent interface
|
|
void GetEventTargetParent(EventChainPreVisitor& aVisitor) override;
|
|
virtual bool IsEventAttributeNameInternal(nsAtom* aName) override;
|
|
|
|
// nsINode methods:
|
|
virtual nsresult Clone(dom::NodeInfo*, nsINode** aResult) const override;
|
|
|
|
// WebIDL
|
|
already_AddRefed<DOMSVGAnimatedLength> X();
|
|
already_AddRefed<DOMSVGAnimatedLength> Y();
|
|
already_AddRefed<DOMSVGAnimatedLength> Width();
|
|
already_AddRefed<DOMSVGAnimatedLength> Height();
|
|
bool UseCurrentView();
|
|
float CurrentScale();
|
|
void SetCurrentScale(float aCurrentScale);
|
|
already_AddRefed<nsISVGPoint> CurrentTranslate();
|
|
void SetCurrentTranslate(float x, float y);
|
|
uint32_t SuspendRedraw(uint32_t max_wait_milliseconds);
|
|
void UnsuspendRedraw(uint32_t suspend_handle_id);
|
|
void UnsuspendRedrawAll();
|
|
void ForceRedraw();
|
|
void PauseAnimations();
|
|
void UnpauseAnimations();
|
|
bool AnimationsPaused();
|
|
float GetCurrentTimeAsFloat();
|
|
void SetCurrentTime(float seconds);
|
|
void DeselectAll();
|
|
already_AddRefed<DOMSVGNumber> CreateSVGNumber();
|
|
already_AddRefed<DOMSVGLength> CreateSVGLength();
|
|
already_AddRefed<DOMSVGAngle> CreateSVGAngle();
|
|
already_AddRefed<nsISVGPoint> CreateSVGPoint();
|
|
already_AddRefed<SVGMatrix> CreateSVGMatrix();
|
|
already_AddRefed<SVGRect> CreateSVGRect();
|
|
already_AddRefed<DOMSVGTransform> CreateSVGTransform();
|
|
already_AddRefed<DOMSVGTransform> CreateSVGTransformFromMatrix(
|
|
SVGMatrix& matrix);
|
|
using nsINode::GetElementById; // This does what we want
|
|
uint16_t ZoomAndPan();
|
|
void SetZoomAndPan(uint16_t aZoomAndPan, ErrorResult& rv);
|
|
|
|
// SVGElement overrides
|
|
|
|
virtual nsresult BindToTree(BindContext&, nsINode& aParent) override;
|
|
virtual void UnbindFromTree(bool aNullParent) override;
|
|
virtual SVGAnimatedTransformList* GetAnimatedTransformList(
|
|
uint32_t aFlags = 0) override;
|
|
|
|
// SVGSVGElement methods:
|
|
|
|
// Returns true IFF our attributes are currently overridden by a <view>
|
|
// element and that element's ID matches the passed-in string.
|
|
bool IsOverriddenBy(const nsAString& aViewID) const {
|
|
return mCurrentViewID && mCurrentViewID->Equals(aViewID);
|
|
}
|
|
|
|
SMILTimeContainer* GetTimedDocumentRoot();
|
|
|
|
// public helpers:
|
|
|
|
/**
|
|
* Returns -1 if the width/height is a percentage, else returns the user unit
|
|
* length clamped to fit in a int32_t.
|
|
* XXX see bug 1112533 comment 3 - we should fix drawImage so that we can
|
|
* change these methods to make zero the error flag for percentages.
|
|
*/
|
|
int32_t GetIntrinsicWidth();
|
|
int32_t GetIntrinsicHeight();
|
|
|
|
// This services any pending notifications for the transform on on this root
|
|
// <svg> node needing to be recalculated. (Only applicable in
|
|
// SVG-as-an-image documents.)
|
|
virtual void FlushImageTransformInvalidation();
|
|
|
|
private:
|
|
// SVGViewportElement methods:
|
|
|
|
virtual SVGViewElement* GetCurrentViewElement() const;
|
|
virtual SVGPreserveAspectRatio GetPreserveAspectRatioWithOverride()
|
|
const override;
|
|
|
|
// implementation helpers:
|
|
|
|
/*
|
|
* While binding to the tree we need to determine if we will be the outermost
|
|
* <svg> element _before_ the children are bound (as they want to know what
|
|
* timed document root to register with) and therefore _before_ our parent is
|
|
* set (both actions are performed by Element::BindToTree) so we
|
|
* can't use GetOwnerSVGElement() as it relies on GetParent(). This code is
|
|
* basically a simplified version of GetOwnerSVGElement that uses the parent
|
|
* parameters passed in instead.
|
|
*
|
|
* FIXME(bug 1596690): GetOwnerSVGElement() uses the flattened tree parent
|
|
* rather than the DOM tree parent nowadays.
|
|
*/
|
|
bool WillBeOutermostSVG(nsINode& aParent) const;
|
|
|
|
// invalidate viewbox -> viewport xform & inform frames
|
|
void InvalidateTransformNotifyFrame();
|
|
|
|
// Methods for <image> elements to override my "PreserveAspectRatio" value.
|
|
// These are private so that only our friends
|
|
// (AutoPreserveAspectRatioOverride in particular) have access.
|
|
void SetImageOverridePreserveAspectRatio(const SVGPreserveAspectRatio& aPAR);
|
|
void ClearImageOverridePreserveAspectRatio();
|
|
|
|
// Set/Clear properties to hold old version of preserveAspectRatio
|
|
// when it's being overridden by an <image> element that we are inside of.
|
|
bool SetPreserveAspectRatioProperty(const SVGPreserveAspectRatio& aPAR);
|
|
const SVGPreserveAspectRatio* GetPreserveAspectRatioProperty() const;
|
|
bool ClearPreserveAspectRatioProperty();
|
|
|
|
virtual SVGPoint GetCurrentTranslate() const override {
|
|
return mCurrentTranslate;
|
|
}
|
|
virtual float GetCurrentScale() const override { return mCurrentScale; }
|
|
|
|
virtual const SVGAnimatedViewBox& GetViewBoxInternal() const override;
|
|
virtual SVGAnimatedTransformList* GetTransformInternal() const override;
|
|
|
|
virtual EnumAttributesInfo GetEnumInfo() override;
|
|
|
|
enum { ZOOMANDPAN };
|
|
SVGAnimatedEnumeration mEnumAttributes[1];
|
|
static SVGEnumMapping sZoomAndPanMap[];
|
|
static EnumInfo sEnumInfo[1];
|
|
|
|
// The time container for animations within this SVG document fragment. Set
|
|
// for all outermost <svg> elements (not nested <svg> elements).
|
|
UniquePtr<SMILTimeContainer> mTimedDocumentRoot;
|
|
|
|
// zoom and pan
|
|
// IMPORTANT: see the comment in RecordCurrentScaleTranslate before writing
|
|
// code to change any of these!
|
|
SVGPoint mCurrentTranslate;
|
|
float mCurrentScale;
|
|
SVGPoint mPreviousTranslate;
|
|
float mPreviousScale;
|
|
|
|
// For outermost <svg> elements created from parsing, animation is started by
|
|
// the onload event in accordance with the SVG spec, but for <svg> elements
|
|
// created by script or promoted from inner <svg> to outermost <svg> we need
|
|
// to manually kick off animation when they are bound to the tree.
|
|
bool mStartAnimationOnBindToTree;
|
|
|
|
bool mImageNeedsTransformInvalidation;
|
|
|
|
// mCurrentViewID and mSVGView are mutually exclusive; we can have
|
|
// at most one non-null.
|
|
UniquePtr<nsString> mCurrentViewID;
|
|
UniquePtr<SVGView> mSVGView;
|
|
};
|
|
|
|
} // namespace dom
|
|
|
|
class MOZ_RAII AutoSVGTimeSetRestore {
|
|
public:
|
|
AutoSVGTimeSetRestore(dom::SVGSVGElement* aRootElem,
|
|
float aFrameTime MOZ_GUARD_OBJECT_NOTIFIER_PARAM)
|
|
: mRootElem(aRootElem),
|
|
mOriginalTime(mRootElem->GetCurrentTimeAsFloat()) {
|
|
MOZ_GUARD_OBJECT_NOTIFIER_INIT;
|
|
mRootElem->SetCurrentTime(
|
|
aFrameTime); // Does nothing if there's no change.
|
|
}
|
|
|
|
~AutoSVGTimeSetRestore() { mRootElem->SetCurrentTime(mOriginalTime); }
|
|
|
|
private:
|
|
const RefPtr<dom::SVGSVGElement> mRootElem;
|
|
const float mOriginalTime;
|
|
MOZ_DECL_USE_GUARD_OBJECT_NOTIFIER
|
|
};
|
|
|
|
class MOZ_RAII AutoPreserveAspectRatioOverride {
|
|
public:
|
|
AutoPreserveAspectRatioOverride(const Maybe<SVGImageContext>& aSVGContext,
|
|
dom::SVGSVGElement* aRootElem
|
|
MOZ_GUARD_OBJECT_NOTIFIER_PARAM)
|
|
: mRootElem(aRootElem), mDidOverride(false) {
|
|
MOZ_GUARD_OBJECT_NOTIFIER_INIT;
|
|
MOZ_ASSERT(mRootElem, "No SVG/Symbol node to manage?");
|
|
|
|
if (aSVGContext.isSome() &&
|
|
aSVGContext->GetPreserveAspectRatio().isSome()) {
|
|
// Override preserveAspectRatio in our helper document.
|
|
// XXXdholbert We should technically be overriding the helper doc's clip
|
|
// and overflow properties here, too. See bug 272288 comment 36.
|
|
mRootElem->SetImageOverridePreserveAspectRatio(
|
|
*aSVGContext->GetPreserveAspectRatio());
|
|
mDidOverride = true;
|
|
}
|
|
}
|
|
|
|
~AutoPreserveAspectRatioOverride() {
|
|
if (mDidOverride) {
|
|
mRootElem->ClearImageOverridePreserveAspectRatio();
|
|
}
|
|
}
|
|
|
|
private:
|
|
const RefPtr<dom::SVGSVGElement> mRootElem;
|
|
bool mDidOverride;
|
|
MOZ_DECL_USE_GUARD_OBJECT_NOTIFIER
|
|
};
|
|
|
|
} // namespace mozilla
|
|
|
|
#endif // SVGSVGElement_h
|