cf9ec9e2ed2f9fef2bbd94ff65e0aae32a384102
[WebKit-https.git] / Source / WebCore / css / CSSSegmentedFontFace.cpp
1 /*
2  * Copyright (C) 2008 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #include "config.h"
27 #include "CSSSegmentedFontFace.h"
28
29 #include "CSSFontFace.h"
30 #include "CSSFontFaceSource.h"
31 #include "CSSFontSelector.h"
32 #include "Document.h"
33 #include "FontDescription.h"
34 #include "SegmentedFontData.h"
35 #include "SimpleFontData.h"
36
37 namespace WebCore {
38
39 CSSSegmentedFontFace::CSSSegmentedFontFace(CSSFontSelector* fontSelector)
40     : m_fontSelector(fontSelector)
41 {
42 }
43
44 CSSSegmentedFontFace::~CSSSegmentedFontFace()
45 {
46     pruneTable();
47     unsigned size = m_fontFaces.size();
48     for (unsigned i = 0; i < size; i++)
49         m_fontFaces[i]->removedFromSegmentedFontFace(this);
50 }
51
52 void CSSSegmentedFontFace::pruneTable()
53 {
54     // Make sure the glyph page tree prunes out all uses of this custom font.
55     if (m_fontDataTable.isEmpty())
56         return;
57
58     m_fontDataTable.clear();
59 }
60
61 bool CSSSegmentedFontFace::isValid() const
62 {
63     // Valid if at least one font face is valid.
64     unsigned size = m_fontFaces.size();
65     for (unsigned i = 0; i < size; i++) {
66         if (m_fontFaces[i]->isValid())
67             return true;
68     }
69     return false;
70 }
71
72 void CSSSegmentedFontFace::fontLoaded(CSSFontFace*)
73 {
74     pruneTable();
75 }
76
77 void CSSSegmentedFontFace::appendFontFace(PassRefPtr<CSSFontFace> fontFace)
78 {
79     pruneTable();
80     fontFace->addedToSegmentedFontFace(this);
81     m_fontFaces.append(fontFace);
82 }
83
84 static void appendFontDataWithInvalidUnicodeRangeIfLoading(SegmentedFontData* newFontData, const SimpleFontData* faceFontData, const Vector<CSSFontFace::UnicodeRange>& ranges)
85 {
86     if (faceFontData->isLoading()) {
87         newFontData->appendRange(FontDataRange(0, 0, faceFontData));
88         return;
89     }
90
91     unsigned numRanges = ranges.size();
92     if (!numRanges) {
93         newFontData->appendRange(FontDataRange(0, 0x7FFFFFFF, faceFontData));
94         return;
95     }
96
97     for (unsigned j = 0; j < numRanges; ++j)
98         newFontData->appendRange(FontDataRange(ranges[j].from(), ranges[j].to(), faceFontData));
99 }
100
101 FontData* CSSSegmentedFontFace::getFontData(const FontDescription& fontDescription)
102 {
103     if (!isValid())
104         return 0;
105
106     FontTraitsMask desiredTraitsMask = fontDescription.traitsMask();
107     unsigned hashKey = ((fontDescription.computedPixelSize() + 1) << (FontTraitsMaskWidth + 1)) | ((fontDescription.orientation() == Vertical ? 1 : 0) << FontTraitsMaskWidth) | desiredTraitsMask;
108
109     SegmentedFontData*& fontData = m_fontDataTable.add(hashKey, 0).iterator->second;
110     if (fontData)
111         return fontData;
112
113     OwnPtr<SegmentedFontData> newFontData = adoptPtr(new SegmentedFontData);
114
115     unsigned size = m_fontFaces.size();
116     for (unsigned i = 0; i < size; i++) {
117         if (!m_fontFaces[i]->isValid())
118             continue;
119         FontTraitsMask traitsMask = m_fontFaces[i]->traitsMask();
120         bool syntheticBold = !(traitsMask & (FontWeight600Mask | FontWeight700Mask | FontWeight800Mask | FontWeight900Mask)) && (desiredTraitsMask & (FontWeight600Mask | FontWeight700Mask | FontWeight800Mask | FontWeight900Mask));
121         bool syntheticItalic = !(traitsMask & FontStyleItalicMask) && (desiredTraitsMask & FontStyleItalicMask);
122         if (const SimpleFontData* faceFontData = m_fontFaces[i]->getFontData(fontDescription, syntheticBold, syntheticItalic)) {
123             ASSERT(!faceFontData->isSegmented());
124             appendFontDataWithInvalidUnicodeRangeIfLoading(newFontData.get(), faceFontData, m_fontFaces[i]->ranges());
125         }
126     }
127     if (newFontData->numRanges()) {
128         ASSERT(m_fontSelector->document());
129         if (Document* document = m_fontSelector->document()) {
130             fontData = newFontData.get();
131             document->registerCustomFont(newFontData.release());
132         }
133     }
134
135     return fontData;
136 }
137
138 }