Move URL from WebCore to WTF
[WebKit-https.git] / Source / WebCore / svg / SVGURIReference.cpp
1 /*
2  * Copyright (C) 2004, 2005, 2008 Nikolas Zimmermann <zimmermann@kde.org>
3  * Copyright (C) 2004, 2005, 2006 Rob Buis <buis@kde.org>
4  * Copyright (C) 2018 Apple Inc. All rights reserved.
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Library General Public
8  * License as published by the Free Software Foundation; either
9  * version 2 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Library General Public License for more details.
15  *
16  * You should have received a copy of the GNU Library General Public License
17  * along with this library; see the file COPYING.LIB.  If not, write to
18  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19  * Boston, MA 02110-1301, USA.
20  */
21
22 #include "config.h"
23 #include "SVGURIReference.h"
24
25 #include "Document.h"
26 #include "Element.h"
27 #include "SVGAttributeOwnerProxy.h"
28 #include <wtf/URL.h>
29 #include "XLinkNames.h"
30
31 namespace WebCore {
32
33 SVGURIReference::SVGURIReference(SVGElement* contextElement)
34     : m_attributeOwnerProxy(std::make_unique<AttributeOwnerProxy>(*this, *contextElement))
35 {
36     registerAttributes();
37 }
38
39 void SVGURIReference::registerAttributes()
40 {
41     auto& registry = attributeRegistry();
42     if (!registry.isEmpty())
43         return;
44     registry.registerAttribute<SVGNames::hrefAttr, &SVGURIReference::m_href>();
45     registry.registerAttribute<XLinkNames::hrefAttr, &SVGURIReference::m_href>();
46 }
47
48 SVGURIReference::AttributeRegistry& SVGURIReference::attributeRegistry()
49 {
50     return AttributeOwnerProxy::attributeRegistry();
51 }
52
53 bool SVGURIReference::isKnownAttribute(const QualifiedName& attributeName)
54 {
55     return AttributeOwnerProxy::isKnownAttribute(attributeName);
56 }
57
58 void SVGURIReference::parseAttribute(const QualifiedName& name, const AtomicString& value)
59 {
60     if (isKnownAttribute(name))
61         m_href.setValue(value);
62 }
63
64 const String& SVGURIReference::href() const
65 {
66     return m_href.currentValue(*m_attributeOwnerProxy);
67 }
68
69 RefPtr<SVGAnimatedString> SVGURIReference::hrefAnimated()
70 {
71     return m_href.animatedProperty(*m_attributeOwnerProxy);
72 }
73
74 String SVGURIReference::fragmentIdentifierFromIRIString(const String& url, const Document& document)
75 {
76     size_t start = url.find('#');
77     if (start == notFound)
78         return emptyString();
79
80     URL base = start ? URL(document.baseURL(), url.substring(0, start)) : document.baseURL();
81     String fragmentIdentifier = url.substring(start);
82     URL kurl(base, fragmentIdentifier);
83     if (equalIgnoringFragmentIdentifier(kurl, document.url()))
84         return fragmentIdentifier.substring(1);
85
86     // The url doesn't have any fragment identifier.
87     return emptyString();
88 }
89
90 auto SVGURIReference::targetElementFromIRIString(const String& iri, const TreeScope& treeScope, RefPtr<Document> externalDocument) -> TargetElementResult
91 {
92     // If there's no fragment identifier contained within the IRI string, we can't lookup an element.
93     size_t startOfFragmentIdentifier = iri.find('#');
94     if (startOfFragmentIdentifier == notFound)
95         return { };
96
97     // Exclude the '#' character when determining the fragmentIdentifier.
98     auto id = iri.substring(startOfFragmentIdentifier + 1);
99     if (id.isEmpty())
100         return { };
101
102     auto& document = treeScope.documentScope();
103     auto url = document.completeURL(iri);
104     if (externalDocument) {
105         // Enforce that the referenced url matches the url of the document that we've loaded for it!
106         ASSERT(equalIgnoringFragmentIdentifier(url, externalDocument->url()));
107         return { externalDocument->getElementById(id), WTFMove(id) };
108     }
109
110     // Exit early if the referenced url is external, and we have no externalDocument given.
111     if (isExternalURIReference(iri, document))
112         return { nullptr, WTFMove(id) };
113
114     return { treeScope.getElementById(id), WTFMove(id) };
115 }
116
117 }