1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
|
/* -*- 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_SVGMPATHELEMENT_H_
#define DOM_SVG_SVGMPATHELEMENT_H_
#include "mozilla/dom/IDTracker.h"
#include "mozilla/dom/SVGElement.h"
#include "nsStubMutationObserver.h"
#include "SVGAnimatedString.h"
nsresult NS_NewSVGMPathElement(
nsIContent** aResult, already_AddRefed<mozilla::dom::NodeInfo>&& aNodeInfo);
namespace mozilla::dom {
class SVGGeometryElement;
using SVGMPathElementBase = SVGElement;
class SVGMPathElement final : public SVGMPathElementBase,
public nsStubMutationObserver {
protected:
friend nsresult(::NS_NewSVGMPathElement(
nsIContent** aResult,
already_AddRefed<mozilla::dom::NodeInfo>&& aNodeInfo));
explicit SVGMPathElement(
already_AddRefed<mozilla::dom::NodeInfo>&& aNodeInfo);
~SVGMPathElement();
JSObject* WrapNode(JSContext* aCx,
JS::Handle<JSObject*> aGivenProto) override;
public:
// interfaces:
NS_DECL_ISUPPORTS_INHERITED
NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(SVGMPathElement, SVGMPathElementBase)
NS_DECL_NSIMUTATIONOBSERVER_ATTRIBUTECHANGED
// nsIContent interface
nsresult Clone(dom::NodeInfo*, nsINode** aResult) const override;
nsresult BindToTree(BindContext&, nsINode& aParent) override;
void UnbindFromTree(bool aNullParent) override;
// Element specializations
void AfterSetAttr(int32_t aNamespaceID, nsAtom* aName,
const nsAttrValue* aValue, const nsAttrValue* aOldValue,
nsIPrincipal* aMaybeScriptedPrincipal,
bool aNotify) override;
// Public helper method: If our xlink:href attribute links to a Shape
// element, this method returns a pointer to that element. Otherwise,
// this returns nullptr.
SVGGeometryElement* GetReferencedPath();
// WebIDL
already_AddRefed<DOMSVGAnimatedString> Href();
protected:
/**
* Helper that provides a reference to the 'path' element with the ID that is
* referenced by the 'mpath' element's 'href' attribute, and that will
* invalidate the parent of the 'mpath' and update mutation observers to the
* new path element if the element that that ID identifies changes to a
* different element (or none).
*/
class PathElementTracker final : public IDTracker {
public:
explicit PathElementTracker(SVGMPathElement* aMpathElement)
: mMpathElement(aMpathElement) {}
protected:
// We need to be notified when target changes, in order to request a sample
// (which will clear animation effects that used the old target-path
// and recompute the animation effects using the new target-path).
void ElementChanged(Element* aFrom, Element* aTo) override {
IDTracker::ElementChanged(aFrom, aTo);
if (aFrom) {
aFrom->RemoveMutationObserver(mMpathElement);
}
if (aTo) {
aTo->AddMutationObserver(mMpathElement);
}
mMpathElement->NotifyParentOfMpathChange(mMpathElement->GetParent());
}
// We need to override IsPersistent to get persistent tracking (beyond the
// first time the target changes)
bool IsPersistent() override { return true; }
private:
SVGMPathElement* const mMpathElement;
};
StringAttributesInfo GetStringInfo() override;
void UpdateHrefTarget(nsIContent* aParent, const nsAString& aHrefStr);
void UnlinkHrefTarget(bool aNotifyParent);
void NotifyParentOfMpathChange(nsIContent* aParent);
enum { HREF, XLINK_HREF };
SVGAnimatedString mStringAttributes[2];
static StringInfo sStringInfo[2];
PathElementTracker mPathTracker;
};
} // namespace mozilla::dom
#endif // DOM_SVG_SVGMPATHELEMENT_H_
|