Replace WTF::move with WTFMove
[WebKit-https.git] / Source / WebCore / svg / SVGUseElement.cpp
1 /*
2  * Copyright (C) 2004, 2005, 2006, 2007, 2008 Nikolas Zimmermann <zimmermann@kde.org>
3  * Copyright (C) 2004, 2005, 2006, 2007 Rob Buis <buis@kde.org>
4  * Copyright (C) Research In Motion Limited 2009-2010. All rights reserved.
5  * Copyright (C) 2011 Torch Mobile (Beijing) Co. Ltd. All rights reserved.
6  * Copyright (C) 2012 University of Szeged
7  * Copyright (C) 2012 Renata Hodovan <reni@webkit.org>
8  * Copyright (C) 2015 Apple Inc. All rights reserved.
9  *
10  * This library is free software; you can redistribute it and/or
11  * modify it under the terms of the GNU Library General Public
12  * License as published by the Free Software Foundation; either
13  * version 2 of the License, or (at your option) any later version.
14  *
15  * This library is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
18  * Library General Public License for more details.
19  *
20  * You should have received a copy of the GNU Library General Public License
21  * along with this library; see the file COPYING.LIB.  If not, write to
22  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
23  * Boston, MA 02110-1301, USA.
24  */
25
26 #include "config.h"
27 #include "SVGUseElement.h"
28
29 #include "CachedResourceLoader.h"
30 #include "CachedSVGDocument.h"
31 #include "ElementIterator.h"
32 #include "Event.h"
33 #include "RenderSVGResource.h"
34 #include "RenderSVGTransformableContainer.h"
35 #include "ShadowRoot.h"
36 #include "SVGGElement.h"
37 #include "SVGSVGElement.h"
38 #include "SVGSymbolElement.h"
39 #include "XLinkNames.h"
40
41 namespace WebCore {
42
43 DEFINE_ANIMATED_LENGTH(SVGUseElement, SVGNames::xAttr, X, x)
44 DEFINE_ANIMATED_LENGTH(SVGUseElement, SVGNames::yAttr, Y, y)
45 DEFINE_ANIMATED_LENGTH(SVGUseElement, SVGNames::widthAttr, Width, width)
46 DEFINE_ANIMATED_LENGTH(SVGUseElement, SVGNames::heightAttr, Height, height)
47 DEFINE_ANIMATED_STRING(SVGUseElement, XLinkNames::hrefAttr, Href, href)
48 DEFINE_ANIMATED_BOOLEAN(SVGUseElement, SVGNames::externalResourcesRequiredAttr, ExternalResourcesRequired, externalResourcesRequired)
49
50 BEGIN_REGISTER_ANIMATED_PROPERTIES(SVGUseElement)
51     REGISTER_LOCAL_ANIMATED_PROPERTY(x)
52     REGISTER_LOCAL_ANIMATED_PROPERTY(y)
53     REGISTER_LOCAL_ANIMATED_PROPERTY(width)
54     REGISTER_LOCAL_ANIMATED_PROPERTY(height)
55     REGISTER_LOCAL_ANIMATED_PROPERTY(href)
56     REGISTER_LOCAL_ANIMATED_PROPERTY(externalResourcesRequired)
57     REGISTER_PARENT_ANIMATED_PROPERTIES(SVGGraphicsElement)
58 END_REGISTER_ANIMATED_PROPERTIES
59
60 inline SVGUseElement::SVGUseElement(const QualifiedName& tagName, Document& document)
61     : SVGGraphicsElement(tagName, document)
62     , m_x(LengthModeWidth)
63     , m_y(LengthModeHeight)
64     , m_width(LengthModeWidth)
65     , m_height(LengthModeHeight)
66     , m_svgLoadEventTimer(*this, &SVGElement::svgLoadEventTimerFired)
67 {
68     ASSERT(hasCustomStyleResolveCallbacks());
69     ASSERT(hasTagName(SVGNames::useTag));
70     registerAnimatedPropertiesForSVGUseElement();
71 }
72
73 Ref<SVGUseElement> SVGUseElement::create(const QualifiedName& tagName, Document& document)
74 {
75     return adoptRef(*new SVGUseElement(tagName, document));
76 }
77
78 SVGUseElement::~SVGUseElement()
79 {
80     if (m_externalDocument)
81         m_externalDocument->removeClient(this);
82 }
83
84 void SVGUseElement::parseAttribute(const QualifiedName& name, const AtomicString& value)
85 {
86     SVGParsingError parseError = NoError;
87
88     if (name == SVGNames::xAttr)
89         setXBaseValue(SVGLength::construct(LengthModeWidth, value, parseError));
90     else if (name == SVGNames::yAttr)
91         setYBaseValue(SVGLength::construct(LengthModeHeight, value, parseError));
92     else if (name == SVGNames::widthAttr)
93         setWidthBaseValue(SVGLength::construct(LengthModeWidth, value, parseError, ForbidNegativeLengths));
94     else if (name == SVGNames::heightAttr)
95         setHeightBaseValue(SVGLength::construct(LengthModeHeight, value, parseError, ForbidNegativeLengths));
96
97     reportAttributeParsingError(parseError, name, value);
98
99     SVGExternalResourcesRequired::parseAttribute(name, value);
100     SVGGraphicsElement::parseAttribute(name, value);
101     SVGURIReference::parseAttribute(name, value);
102 }
103
104 Node::InsertionNotificationRequest SVGUseElement::insertedInto(ContainerNode& rootParent)
105 {
106     SVGGraphicsElement::insertedInto(rootParent);
107     if (inDocument()) {
108         SVGExternalResourcesRequired::insertedIntoDocument(this);
109         invalidateShadowTree();
110         updateExternalDocument();
111     }
112     return InsertionDone;
113 }
114
115 void SVGUseElement::removedFrom(ContainerNode& rootParent)
116 {
117     SVGGraphicsElement::removedFrom(rootParent);
118     clearShadowTree();
119     updateExternalDocument();
120 }
121
122 inline Document* SVGUseElement::externalDocument() const
123 {
124     return m_externalDocument ? m_externalDocument->document() : nullptr;
125 }
126
127 void SVGUseElement::transferSizeAttributesToTargetClone(SVGElement& shadowElement) const
128 {
129     // FIXME: The check for valueInSpecifiedUnits being non-zero below is a workaround for the fact
130     // that we currently have no good way to tell whether a particular animatable attribute is a value
131     // indicating it was unspecified, or specified but could not be parsed. Would be nice to fix that some day.
132     if (is<SVGSymbolElement>(shadowElement)) {
133         // Spec (<use> on <symbol>): This generated 'svg' will always have explicit values for attributes width and height.
134         // If attributes width and/or height are provided on the 'use' element, then these attributes
135         // will be transferred to the generated 'svg'. If attributes width and/or height are not specified,
136         // the generated 'svg' element will use values of 100% for these attributes.
137         shadowElement.setAttribute(SVGNames::widthAttr, (widthIsValid() && width().valueInSpecifiedUnits()) ? AtomicString(width().valueAsString()) : "100%");
138         shadowElement.setAttribute(SVGNames::heightAttr, (heightIsValid() && height().valueInSpecifiedUnits()) ? AtomicString(height().valueAsString()) : "100%");
139     } else if (is<SVGSVGElement>(shadowElement)) {
140         // Spec (<use> on <svg>): If attributes width and/or height are provided on the 'use' element, then these
141         // values will override the corresponding attributes on the 'svg' in the generated tree.
142         SVGElement* correspondingElement = shadowElement.correspondingElement();
143         shadowElement.setAttribute(SVGNames::widthAttr, (widthIsValid() && width().valueInSpecifiedUnits()) ? AtomicString(width().valueAsString()) : (correspondingElement ? correspondingElement->getAttribute(SVGNames::widthAttr) : nullAtom));
144         shadowElement.setAttribute(SVGNames::heightAttr, (heightIsValid() && height().valueInSpecifiedUnits()) ? AtomicString(height().valueAsString()) : (correspondingElement ? correspondingElement->getAttribute(SVGNames::heightAttr) : nullAtom));
145     }
146 }
147
148 void SVGUseElement::svgAttributeChanged(const QualifiedName& attrName)
149 {
150     InstanceInvalidationGuard guard(*this);
151
152     if (attrName == SVGNames::xAttr || attrName == SVGNames::yAttr || attrName == SVGNames::widthAttr || attrName == SVGNames::heightAttr) {
153         updateRelativeLengthsInformation();
154         if (attrName == SVGNames::widthAttr || attrName == SVGNames::heightAttr) {
155             // FIXME: It's unnecessarily inefficient to update both width and height each time either is changed.
156             if (auto* targetClone = this->targetClone())
157                 transferSizeAttributesToTargetClone(*targetClone);
158         }
159         if (auto* renderer = this->renderer())
160             RenderSVGResource::markForLayoutAndParentResourceInvalidation(*renderer);
161         return;
162     }
163
164     if (SVGExternalResourcesRequired::handleAttributeChange(this, attrName))
165         return;
166
167     if (SVGURIReference::isKnownAttribute(attrName)) {
168         updateExternalDocument();
169         invalidateShadowTree();
170         return;
171     }
172
173     if (SVGLangSpace::isKnownAttribute(attrName) || SVGExternalResourcesRequired::isKnownAttribute(attrName)) {
174         invalidateShadowTree();
175         return;
176     }
177
178     SVGGraphicsElement::svgAttributeChanged(attrName);
179 }
180
181 void SVGUseElement::willAttachRenderers()
182 {
183     if (m_shadowTreeNeedsUpdate)
184         updateShadowTree();
185     SVGGraphicsElement::willAttachRenderers();
186 }
187
188 static HashSet<AtomicString> createAllowedElementSet()
189 {
190     // Spec: "Any 'svg', 'symbol', 'g', graphics element or other 'use' is potentially a template object that can be re-used
191     // (i.e., "instanced") in the SVG document via a 'use' element."
192     // "Graphics Element" is defined as 'circle', 'ellipse', 'image', 'line', 'path', 'polygon', 'polyline', 'rect', 'text'
193     // Excluded are anything that is used by reference or that only make sense to appear once in a document.
194     using namespace SVGNames;
195     HashSet<AtomicString> set;
196     for (auto& tag : { aTag, circleTag, descTag, ellipseTag, gTag, imageTag, lineTag, metadataTag, pathTag, polygonTag, polylineTag, rectTag, svgTag, switchTag, symbolTag, textTag, textPathTag, titleTag, trefTag, tspanTag, useTag })
197         set.add(tag.localName());
198     return set;
199 }
200
201 static inline bool isDisallowedElement(const SVGElement& element)
202 {
203     static NeverDestroyed<HashSet<AtomicString>> set = createAllowedElementSet();
204     return !set.get().contains(element.localName());
205 }
206
207 static inline bool isDisallowedElement(const Element& element)
208 {
209     return !element.isSVGElement() || isDisallowedElement(downcast<SVGElement>(element));
210 }
211
212 void SVGUseElement::clearShadowTree()
213 {
214     if (auto* root = userAgentShadowRoot())
215         root->removeChildren();
216 }
217
218 void SVGUseElement::buildPendingResource()
219 {
220     invalidateShadowTree();
221 }
222
223 void SVGUseElement::updateShadowTree()
224 {
225     m_shadowTreeNeedsUpdate = false;
226
227     // FIXME: It's expensive to re-clone the entire tree every time. We should find a more efficient way to handle this.
228     clearShadowTree();
229
230     if (isInShadowTree() || !inDocument())
231         return;
232
233     String targetID;
234     auto* target = findTarget(&targetID);
235     if (!target) {
236         document().accessSVGExtensions().addPendingResource(targetID, this);
237         return;
238     }
239
240     cloneTarget(ensureUserAgentShadowRoot(), *target);
241     expandUseElementsInShadowTree();
242     expandSymbolElementsInShadowTree();
243     transferEventListenersToShadowTree();
244
245     updateRelativeLengthsInformation();
246
247     // When we invalidate the other shadow trees, it's important that we don't
248     // follow any cycles and invalidate ourselves. To avoid that, we temporarily
249     // set m_shadowTreeNeedsUpdate to true so invalidateShadowTree will
250     // quickly return and do nothing.
251     ASSERT(!m_shadowTreeNeedsUpdate);
252     m_shadowTreeNeedsUpdate = true;
253     invalidateDependentShadowTrees();
254     m_shadowTreeNeedsUpdate = false;
255 }
256
257 SVGElement* SVGUseElement::targetClone() const
258 {
259     auto* root = userAgentShadowRoot();
260     if (!root)
261         return nullptr;
262     return downcast<SVGElement>(root->firstChild());
263 }
264
265 RenderPtr<RenderElement> SVGUseElement::createElementRenderer(Ref<RenderStyle>&& style, const RenderTreePosition&)
266 {
267     return createRenderer<RenderSVGTransformableContainer>(*this, WTFMove(style));
268 }
269
270 static bool isDirectReference(const SVGElement& element)
271 {
272     using namespace SVGNames;
273     return element.hasTagName(circleTag)
274         || element.hasTagName(ellipseTag)
275         || element.hasTagName(pathTag)
276         || element.hasTagName(polygonTag)
277         || element.hasTagName(polylineTag)
278         || element.hasTagName(rectTag)
279         || element.hasTagName(textTag);
280 }
281
282 void SVGUseElement::toClipPath(Path& path)
283 {
284     ASSERT(path.isEmpty());
285
286     auto* targetClone = this->targetClone();
287     if (!is<SVGGraphicsElement>(targetClone))
288         return;
289
290     if (!isDirectReference(*targetClone)) {
291         // Spec: Indirect references are an error (14.3.5)
292         document().accessSVGExtensions().reportError(ASCIILiteral("Not allowed to use indirect reference in <clip-path>"));
293         return;
294     }
295
296     downcast<SVGGraphicsElement>(*targetClone).toClipPath(path);
297     SVGLengthContext lengthContext(this);
298     // FIXME: Find a way to do this without manual resolution of x/y here. It's potentially incorrect.
299     path.translate(FloatSize(x().value(lengthContext), y().value(lengthContext)));
300     path.transform(animatedLocalTransform());
301 }
302
303 RenderElement* SVGUseElement::rendererClipChild() const
304 {
305     auto* targetClone = this->targetClone();
306     if (!targetClone)
307         return nullptr;
308     if (!isDirectReference(*targetClone))
309         return nullptr;
310     return targetClone->renderer();
311 }
312
313 static void removeDisallowedElementsFromSubtree(SVGElement& subtree)
314 {
315     // Remove disallowed elements after the fact rather than not cloning them in the first place.
316     // This optimizes for the normal case where none of those elements are present.
317
318     // This function is used only on elements in subtrees that are not yet in documents, so
319     // mutation events are not a factor; there are no event listeners to handle those events.
320     // Assert that it's not in a document to make sure callers are still using it this way.
321     ASSERT(!subtree.inDocument());
322
323     Vector<Element*> disallowedElements;
324     auto descendants = descendantsOfType<Element>(subtree);
325     for (auto it = descendants.begin(), end = descendants.end(); it != end; ) {
326         if (isDisallowedElement(*it)) {
327             disallowedElements.append(&*it);
328             it.traverseNextSkippingChildren();
329             continue;
330         }
331         ++it;
332     }
333     for (auto* element : disallowedElements) {
334         for (auto& descendant : descendantsOfType<SVGElement>(*element))
335             descendant.setCorrespondingElement(nullptr);
336         element->parentNode()->removeChild(*element);
337     }
338 }
339
340 static void associateClonesWithOriginals(SVGElement& clone, SVGElement& original)
341 {
342     // This assertion checks that we don't call this with the arguments backwards.
343     // The clone is new and so it's not installed in a parent yet.
344     ASSERT(!clone.parentNode());
345
346     // The loop below works because we are associating these clones immediately, before
347     // doing transformations like removing disallowed elements or expanding elements.
348     clone.setCorrespondingElement(&original);
349     for (auto pair : descendantsOfType<SVGElement>(clone, original))
350         pair.first.setCorrespondingElement(&pair.second);
351 }
352
353 static void associateReplacementCloneWithOriginal(SVGElement& replacementClone, SVGElement& originalClone)
354 {
355     auto* correspondingElement = originalClone.correspondingElement();
356     ASSERT(correspondingElement);
357     originalClone.setCorrespondingElement(nullptr);
358     replacementClone.setCorrespondingElement(correspondingElement);
359 }
360
361 static void associateReplacementClonesWithOriginals(SVGElement& replacementClone, SVGElement& originalClone)
362 {
363     // This assertion checks that we don't call this with the arguments backwards.
364     // The replacement clone is new and so it's not installed in a parent yet.
365     ASSERT(!replacementClone.parentNode());
366
367     // The loop below works because we are associating these clones immediately, before
368     // doing transformations like removing disallowed elements or expanding elements.
369     associateReplacementCloneWithOriginal(replacementClone, originalClone);
370     for (auto pair : descendantsOfType<SVGElement>(replacementClone, originalClone))
371         associateReplacementCloneWithOriginal(pair.first, pair.second);
372 }
373
374 SVGElement* SVGUseElement::findTarget(String* targetID) const
375 {
376     auto* correspondingElement = this->correspondingElement();
377     auto& original = correspondingElement ? downcast<SVGUseElement>(*correspondingElement) : *this;
378
379     auto* targetCandidate = targetElementFromIRIString(original.href(), original.document(), targetID, original.externalDocument());
380     if (targetID && !targetID->isNull()) {
381         // If the reference is external, don't return the target ID to the caller.
382         // The caller would use the target ID to wait for a pending resource on the wrong document.
383         // If we ever want the change that and let the caller to wait on the external document,
384         // we should change this function so it returns the appropriate document to go with the ID.
385         if (isExternalURIReference(original.href(), original.document()))
386             *targetID = String();
387     }
388     if (!is<SVGElement>(targetCandidate))
389         return nullptr;
390     auto& target = downcast<SVGElement>(*targetCandidate);
391
392     if (!target.inDocument() || isDisallowedElement(target))
393         return nullptr;
394
395     // Reject any target that has already been cloned to create one of the ancestors of this element,
396     // already in the shadow tree. This is sufficient to prevent cycles.
397     if (correspondingElement) {
398         for (auto& ancestor : lineageOfType<SVGElement>(*this)) {
399             if (ancestor.correspondingElement() == &target)
400                 return nullptr;
401         }
402     }
403
404     return &target;
405 }
406
407 void SVGUseElement::cloneTarget(ContainerNode& container, SVGElement& target) const
408 {
409     Ref<SVGElement> targetClone = static_cast<SVGElement&>(target.cloneElementWithChildren(document()).get());
410     associateClonesWithOriginals(targetClone.get(), target);
411     removeDisallowedElementsFromSubtree(targetClone.get());
412     transferSizeAttributesToTargetClone(targetClone.get());
413     container.appendChild(WTFMove(targetClone));
414 }
415
416 static void cloneDataAndChildren(SVGElement& replacementClone, SVGElement& originalClone)
417 {
418     // This assertion checks that we don't call this with the arguments backwards.
419     // The replacement clone is new and so it's not installed in a parent yet.
420     ASSERT(!replacementClone.parentNode());
421
422     replacementClone.cloneDataFromElement(originalClone);
423     originalClone.cloneChildNodes(replacementClone);
424     associateReplacementClonesWithOriginals(replacementClone, originalClone);
425     removeDisallowedElementsFromSubtree(replacementClone);
426 }
427
428 void SVGUseElement::expandUseElementsInShadowTree() const
429 {
430     auto descendants = descendantsOfType<SVGUseElement>(*userAgentShadowRoot());
431     for (auto it = descendants.begin(), end = descendants.end(); it != end; ) {
432         SVGUseElement& originalClone = *it;
433         it = end; // Efficiently quiets assertions due to the outstanding iterator.
434
435         auto* target = originalClone.findTarget();
436
437         // Spec: In the generated content, the 'use' will be replaced by 'g', where all attributes from the
438         // 'use' element except for x, y, width, height and xlink:href are transferred to the generated 'g' element.
439
440         auto replacementClone = SVGGElement::create(document());
441         cloneDataAndChildren(replacementClone.get(), originalClone);
442
443         replacementClone->removeAttribute(SVGNames::xAttr);
444         replacementClone->removeAttribute(SVGNames::yAttr);
445         replacementClone->removeAttribute(SVGNames::widthAttr);
446         replacementClone->removeAttribute(SVGNames::heightAttr);
447         replacementClone->removeAttribute(XLinkNames::hrefAttr);
448
449         if (target)
450             originalClone.cloneTarget(replacementClone.get(), *target);
451
452         originalClone.parentNode()->replaceChild(replacementClone.copyRef(), originalClone);
453
454         // Resume iterating, starting just inside the replacement clone.
455         it = descendants.from(replacementClone.get());
456     }
457 }
458
459 void SVGUseElement::expandSymbolElementsInShadowTree() const
460 {
461     auto descendants = descendantsOfType<SVGSymbolElement>(*userAgentShadowRoot());
462     for (auto it = descendants.begin(), end = descendants.end(); it != end; ) {
463         SVGSymbolElement& originalClone = *it;
464         it = end; // Efficiently quiets assertions due to the outstanding iterator.
465
466         // Spec: The referenced 'symbol' and its contents are deep-cloned into the generated tree,
467         // with the exception that the 'symbol' is replaced by an 'svg'. This generated 'svg' will
468         // always have explicit values for attributes width and height. If attributes width and/or
469         // height are provided on the 'use' element, then these attributes will be transferred to
470         // the generated 'svg'. If attributes width and/or height are not specified, the generated
471         // 'svg' element will use values of 100% for these attributes.
472
473         auto replacementClone = SVGSVGElement::create(document());
474         cloneDataAndChildren(replacementClone.get(), originalClone);
475
476         originalClone.parentNode()->replaceChild(replacementClone.copyRef(), originalClone);
477
478         // Resume iterating, starting just inside the replacement clone.
479         it = descendants.from(replacementClone.get());
480     }
481 }
482
483 void SVGUseElement::transferEventListenersToShadowTree() const
484 {
485     for (auto& descendant : descendantsOfType<SVGElement>(*userAgentShadowRoot())) {
486         if (EventTargetData* data = descendant.correspondingElement()->eventTargetData())
487             data->eventListenerMap.copyEventListenersNotCreatedFromMarkupToTarget(&descendant);
488     }
489 }
490
491 void SVGUseElement::invalidateShadowTree()
492 {
493     if (m_shadowTreeNeedsUpdate)
494         return;
495     m_shadowTreeNeedsUpdate = true;
496     setNeedsStyleRecalc(ReconstructRenderTree);
497     invalidateDependentShadowTrees();
498 }
499
500 void SVGUseElement::invalidateDependentShadowTrees()
501 {
502     for (auto* instance : instances()) {
503         if (auto* element = instance->correspondingUseElement())
504             element->invalidateShadowTree();
505     }
506 }
507
508 bool SVGUseElement::selfHasRelativeLengths() const
509 {
510     if (x().isRelative() || y().isRelative() || width().isRelative() || height().isRelative())
511         return true;
512
513     auto* targetClone = this->targetClone();
514     return targetClone && targetClone->hasRelativeLengths();
515 }
516
517 void SVGUseElement::notifyFinished(CachedResource* resource)
518 {
519     invalidateShadowTree();
520     if (resource->errorOccurred())
521         dispatchEvent(Event::create(eventNames().errorEvent, false, false));
522     else if (!resource->wasCanceled())
523         SVGExternalResourcesRequired::dispatchLoadEvent(this);
524 }
525
526 void SVGUseElement::finishParsingChildren()
527 {
528     SVGGraphicsElement::finishParsingChildren();
529     SVGExternalResourcesRequired::finishParsingChildren();
530 }
531
532 void SVGUseElement::updateExternalDocument()
533 {
534     URL externalDocumentURL;
535     if (inDocument() && isExternalURIReference(href(), document())) {
536         externalDocumentURL = document().completeURL(href());
537         if (!externalDocumentURL.hasFragmentIdentifier())
538             externalDocumentURL = URL();
539     }
540
541     if (externalDocumentURL == (m_externalDocument ? m_externalDocument->url() : URL()))
542         return;
543
544     if (m_externalDocument)
545         m_externalDocument->removeClient(this);
546
547     if (externalDocumentURL.isNull())
548         m_externalDocument = nullptr;
549     else {
550         ResourceLoaderOptions options = CachedResourceLoader::defaultCachedResourceOptions();
551         options.setContentSecurityPolicyImposition(isInUserAgentShadowTree() ? ContentSecurityPolicyImposition::SkipPolicyCheck : ContentSecurityPolicyImposition::DoPolicyCheck);
552
553         CachedResourceRequest request { ResourceRequest { externalDocumentURL }, options };
554         request.setInitiator(this);
555         m_externalDocument = document().cachedResourceLoader().requestSVGDocument(request);
556         if (m_externalDocument)
557             m_externalDocument->addClient(this);
558     }
559
560     invalidateShadowTree();
561 }
562
563 bool SVGUseElement::isValid() const
564 {
565     return SVGTests::isValid();
566 }
567
568 bool SVGUseElement::haveLoadedRequiredResources()
569 {
570     return SVGExternalResourcesRequired::haveLoadedRequiredResources();
571 }
572
573 void SVGUseElement::setHaveFiredLoadEvent(bool haveFiredLoadEvent)
574 {
575     m_haveFiredLoadEvent = haveFiredLoadEvent;
576 }
577
578 bool SVGUseElement::haveFiredLoadEvent() const
579 {
580     return m_haveFiredLoadEvent;
581 }
582
583 Timer* SVGUseElement::svgLoadEventTimer()
584 {
585     return &m_svgLoadEventTimer;
586 }
587
588 }