4248af4ebb47494869f0e3cf8e0b84a27ea0953f
[WebKit-https.git] / Source / WebCore / loader / cache / CachedFont.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 COMPUTER, 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 COMPUTER, 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 "CachedFont.h"
29
30 #include "CachedFontClient.h"
31 #include "CachedResourceClientWalker.h"
32 #include "CachedResourceLoader.h"
33 #include "FontCustomPlatformData.h"
34 #include "FontPlatformData.h"
35 #include "MemoryCache.h"
36 #include "ResourceBuffer.h"
37 #include "TextResourceDecoder.h"
38 #include <wtf/Vector.h>
39
40 #if ENABLE(SVG_FONTS)
41 #include "NodeList.h"
42 #include "SVGDocument.h"
43 #include "SVGElement.h"
44 #include "SVGFontElement.h"
45 #include "SVGGElement.h"
46 #include "SVGNames.h"
47 #endif
48
49 namespace WebCore {
50
51 CachedFont::CachedFont(const ResourceRequest& resourceRequest)
52     : CachedResource(resourceRequest, FontResource)
53     , m_fontData(0)
54     , m_loadInitiated(false)
55     , m_hasCreatedFontData(false)
56 {
57 }
58
59 CachedFont::~CachedFont()
60 {
61     delete m_fontData;
62 }
63
64 void CachedFont::load(CachedResourceLoader*, const ResourceLoaderOptions& options)
65 {
66     // Don't load the file yet.  Wait for an access before triggering the load.
67     setLoading(true);
68     m_options = options;
69 }
70
71 void CachedFont::didAddClient(CachedResourceClient* c)
72 {
73     ASSERT(c->resourceClientType() == CachedFontClient::expectedType());
74     if (!isLoading())
75         static_cast<CachedFontClient*>(c)->fontLoaded(this);
76 }
77
78 void CachedFont::finishLoading(ResourceBuffer* data)
79 {
80     m_data = data;
81     setEncodedSize(m_data.get() ? m_data->size() : 0);
82     setLoading(false);
83     checkNotify();
84 }
85
86 void CachedFont::beginLoadIfNeeded(CachedResourceLoader* dl)
87 {
88     if (!m_loadInitiated) {
89         m_loadInitiated = true;
90         CachedResource::load(dl, m_options);
91     }
92 }
93
94 bool CachedFont::ensureCustomFontData()
95 {
96     if (!m_fontData && !errorOccurred() && !isLoading() && m_data) {
97         m_fontData = createFontCustomPlatformData(m_data.get()->sharedBuffer());
98         if (m_fontData)
99             m_hasCreatedFontData = true;
100         else
101             setStatus(DecodeError);
102     }
103     return m_fontData;
104 }
105
106 FontPlatformData CachedFont::platformDataFromCustomData(float size, bool bold, bool italic, FontOrientation orientation, FontWidthVariant widthVariant, FontRenderingMode renderingMode)
107 {
108 #if ENABLE(SVG_FONTS)
109     if (m_externalSVGDocument)
110         return FontPlatformData(size, bold, italic);
111 #endif
112     ASSERT(m_fontData);
113     return m_fontData->fontPlatformData(static_cast<int>(size), bold, italic, orientation, widthVariant, renderingMode);
114 }
115
116 #if ENABLE(SVG_FONTS)
117 bool CachedFont::ensureSVGFontData()
118 {
119     if (!m_externalSVGDocument && !errorOccurred() && !isLoading() && m_data) {
120         m_externalSVGDocument = SVGDocument::create(0, KURL());
121
122         RefPtr<TextResourceDecoder> decoder = TextResourceDecoder::create("application/xml");
123         String svgSource = decoder->decode(m_data->data(), m_data->size());
124         svgSource.append(decoder->flush());
125         
126         m_externalSVGDocument->setContent(svgSource);
127         
128         if (decoder->sawError())
129             m_externalSVGDocument = 0;
130     }
131
132     return m_externalSVGDocument;
133 }
134
135 SVGFontElement* CachedFont::getSVGFontById(const String& fontName) const
136 {
137     RefPtr<NodeList> list = m_externalSVGDocument->getElementsByTagNameNS(SVGNames::fontTag.namespaceURI(), SVGNames::fontTag.localName());
138     if (!list)
139         return 0;
140
141     unsigned listLength = list->length();
142     if (!listLength)
143         return 0;
144
145 #ifndef NDEBUG
146     for (unsigned i = 0; i < listLength; ++i) {
147         ASSERT(list->item(i));
148         ASSERT(isSVGFontElement(list->item(i)));
149     }
150 #endif
151
152     if (fontName.isEmpty())
153         return static_cast<SVGFontElement*>(list->item(0));
154
155     for (unsigned i = 0; i < listLength; ++i) {
156         SVGFontElement* element = static_cast<SVGFontElement*>(list->item(i));
157         if (element->getIdAttribute() == fontName)
158             return element;
159     }
160
161     return 0;
162 }
163 #endif
164
165 void CachedFont::allClientsRemoved()
166 {
167     if (m_fontData) {
168         delete m_fontData;
169         m_fontData = 0;
170     }
171 }
172
173 void CachedFont::checkNotify()
174 {
175     if (isLoading())
176         return;
177     
178     CachedResourceClientWalker<CachedFontClient> w(m_clients);
179     while (CachedFontClient* c = w.next())
180          c->fontLoaded(this);
181 }
182
183 bool CachedFont::mayTryReplaceEncodedData() const
184 {
185     // If the FontCustomPlatformData has ever been constructed then it still might be in use somewhere.
186     // That platform font object might directly reference the encoded data buffer behind this CachedFont,
187     // so replacing it is unsafe.
188
189     return !m_hasCreatedFontData;
190 }
191
192 }