Separate render tree updating from style resolve
[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 bool SVGUseElement::willRecalcStyle(Style::Change change)
182 {
183     // FIXME: Shadow tree should be updated before style recalc.
184     if (m_shadowTreeNeedsUpdate)
185         updateShadowTree();
186     return SVGGraphicsElement::willRecalcStyle(change);
187 }
188
189 static HashSet<AtomicString> createAllowedElementSet()
190 {
191     // Spec: "Any 'svg', 'symbol', 'g', graphics element or other 'use' is potentially a template object that can be re-used
192     // (i.e., "instanced") in the SVG document via a 'use' element."
193     // "Graphics Element" is defined as 'circle', 'ellipse', 'image', 'line', 'path', 'polygon', 'polyline', 'rect', 'text'
194     // Excluded are anything that is used by reference or that only make sense to appear once in a document.
195     using namespace SVGNames;
196     HashSet<AtomicString> set;
197     for (auto& tag : { aTag, circleTag, descTag, ellipseTag, gTag, imageTag, lineTag, metadataTag, pathTag, polygonTag, polylineTag, rectTag, svgTag, switchTag, symbolTag, textTag, textPathTag, titleTag, trefTag, tspanTag, useTag })
198         set.add(tag.localName());
199     return set;
200 }
201
202 static inline bool isDisallowedElement(const SVGElement& element)
203 {
204     static NeverDestroyed<HashSet<AtomicString>> set = createAllowedElementSet();
205     return !set.get().contains(element.localName());
206 }
207
208 static inline bool isDisallowedElement(const Element& element)
209 {
210     return !element.isSVGElement() || isDisallowedElement(downcast<SVGElement>(element));
211 }
212
213 void SVGUseElement::clearShadowTree()
214 {
215     if (auto* root = userAgentShadowRoot())
216         root->removeChildren();
217 }
218
219 void SVGUseElement::buildPendingResource()
220 {
221     invalidateShadowTree();
222 }
223
224 void SVGUseElement::updateShadowTree()
225 {
226     m_shadowTreeNeedsUpdate = false;
227
228     // FIXME: It's expensive to re-clone the entire tree every time. We should find a more efficient way to handle this.
229     clearShadowTree();
230
231     if (isInShadowTree() || !inDocument())
232         return;
233
234     String targetID;
235     auto* target = findTarget(&targetID);
236     if (!target) {
237         document().accessSVGExtensions().addPendingResource(targetID, this);
238         return;
239     }
240
241     cloneTarget(ensureUserAgentShadowRoot(), *target);
242     expandUseElementsInShadowTree();
243     expandSymbolElementsInShadowTree();
244     transferEventListenersToShadowTree();
245
246     updateRelativeLengthsInformation();
247
248     // When we invalidate the other shadow trees, it's important that we don't
249     // follow any cycles and invalidate ourselves. To avoid that, we temporarily
250     // set m_shadowTreeNeedsUpdate to true so invalidateShadowTree will
251     // quickly return and do nothing.
252     ASSERT(!m_shadowTreeNeedsUpdate);
253     m_shadowTreeNeedsUpdate = true;
254     invalidateDependentShadowTrees();
255     m_shadowTreeNeedsUpdate = false;
256 }
257
258 SVGElement* SVGUseElement::targetClone() const
259 {
260     auto* root = userAgentShadowRoot();
261     if (!root)
262         return nullptr;
263     return downcast<SVGElement>(root->firstChild());
264 }
265
266 RenderPtr<RenderElement> SVGUseElement::createElementRenderer(Ref<RenderStyle>&& style, const RenderTreePosition&)
267 {
268     return createRenderer<RenderSVGTransformableContainer>(*this, WTFMove(style));
269 }
270
271 static bool isDirectReference(const SVGElement& element)
272 {
273     using namespace SVGNames;
274     return element.hasTagName(circleTag)
275         || element.hasTagName(ellipseTag)
276         || element.hasTagName(pathTag)
277         || element.hasTagName(polygonTag)
278         || element.hasTagName(polylineTag)
279         || element.hasTagName(rectTag)
280         || element.hasTagName(textTag);
281 }
282
283 void SVGUseElement::toClipPath(Path& path)
284 {
285     ASSERT(path.isEmpty());
286
287     auto* targetClone = this->targetClone();
288     if (!is<SVGGraphicsElement>(targetClone))
289         return;
290
291     if (!isDirectReference(*targetClone)) {
292         // Spec: Indirect references are an error (14.3.5)
293         document().accessSVGExtensions().reportError(ASCIILiteral("Not allowed to use indirect reference in <clip-path>"));
294         return;
295     }
296
297     downcast<SVGGraphicsElement>(*targetClone).toClipPath(path);
298     SVGLengthContext lengthContext(this);
299     // FIXME: Find a way to do this without manual resolution of x/y here. It's potentially incorrect.
300     path.translate(FloatSize(x().value(lengthContext), y().value(lengthContext)));
301     path.transform(animatedLocalTransform());
302 }
303
304 RenderElement* SVGUseElement::rendererClipChild() const
305 {
306     auto* targetClone = this->targetClone();
307     if (!targetClone)
308         return nullptr;
309     if (!isDirectReference(*targetClone))
310         return nullptr;
311     return targetClone->renderer();
312 }
313
314 static inline void disassociateAndRemoveClones(const Vector<Element*>& clones)
315 {
316     for (auto& clone : clones) {
317         for (auto& descendant : descendantsOfType<SVGElement>(*clone))
318             descendant.setCorrespondingElement(nullptr);
319         clone->parentNode()->removeChild(*clone);
320     }
321 }
322
323 static void removeDisallowedElementsFromSubtree(SVGElement& subtree)
324 {
325     // Remove disallowed elements after the fact rather than not cloning them in the first place.
326     // This optimizes for the normal case where none of those elements are present.
327
328     // This function is used only on elements in subtrees that are not yet in documents, so
329     // mutation events are not a factor; there are no event listeners to handle those events.
330     // Assert that it's not in a document to make sure callers are still using it this way.
331     ASSERT(!subtree.inDocument());
332
333     Vector<Element*> disallowedElements;
334     auto descendants = descendantsOfType<Element>(subtree);
335     for (auto it = descendants.begin(), end = descendants.end(); it != end; ) {
336         if (isDisallowedElement(*it)) {
337             disallowedElements.append(&*it);
338             it.traverseNextSkippingChildren();
339             continue;
340         }
341         ++it;
342     }
343
344     disassociateAndRemoveClones(disallowedElements);
345 }
346
347 static void removeSymbolElementsFromSubtree(SVGElement& subtree)
348 {
349     // Symbol elements inside the subtree should not be cloned for two reasons: 1) They are invisible and
350     // don't need to be cloned to get correct rendering. 2) expandSymbolElementsInShadowTree will turn them
351     // into <svg> elements, which is correct for symbol elements directly referenced by use elements,
352     // but incorrect for ones that just happen to be in a subtree.
353     Vector<Element*> symbolElements;
354     for (auto& descendant : descendantsOfType<SVGSymbolElement>(subtree))
355         symbolElements.append(&descendant);
356     disassociateAndRemoveClones(symbolElements);
357 }
358
359 static void associateClonesWithOriginals(SVGElement& clone, SVGElement& original)
360 {
361     // This assertion checks that we don't call this with the arguments backwards.
362     // The clone is new and so it's not installed in a parent yet.
363     ASSERT(!clone.parentNode());
364
365     // The loop below works because we are associating these clones immediately, before
366     // doing transformations like removing disallowed elements or expanding elements.
367     clone.setCorrespondingElement(&original);
368     for (auto pair : descendantsOfType<SVGElement>(clone, original))
369         pair.first.setCorrespondingElement(&pair.second);
370 }
371
372 static void associateReplacementCloneWithOriginal(SVGElement& replacementClone, SVGElement& originalClone)
373 {
374     auto* correspondingElement = originalClone.correspondingElement();
375     ASSERT(correspondingElement);
376     originalClone.setCorrespondingElement(nullptr);
377     replacementClone.setCorrespondingElement(correspondingElement);
378 }
379
380 static void associateReplacementClonesWithOriginals(SVGElement& replacementClone, SVGElement& originalClone)
381 {
382     // This assertion checks that we don't call this with the arguments backwards.
383     // The replacement clone is new and so it's not installed in a parent yet.
384     ASSERT(!replacementClone.parentNode());
385
386     // The loop below works because we are associating these clones immediately, before
387     // doing transformations like removing disallowed elements or expanding elements.
388     associateReplacementCloneWithOriginal(replacementClone, originalClone);
389     for (auto pair : descendantsOfType<SVGElement>(replacementClone, originalClone))
390         associateReplacementCloneWithOriginal(pair.first, pair.second);
391 }
392
393 SVGElement* SVGUseElement::findTarget(String* targetID) const
394 {
395     auto* correspondingElement = this->correspondingElement();
396     auto& original = correspondingElement ? downcast<SVGUseElement>(*correspondingElement) : *this;
397
398     auto* targetCandidate = targetElementFromIRIString(original.href(), original.document(), targetID, original.externalDocument());
399     if (targetID && !targetID->isNull()) {
400         // If the reference is external, don't return the target ID to the caller.
401         // The caller would use the target ID to wait for a pending resource on the wrong document.
402         // If we ever want the change that and let the caller to wait on the external document,
403         // we should change this function so it returns the appropriate document to go with the ID.
404         if (isExternalURIReference(original.href(), original.document()))
405             *targetID = String();
406     }
407     if (!is<SVGElement>(targetCandidate))
408         return nullptr;
409     auto& target = downcast<SVGElement>(*targetCandidate);
410
411     if (!target.inDocument() || isDisallowedElement(target))
412         return nullptr;
413
414     // Reject any target that has already been cloned to create one of the ancestors of this element,
415     // already in the shadow tree. This is sufficient to prevent cycles.
416     if (correspondingElement) {
417         for (auto& ancestor : lineageOfType<SVGElement>(*this)) {
418             if (ancestor.correspondingElement() == &target)
419                 return nullptr;
420         }
421     }
422
423     return &target;
424 }
425
426 void SVGUseElement::cloneTarget(ContainerNode& container, SVGElement& target) const
427 {
428     Ref<SVGElement> targetClone = static_cast<SVGElement&>(target.cloneElementWithChildren(document()).get());
429     associateClonesWithOriginals(targetClone.get(), target);
430     removeDisallowedElementsFromSubtree(targetClone.get());
431     removeSymbolElementsFromSubtree(targetClone.get());
432     transferSizeAttributesToTargetClone(targetClone.get());
433     container.appendChild(WTFMove(targetClone));
434 }
435
436 static void cloneDataAndChildren(SVGElement& replacementClone, SVGElement& originalClone)
437 {
438     // This assertion checks that we don't call this with the arguments backwards.
439     // The replacement clone is new and so it's not installed in a parent yet.
440     ASSERT(!replacementClone.parentNode());
441
442     replacementClone.cloneDataFromElement(originalClone);
443     originalClone.cloneChildNodes(replacementClone);
444     associateReplacementClonesWithOriginals(replacementClone, originalClone);
445     removeDisallowedElementsFromSubtree(replacementClone);
446 }
447
448 void SVGUseElement::expandUseElementsInShadowTree() const
449 {
450     auto descendants = descendantsOfType<SVGUseElement>(*userAgentShadowRoot());
451     for (auto it = descendants.begin(), end = descendants.end(); it != end; ) {
452         SVGUseElement& originalClone = *it;
453         it = end; // Efficiently quiets assertions due to the outstanding iterator.
454
455         auto* target = originalClone.findTarget();
456
457         // Spec: In the generated content, the 'use' will be replaced by 'g', where all attributes from the
458         // 'use' element except for x, y, width, height and xlink:href are transferred to the generated 'g' element.
459
460         auto replacementClone = SVGGElement::create(document());
461         cloneDataAndChildren(replacementClone.get(), originalClone);
462
463         replacementClone->removeAttribute(SVGNames::xAttr);
464         replacementClone->removeAttribute(SVGNames::yAttr);
465         replacementClone->removeAttribute(SVGNames::widthAttr);
466         replacementClone->removeAttribute(SVGNames::heightAttr);
467         replacementClone->removeAttribute(XLinkNames::hrefAttr);
468
469         if (target)
470             originalClone.cloneTarget(replacementClone.get(), *target);
471
472         originalClone.parentNode()->replaceChild(replacementClone.copyRef(), originalClone);
473
474         // Resume iterating, starting just inside the replacement clone.
475         it = descendants.from(replacementClone.get());
476     }
477 }
478
479 void SVGUseElement::expandSymbolElementsInShadowTree() const
480 {
481     auto descendants = descendantsOfType<SVGSymbolElement>(*userAgentShadowRoot());
482     for (auto it = descendants.begin(), end = descendants.end(); it != end; ) {
483         SVGSymbolElement& originalClone = *it;
484         it = end; // Efficiently quiets assertions due to the outstanding iterator.
485
486         // Spec: The referenced 'symbol' and its contents are deep-cloned into the generated tree,
487         // with the exception that the 'symbol' is replaced by an 'svg'. This generated 'svg' will
488         // always have explicit values for attributes width and height. If attributes width and/or
489         // height are provided on the 'use' element, then these attributes will be transferred to
490         // the generated 'svg'. If attributes width and/or height are not specified, the generated
491         // 'svg' element will use values of 100% for these attributes.
492
493         auto replacementClone = SVGSVGElement::create(document());
494         cloneDataAndChildren(replacementClone.get(), originalClone);
495
496         originalClone.parentNode()->replaceChild(replacementClone.copyRef(), originalClone);
497
498         // Resume iterating, starting just inside the replacement clone.
499         it = descendants.from(replacementClone.get());
500     }
501 }
502
503 void SVGUseElement::transferEventListenersToShadowTree() const
504 {
505     for (auto& descendant : descendantsOfType<SVGElement>(*userAgentShadowRoot())) {
506         if (EventTargetData* data = descendant.correspondingElement()->eventTargetData())
507             data->eventListenerMap.copyEventListenersNotCreatedFromMarkupToTarget(&descendant);
508     }
509 }
510
511 void SVGUseElement::invalidateShadowTree()
512 {
513     if (m_shadowTreeNeedsUpdate)
514         return;
515     m_shadowTreeNeedsUpdate = true;
516     setNeedsStyleRecalc(ReconstructRenderTree);
517     invalidateDependentShadowTrees();
518 }
519
520 void SVGUseElement::invalidateDependentShadowTrees()
521 {
522     for (auto* instance : instances()) {
523         if (auto* element = instance->correspondingUseElement())
524             element->invalidateShadowTree();
525     }
526 }
527
528 bool SVGUseElement::selfHasRelativeLengths() const
529 {
530     if (x().isRelative() || y().isRelative() || width().isRelative() || height().isRelative())
531         return true;
532
533     auto* targetClone = this->targetClone();
534     return targetClone && targetClone->hasRelativeLengths();
535 }
536
537 void SVGUseElement::notifyFinished(CachedResource* resource)
538 {
539     invalidateShadowTree();
540     if (resource->errorOccurred())
541         dispatchEvent(Event::create(eventNames().errorEvent, false, false));
542     else if (!resource->wasCanceled())
543         SVGExternalResourcesRequired::dispatchLoadEvent(this);
544 }
545
546 void SVGUseElement::finishParsingChildren()
547 {
548     SVGGraphicsElement::finishParsingChildren();
549     SVGExternalResourcesRequired::finishParsingChildren();
550 }
551
552 void SVGUseElement::updateExternalDocument()
553 {
554     URL externalDocumentURL;
555     if (inDocument() && isExternalURIReference(href(), document())) {
556         externalDocumentURL = document().completeURL(href());
557         if (!externalDocumentURL.hasFragmentIdentifier())
558             externalDocumentURL = URL();
559     }
560
561     if (externalDocumentURL == (m_externalDocument ? m_externalDocument->url() : URL()))
562         return;
563
564     if (m_externalDocument)
565         m_externalDocument->removeClient(this);
566
567     if (externalDocumentURL.isNull())
568         m_externalDocument = nullptr;
569     else {
570         ResourceLoaderOptions options = CachedResourceLoader::defaultCachedResourceOptions();
571         options.setContentSecurityPolicyImposition(isInUserAgentShadowTree() ? ContentSecurityPolicyImposition::SkipPolicyCheck : ContentSecurityPolicyImposition::DoPolicyCheck);
572
573         CachedResourceRequest request { ResourceRequest { externalDocumentURL }, options };
574         request.setInitiator(this);
575         m_externalDocument = document().cachedResourceLoader().requestSVGDocument(request);
576         if (m_externalDocument)
577             m_externalDocument->addClient(this);
578     }
579
580     invalidateShadowTree();
581 }
582
583 bool SVGUseElement::isValid() const
584 {
585     return SVGTests::isValid();
586 }
587
588 bool SVGUseElement::haveLoadedRequiredResources()
589 {
590     return SVGExternalResourcesRequired::haveLoadedRequiredResources();
591 }
592
593 void SVGUseElement::setHaveFiredLoadEvent(bool haveFiredLoadEvent)
594 {
595     m_haveFiredLoadEvent = haveFiredLoadEvent;
596 }
597
598 bool SVGUseElement::haveFiredLoadEvent() const
599 {
600     return m_haveFiredLoadEvent;
601 }
602
603 Timer* SVGUseElement::svgLoadEventTimer()
604 {
605     return &m_svgLoadEventTimer;
606 }
607
608 }