Remove the SVG elements' attributes macros
[WebKit-https.git] / Source / WebCore / svg / SVGFEImageElement.cpp
1 /*
2  * Copyright (C) 2004, 2005, 2007 Nikolas Zimmermann <zimmermann@kde.org>
3  * Copyright (C) 2004, 2005 Rob Buis <buis@kde.org>
4  * Copyright (C) 2010 Dirk Schulze <krit@webkit.org>
5  * Copyright (C) 2018 Apple Inc. All rights reserved.
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Library General Public
9  * License as published by the Free Software Foundation; either
10  * version 2 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Library General Public License for more details.
16  *
17  * You should have received a copy of the GNU Library General Public License
18  * along with this library; see the file COPYING.LIB.  If not, write to
19  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
20  * Boston, MA 02110-1301, USA.
21  */
22
23 #include "config.h"
24 #include "SVGFEImageElement.h"
25
26 #include "CachedImage.h"
27 #include "CachedResourceLoader.h"
28 #include "CachedResourceRequest.h"
29 #include "Document.h"
30 #include "Image.h"
31 #include "RenderObject.h"
32 #include "RenderSVGResource.h"
33 #include "SVGNames.h"
34 #include "SVGPreserveAspectRatioValue.h"
35 #include "XLinkNames.h"
36 #include <wtf/IsoMallocInlines.h>
37
38 namespace WebCore {
39
40 WTF_MAKE_ISO_ALLOCATED_IMPL(SVGFEImageElement);
41
42 inline SVGFEImageElement::SVGFEImageElement(const QualifiedName& tagName, Document& document)
43     : SVGFilterPrimitiveStandardAttributes(tagName, document)
44     , SVGExternalResourcesRequired(this)
45     , SVGURIReference(this)
46 {
47     ASSERT(hasTagName(SVGNames::feImageTag));
48     registerAttributes();
49 }
50
51 Ref<SVGFEImageElement> SVGFEImageElement::create(const QualifiedName& tagName, Document& document)
52 {
53     return adoptRef(*new SVGFEImageElement(tagName, document));
54 }
55
56 SVGFEImageElement::~SVGFEImageElement()
57 {
58     clearResourceReferences();
59 }
60
61 bool SVGFEImageElement::hasSingleSecurityOrigin() const
62 {
63     if (!m_cachedImage)
64         return true;
65     auto* image = m_cachedImage->image();
66     return !image || image->hasSingleSecurityOrigin();
67 }
68
69 void SVGFEImageElement::clearResourceReferences()
70 {
71     if (m_cachedImage) {
72         m_cachedImage->removeClient(*this);
73         m_cachedImage = nullptr;
74     }
75
76     document().accessSVGExtensions().removeAllTargetReferencesForElement(this);
77 }
78
79 void SVGFEImageElement::requestImageResource()
80 {
81     ResourceLoaderOptions options = CachedResourceLoader::defaultCachedResourceOptions();
82     options.contentSecurityPolicyImposition = isInUserAgentShadowTree() ? ContentSecurityPolicyImposition::SkipPolicyCheck : ContentSecurityPolicyImposition::DoPolicyCheck;
83
84     CachedResourceRequest request(ResourceRequest(document().completeURL(href())), options);
85     request.setInitiator(*this);
86     m_cachedImage = document().cachedResourceLoader().requestImage(WTFMove(request)).value_or(nullptr);
87
88     if (m_cachedImage)
89         m_cachedImage->addClient(*this);
90 }
91
92 void SVGFEImageElement::buildPendingResource()
93 {
94     clearResourceReferences();
95     if (!isConnected())
96         return;
97
98     String id;
99     auto target = makeRefPtr(SVGURIReference::targetElementFromIRIString(href(), document(), &id));
100     if (!target) {
101         if (id.isEmpty())
102             requestImageResource();
103         else {
104             document().accessSVGExtensions().addPendingResource(id, this);
105             ASSERT(hasPendingResources());
106         }
107     } else if (target->isSVGElement()) {
108         // Register us with the target in the dependencies map. Any change of hrefElement
109         // that leads to relayout/repainting now informs us, so we can react to it.
110         document().accessSVGExtensions().addElementReferencingTarget(this, downcast<SVGElement>(target.get()));
111     }
112
113     invalidate();
114 }
115
116 void SVGFEImageElement::registerAttributes()
117 {
118     auto& registry = attributeRegistry();
119     if (!registry.isEmpty())
120         return;
121     registry.registerAttribute<SVGNames::preserveAspectRatioAttr, &SVGFEImageElement::m_preserveAspectRatio>();
122 }
123
124 void SVGFEImageElement::parseAttribute(const QualifiedName& name, const AtomicString& value)
125 {
126     if (name == SVGNames::preserveAspectRatioAttr) {
127         SVGPreserveAspectRatioValue preserveAspectRatio;
128         preserveAspectRatio.parse(value);
129         m_preserveAspectRatio.setValue(preserveAspectRatio);
130         return;
131     }
132
133     SVGFilterPrimitiveStandardAttributes::parseAttribute(name, value);
134     SVGURIReference::parseAttribute(name, value);
135     SVGExternalResourcesRequired::parseAttribute(name, value);
136 }
137
138 void SVGFEImageElement::svgAttributeChanged(const QualifiedName& attrName)
139 {
140     if (attrName == SVGNames::preserveAspectRatioAttr) {
141         InstanceInvalidationGuard guard(*this);
142         invalidate();
143         return;
144     }
145
146     if (SVGURIReference::isKnownAttribute(attrName)) {
147         InstanceInvalidationGuard guard(*this);
148         buildPendingResource();
149         return;
150     }
151
152     SVGFilterPrimitiveStandardAttributes::svgAttributeChanged(attrName);
153 }
154
155 Node::InsertedIntoAncestorResult SVGFEImageElement::insertedIntoAncestor(InsertionType insertionType, ContainerNode& parentOfInsertedTree)
156 {
157     SVGFilterPrimitiveStandardAttributes::insertedIntoAncestor(insertionType, parentOfInsertedTree);
158     return InsertedIntoAncestorResult::NeedsPostInsertionCallback;
159 }
160
161 void SVGFEImageElement::didFinishInsertingNode()
162 {
163     buildPendingResource();
164 }
165
166 void SVGFEImageElement::removedFromAncestor(RemovalType removalType, ContainerNode& oldParentOfRemovedTree)
167 {
168     SVGFilterPrimitiveStandardAttributes::removedFromAncestor(removalType, oldParentOfRemovedTree);
169     if (removalType.disconnectedFromDocument)
170         clearResourceReferences();
171 }
172
173 void SVGFEImageElement::notifyFinished(CachedResource&)
174 {
175     if (!isConnected())
176         return;
177
178     auto parent = makeRefPtr(parentElement());
179
180     if (!parent || !parent->hasTagName(SVGNames::filterTag))
181         return;
182
183     RenderElement* parentRenderer = parent->renderer();
184     if (!parentRenderer)
185         return;
186
187     RenderSVGResource::markForLayoutAndParentResourceInvalidation(*parentRenderer);
188 }
189
190 RefPtr<FilterEffect> SVGFEImageElement::build(SVGFilterBuilder*, Filter& filter)
191 {
192     if (m_cachedImage)
193         return FEImage::createWithImage(filter, m_cachedImage->imageForRenderer(renderer()), preserveAspectRatio());
194     return FEImage::createWithIRIReference(filter, document(), href(), preserveAspectRatio());
195 }
196
197 void SVGFEImageElement::addSubresourceAttributeURLs(ListHashSet<URL>& urls) const
198 {
199     SVGFilterPrimitiveStandardAttributes::addSubresourceAttributeURLs(urls);
200
201     addSubresourceURL(urls, document().completeURL(href()));
202 }
203
204 }