Converting time, angle and frequency units in CSS calc() function
[WebKit-https.git] / Source / WebCore / loader / cache / CachedSVGFont.cpp
1 /*
2  * Copyright (C) 2006, 2007, 2008 Apple Inc. All rights reserved.
3  * Copyright (C) 2009 Torch Mobile, Inc.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
15  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
17  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
18  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
19  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
20  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
21  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
22  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
24  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
25  */
26
27 #include "config.h"
28 #include "CachedSVGFont.h"
29
30 #if ENABLE(SVG_FONTS)
31
32 #include "FontDescription.h"
33 #include "FontPlatformData.h"
34 #include "SVGDocument.h"
35 #include "SVGFontData.h"
36 #include "SVGFontElement.h"
37 #include "SVGFontFaceElement.h"
38 #include "SharedBuffer.h"
39 #include "TextResourceDecoder.h"
40 #include "TypedElementDescendantIterator.h"
41
42 #if ENABLE(SVG_OTF_CONVERTER)
43 #include "SVGToOTFFontConversion.h"
44 #endif
45
46 namespace WebCore {
47
48 CachedSVGFont::CachedSVGFont(const ResourceRequest& resourceRequest, SessionID sessionID)
49     : CachedFont(resourceRequest, sessionID, SVGFontResource)
50     , m_externalSVGFontElement(nullptr)
51 {
52 }
53
54 PassRefPtr<SimpleFontData> CachedSVGFont::getFontData(const FontDescription& fontDescription, const AtomicString& remoteURI, bool syntheticBold, bool syntheticItalic, bool externalSVG)
55 {
56 #if !ENABLE(SVG_OTF_CONVERTER)
57     if (!externalSVG)
58 #endif
59         return CachedFont::getFontData(fontDescription, remoteURI, syntheticBold, syntheticItalic, externalSVG);
60
61     if (SVGFontFaceElement* firstFontFace = this->firstFontFace(remoteURI))
62         return SimpleFontData::create(std::make_unique<SVGFontData>(firstFontFace), fontDescription.computedPixelSize(), syntheticBold, syntheticItalic);
63     return nullptr;
64 }
65
66 FontPlatformData CachedSVGFont::platformDataFromCustomData(float size, bool bold, bool italic, FontOrientation orientation, FontWidthVariant widthVariant, FontRenderingMode renderingMode)
67 {
68     if (m_externalSVGDocument)
69         return FontPlatformData(size, bold, italic);
70     return CachedFont::platformDataFromCustomData(size, bold, italic, orientation, widthVariant, renderingMode);
71 }
72
73 bool CachedSVGFont::ensureCustomFontData(bool externalSVG, const AtomicString& remoteURI)
74 {
75     if (!externalSVG)
76         return CachedFont::ensureCustomFontData(externalSVG, remoteURI);
77
78     if (!m_externalSVGDocument && !errorOccurred() && !isLoading() && m_data) {
79         m_externalSVGDocument = SVGDocument::create(nullptr, URL());
80         RefPtr<TextResourceDecoder> decoder = TextResourceDecoder::create("application/xml");
81         m_externalSVGDocument->setContent(decoder->decodeAndFlush(m_data->data(), m_data->size()));
82         if (decoder->sawError())
83             m_externalSVGDocument = nullptr;
84 #if ENABLE(SVG_OTF_CONVERTER)
85         firstFontFace(remoteURI); // Sets m_externalSVGFontElement
86         if (m_externalSVGFontElement) {
87             Vector<char> convertedFont = convertSVGToOTFFont(*m_externalSVGFontElement);
88             return CachedFont::ensureCustomFontData(SharedBuffer::adoptVector(convertedFont));
89         }
90 #endif
91     }
92     return m_externalSVGDocument;
93 }
94
95 SVGFontElement* CachedSVGFont::getSVGFontById(const String& fontName) const
96 {
97     auto elements = descendantsOfType<SVGFontElement>(*m_externalSVGDocument);
98
99     if (fontName.isEmpty())
100         return elements.first();
101
102     for (auto& element : elements) {
103         if (element.getIdAttribute() == fontName)
104             return &element;
105     }
106     return nullptr;
107 }
108
109 SVGFontFaceElement* CachedSVGFont::firstFontFace(const AtomicString& remoteURI)
110 {
111     if (!m_externalSVGFontElement) {
112         String fragmentIdentifier;
113         size_t start = remoteURI.find('#');
114         if (start != notFound)
115             fragmentIdentifier = remoteURI.string().substring(start + 1);
116         m_externalSVGFontElement = getSVGFontById(fragmentIdentifier);
117     }
118
119     if (!m_externalSVGFontElement)
120         return nullptr;
121
122     if (auto* firstFontFace = childrenOfType<SVGFontFaceElement>(*m_externalSVGFontElement).first())
123         return firstFontFace;
124     return nullptr;
125 }
126
127 }
128
129 #endif