/src/mozilla-central/dom/svg/SVGNumberList.h
Line | Count | Source (jump to first uncovered line) |
1 | | /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ |
2 | | /* vim: set ts=8 sts=2 et sw=2 tw=80: */ |
3 | | /* This Source Code Form is subject to the terms of the Mozilla Public |
4 | | * License, v. 2.0. If a copy of the MPL was not distributed with this |
5 | | * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ |
6 | | |
7 | | #ifndef MOZILLA_SVGNUMBERLIST_H__ |
8 | | #define MOZILLA_SVGNUMBERLIST_H__ |
9 | | |
10 | | #include "nsCOMPtr.h" |
11 | | #include "nsDebug.h" |
12 | | #include "nsIContent.h" |
13 | | #include "nsINode.h" |
14 | | #include "nsIWeakReferenceUtils.h" |
15 | | #include "nsSVGElement.h" |
16 | | #include "nsTArray.h" |
17 | | |
18 | | namespace mozilla { |
19 | | |
20 | | /** |
21 | | * ATTENTION! WARNING! WATCH OUT!! |
22 | | * |
23 | | * Consumers that modify objects of this type absolutely MUST keep the DOM |
24 | | * wrappers for those lists (if any) in sync!! That's why this class is so |
25 | | * locked down. |
26 | | * |
27 | | * The DOM wrapper class for this class is DOMSVGNumberList. |
28 | | */ |
29 | | class SVGNumberList |
30 | | { |
31 | | friend class SVGAnimatedNumberList; |
32 | | friend class DOMSVGNumberList; |
33 | | friend class DOMSVGNumber; |
34 | | |
35 | | public: |
36 | | |
37 | 0 | SVGNumberList(){} |
38 | 0 | ~SVGNumberList(){} |
39 | | |
40 | | // Only methods that don't make/permit modification to this list are public. |
41 | | // Only our friend classes can access methods that may change us. |
42 | | |
43 | | /// This may return an incomplete string on OOM, but that's acceptable. |
44 | | void GetValueAsString(nsAString& aValue) const; |
45 | | |
46 | 0 | bool IsEmpty() const { |
47 | 0 | return mNumbers.IsEmpty(); |
48 | 0 | } |
49 | | |
50 | 0 | uint32_t Length() const { |
51 | 0 | return mNumbers.Length(); |
52 | 0 | } |
53 | | |
54 | 0 | const float& operator[](uint32_t aIndex) const { |
55 | 0 | return mNumbers[aIndex]; |
56 | 0 | } |
57 | | |
58 | 0 | bool operator==(const SVGNumberList& rhs) const { |
59 | 0 | return mNumbers == rhs.mNumbers; |
60 | 0 | } |
61 | | |
62 | 0 | bool SetCapacity(uint32_t size) { |
63 | 0 | return mNumbers.SetCapacity(size, fallible); |
64 | 0 | } |
65 | | |
66 | 0 | void Compact() { |
67 | 0 | mNumbers.Compact(); |
68 | 0 | } |
69 | | |
70 | | // Access to methods that can modify objects of this type is deliberately |
71 | | // limited. This is to reduce the chances of someone modifying objects of |
72 | | // this type without taking the necessary steps to keep DOM wrappers in sync. |
73 | | // If you need wider access to these methods, consider adding a method to |
74 | | // SVGAnimatedNumberList and having that class act as an intermediary so it |
75 | | // can take care of keeping DOM wrappers in sync. |
76 | | |
77 | | protected: |
78 | | |
79 | | /** |
80 | | * This may fail on OOM if the internal capacity needs to be increased, in |
81 | | * which case the list will be left unmodified. |
82 | | */ |
83 | | nsresult CopyFrom(const SVGNumberList& rhs); |
84 | | |
85 | 0 | float& operator[](uint32_t aIndex) { |
86 | 0 | return mNumbers[aIndex]; |
87 | 0 | } |
88 | | |
89 | | /** |
90 | | * This may fail (return false) on OOM if the internal capacity is being |
91 | | * increased, in which case the list will be left unmodified. |
92 | | */ |
93 | 0 | bool SetLength(uint32_t aNumberOfItems) { |
94 | 0 | return mNumbers.SetLength(aNumberOfItems, fallible); |
95 | 0 | } |
96 | | |
97 | | private: |
98 | | |
99 | | // Marking the following private only serves to show which methods are only |
100 | | // used by our friend classes (as opposed to our subclasses) - it doesn't |
101 | | // really provide additional safety. |
102 | | |
103 | | nsresult SetValueFromString(const nsAString& aValue); |
104 | | |
105 | 0 | void Clear() { |
106 | 0 | mNumbers.Clear(); |
107 | 0 | } |
108 | | |
109 | 0 | bool InsertItem(uint32_t aIndex, const float &aNumber) { |
110 | 0 | if (aIndex >= mNumbers.Length()) { |
111 | 0 | aIndex = mNumbers.Length(); |
112 | 0 | } |
113 | 0 | return !!mNumbers.InsertElementAt(aIndex, aNumber, fallible); |
114 | 0 | } |
115 | | |
116 | 0 | void ReplaceItem(uint32_t aIndex, const float &aNumber) { |
117 | 0 | MOZ_ASSERT(aIndex < mNumbers.Length(), |
118 | 0 | "DOM wrapper caller should have raised INDEX_SIZE_ERR"); |
119 | 0 | mNumbers[aIndex] = aNumber; |
120 | 0 | } |
121 | | |
122 | 0 | void RemoveItem(uint32_t aIndex) { |
123 | 0 | MOZ_ASSERT(aIndex < mNumbers.Length(), |
124 | 0 | "DOM wrapper caller should have raised INDEX_SIZE_ERR"); |
125 | 0 | mNumbers.RemoveElementAt(aIndex); |
126 | 0 | } |
127 | | |
128 | 0 | bool AppendItem(float aNumber) { |
129 | 0 | return !!mNumbers.AppendElement(aNumber, fallible); |
130 | 0 | } |
131 | | |
132 | | protected: |
133 | | |
134 | | /* See SVGLengthList for the rationale for using FallibleTArray<float> instead |
135 | | * of FallibleTArray<float, 1>. |
136 | | */ |
137 | | FallibleTArray<float> mNumbers; |
138 | | }; |
139 | | |
140 | | |
141 | | /** |
142 | | * This SVGNumberList subclass is used by the SMIL code when a number list |
143 | | * is to be stored in an nsISMILValue instance. Since nsISMILValue objects may |
144 | | * be cached, it is necessary for us to hold a strong reference to our element |
145 | | * so that it doesn't disappear out from under us if, say, the element is |
146 | | * removed from the DOM tree. |
147 | | */ |
148 | | class SVGNumberListAndInfo : public SVGNumberList |
149 | | { |
150 | | public: |
151 | | |
152 | | SVGNumberListAndInfo() |
153 | | : mElement(nullptr) |
154 | 0 | {} |
155 | | |
156 | | explicit SVGNumberListAndInfo(nsSVGElement *aElement) |
157 | | : mElement(do_GetWeakReference(static_cast<nsINode*>(aElement))) |
158 | 0 | {} |
159 | | |
160 | 0 | void SetInfo(nsSVGElement *aElement) { |
161 | 0 | mElement = do_GetWeakReference(static_cast<nsINode*>(aElement)); |
162 | 0 | } |
163 | | |
164 | 0 | nsSVGElement* Element() const { |
165 | 0 | nsCOMPtr<nsIContent> e = do_QueryReferent(mElement); |
166 | 0 | return static_cast<nsSVGElement*>(e.get()); |
167 | 0 | } |
168 | | |
169 | 0 | nsresult CopyFrom(const SVGNumberListAndInfo& rhs) { |
170 | 0 | mElement = rhs.mElement; |
171 | 0 | return SVGNumberList::CopyFrom(rhs); |
172 | 0 | } |
173 | | |
174 | | // Instances of this special subclass do not have DOM wrappers that we need |
175 | | // to worry about keeping in sync, so it's safe to expose any hidden base |
176 | | // class methods required by the SMIL code, as we do below. |
177 | | |
178 | | /** |
179 | | * Exposed so that SVGNumberList baseVals can be copied to |
180 | | * SVGNumberListAndInfo objects. Note that callers should also call |
181 | | * SetInfo() when using this method! |
182 | | */ |
183 | 0 | nsresult CopyFrom(const SVGNumberList& rhs) { |
184 | 0 | return SVGNumberList::CopyFrom(rhs); |
185 | 0 | } |
186 | 0 | const float& operator[](uint32_t aIndex) const { |
187 | 0 | return SVGNumberList::operator[](aIndex); |
188 | 0 | } |
189 | 0 | float& operator[](uint32_t aIndex) { |
190 | 0 | return SVGNumberList::operator[](aIndex); |
191 | 0 | } |
192 | 0 | bool SetLength(uint32_t aNumberOfItems) { |
193 | 0 | return SVGNumberList::SetLength(aNumberOfItems); |
194 | 0 | } |
195 | | |
196 | | private: |
197 | | // We must keep a weak reference to our element because we may belong to a |
198 | | // cached baseVal nsSMILValue. See the comments starting at: |
199 | | // https://bugzilla.mozilla.org/show_bug.cgi?id=515116#c15 |
200 | | // See also https://bugzilla.mozilla.org/show_bug.cgi?id=653497 |
201 | | nsWeakPtr mElement; |
202 | | }; |
203 | | |
204 | | } // namespace mozilla |
205 | | |
206 | | #endif // MOZILLA_SVGNUMBERLIST_H__ |