Unreviewed, rolling out r205344.
[WebKit-https.git] / Source / WebCore / css / CSSCursorImageValue.cpp
1 /*
2  * Copyright (C) 2006 Rob Buis <buis@kde.org>
3  *           (C) 2008 Nikolas Zimmermann <zimmermann@kde.org>
4  * Copyright (C) 2008 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 "CSSCursorImageValue.h"
24
25 #include "CSSImageSetValue.h"
26 #include "CSSImageValue.h"
27 #include "CachedImage.h"
28 #include "CachedResourceLoader.h"
29 #include "StyleCachedImage.h"
30 #include "StyleImage.h"
31 #include "SVGCursorElement.h"
32 #include "SVGLengthContext.h"
33 #include "SVGNames.h"
34 #include "SVGURIReference.h"
35 #include "TreeScope.h"
36 #include <wtf/MathExtras.h>
37 #include <wtf/text/StringBuilder.h>
38 #include <wtf/text/WTFString.h>
39
40 namespace WebCore {
41
42 CSSCursorImageValue::CSSCursorImageValue(Ref<CSSValue>&& imageValue, bool hasHotSpot, const IntPoint& hotSpot)
43     : CSSValue(CursorImageClass)
44     , m_imageValue(WTFMove(imageValue))
45     , m_hasHotSpot(hasHotSpot)
46     , m_hotSpot(hotSpot)
47 {
48     if (is<CSSImageValue>(m_imageValue.get()))
49         m_originalURL = { ParsedURLString, downcast<CSSImageValue>(m_imageValue.get()).url() };
50 }
51
52 CSSCursorImageValue::~CSSCursorImageValue()
53 {
54     for (auto* element : m_cursorElements)
55         element->removeClient(*this);
56 }
57
58 String CSSCursorImageValue::customCSSText() const
59 {
60     StringBuilder result;
61     result.append(m_imageValue.get().cssText());
62     if (m_hasHotSpot) {
63         result.append(' ');
64         result.appendNumber(m_hotSpot.x());
65         result.append(' ');
66         result.appendNumber(m_hotSpot.y());
67     }
68     return result.toString();
69 }
70
71 SVGCursorElement* CSSCursorImageValue::updateCursorElement(const Document& document)
72 {
73     if (!m_originalURL.hasFragmentIdentifier())
74         return nullptr;
75
76     auto* element = SVGURIReference::targetElementFromIRIString(m_originalURL, document);
77     if (!is<SVGCursorElement>(element))
78         return nullptr;
79
80     auto& cursorElement = downcast<SVGCursorElement>(*element);
81     if (m_cursorElements.add(&cursorElement).isNewEntry) {
82         cursorElementChanged(cursorElement);
83         cursorElement.addClient(*this);
84     }
85     return &cursorElement;
86 }
87
88 void CSSCursorImageValue::cursorElementRemoved(SVGCursorElement& cursorElement)
89 {
90     m_cursorElements.remove(&cursorElement);
91 }
92
93 void CSSCursorImageValue::cursorElementChanged(SVGCursorElement& cursorElement)
94 {
95     // FIXME: This will override hot spot specified in CSS, which is probably incorrect.
96     SVGLengthContext lengthContext(nullptr);
97     m_hasHotSpot = true;
98     float x = std::round(cursorElement.x().value(lengthContext));
99     m_hotSpot.setX(static_cast<int>(x));
100
101     float y = std::round(cursorElement.y().value(lengthContext));
102     m_hotSpot.setY(static_cast<int>(y));
103 }
104
105 void CSSCursorImageValue::loadImage(CachedResourceLoader& loader, const ResourceLoaderOptions& options)
106 {
107     if (is<CSSImageSetValue>(m_imageValue.get())) {
108         downcast<CSSImageSetValue>(m_imageValue.get()).loadBestFitImage(loader, options);
109         return;
110     }
111
112     downcast<CSSImageValue>(m_imageValue.get()).loadImage(loader, options);
113 }
114
115 StyleCachedImage& CSSCursorImageValue::styleImage(const Document& document)
116 {
117     // Need to delegate completely so that changes in device scale factor can be handled appropriately.
118     StyleCachedImage* styleImage;
119     if (is<CSSImageSetValue>(m_imageValue.get()))
120         styleImage = &downcast<CSSImageSetValue>(m_imageValue.get()).styleImage(document);
121     else {
122         if (auto* cursorElement = updateCursorElement(document)) {
123             if (cursorElement->href() != downcast<CSSImageValue>(m_imageValue.get()).url())
124                 m_imageValue = CSSImageValue::create(cursorElement->href());
125         }
126         styleImage = &downcast<CSSImageValue>(m_imageValue.get()).styleImage();
127     }
128     styleImage->setCSSValue(*this);
129
130     return *styleImage;
131 }
132
133 bool CSSCursorImageValue::equals(const CSSCursorImageValue& other) const
134 {
135     return m_hasHotSpot ? other.m_hasHotSpot && m_hotSpot == other.m_hotSpot : !other.m_hasHotSpot
136         && compareCSSValue(m_imageValue, other.m_imageValue);
137 }
138
139 } // namespace WebCore