Rename AtomicString to AtomString
[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-2019 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 "SVGElement.h"
28 #include <wtf/URL.h>
29 #include "XLinkNames.h"
30
31 namespace WebCore {
32
33 SVGURIReference::SVGURIReference(SVGElement* contextElement)
34     : m_href(SVGAnimatedString::create(contextElement))
35 {
36     static std::once_flag onceFlag;
37     std::call_once(onceFlag, [] {
38         PropertyRegistry::registerProperty<SVGNames::hrefAttr, &SVGURIReference::m_href>();
39         PropertyRegistry::registerProperty<XLinkNames::hrefAttr, &SVGURIReference::m_href>();
40     });
41 }
42
43 bool SVGURIReference::isKnownAttribute(const QualifiedName& attributeName)
44 {
45     return PropertyRegistry::isKnownAttribute(attributeName);
46 }
47
48 void SVGURIReference::parseAttribute(const QualifiedName& name, const AtomString& value)
49 {
50     if (isKnownAttribute(name))
51         m_href->setBaseValInternal(value);
52 }
53
54 String SVGURIReference::fragmentIdentifierFromIRIString(const String& url, const Document& document)
55 {
56     size_t start = url.find('#');
57     if (start == notFound)
58         return emptyString();
59
60     URL base = start ? URL(document.baseURL(), url.substring(0, start)) : document.baseURL();
61     String fragmentIdentifier = url.substring(start);
62     URL kurl(base, fragmentIdentifier);
63     if (equalIgnoringFragmentIdentifier(kurl, document.url()))
64         return fragmentIdentifier.substring(1);
65
66     // The url doesn't have any fragment identifier.
67     return emptyString();
68 }
69
70 auto SVGURIReference::targetElementFromIRIString(const String& iri, const TreeScope& treeScope, RefPtr<Document> externalDocument) -> TargetElementResult
71 {
72     // If there's no fragment identifier contained within the IRI string, we can't lookup an element.
73     size_t startOfFragmentIdentifier = iri.find('#');
74     if (startOfFragmentIdentifier == notFound)
75         return { };
76
77     // Exclude the '#' character when determining the fragmentIdentifier.
78     auto id = iri.substring(startOfFragmentIdentifier + 1);
79     if (id.isEmpty())
80         return { };
81
82     auto& document = treeScope.documentScope();
83     auto url = document.completeURL(iri);
84     if (externalDocument) {
85         // Enforce that the referenced url matches the url of the document that we've loaded for it!
86         ASSERT(equalIgnoringFragmentIdentifier(url, externalDocument->url()));
87         return { externalDocument->getElementById(id), WTFMove(id) };
88     }
89
90     // Exit early if the referenced url is external, and we have no externalDocument given.
91     if (isExternalURIReference(iri, document))
92         return { nullptr, WTFMove(id) };
93
94     return { treeScope.getElementById(id), WTFMove(id) };
95 }
96
97 }