| OLD | NEW |
| 1 /* | 1 /* |
| 2 * Copyright (C) 2005, 2008 Apple Computer, Inc. All rights reserved. | 2 * Copyright (C) 2005, 2008 Apple Computer, Inc. All rights reserved. |
| 3 * Copyright (C) 2006 Alexey Proskuryakov | 3 * Copyright (C) 2006 Alexey Proskuryakov |
| 4 * | 4 * |
| 5 * Redistribution and use in source and binary forms, with or without | 5 * Redistribution and use in source and binary forms, with or without |
| 6 * modification, are permitted provided that the following conditions | 6 * modification, are permitted provided that the following conditions |
| 7 * are met: | 7 * are met: |
| 8 * | 8 * |
| 9 * 1. Redistributions of source code must retain the above copyright | 9 * 1. Redistributions of source code must retain the above copyright |
| 10 * notice, this list of conditions and the following disclaimer. | 10 * notice, this list of conditions and the following disclaimer. |
| (...skipping 34 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 45 SimpleFontData::SimpleFontData(const FontPlatformData& f, bool customFont, bool
loading, SVGFontData* svgFontData) | 45 SimpleFontData::SimpleFontData(const FontPlatformData& f, bool customFont, bool
loading, SVGFontData* svgFontData) |
| 46 : m_unitsPerEm(defaultUnitsPerEm) | 46 : m_unitsPerEm(defaultUnitsPerEm) |
| 47 , m_font(f) | 47 , m_font(f) |
| 48 , m_treatAsFixedPitch(false) | 48 , m_treatAsFixedPitch(false) |
| 49 #if ENABLE(SVG_FONTS) | 49 #if ENABLE(SVG_FONTS) |
| 50 , m_svgFontData(svgFontData) | 50 , m_svgFontData(svgFontData) |
| 51 #endif | 51 #endif |
| 52 , m_isCustomFont(customFont) | 52 , m_isCustomFont(customFont) |
| 53 , m_isLoading(loading) | 53 , m_isLoading(loading) |
| 54 , m_smallCapsFontData(0) | 54 , m_smallCapsFontData(0) |
| 55 , m_zeroWidthFontData(new ZeroWidthFontData()) | |
| 56 , m_cjkWidthFontData(new CJKWidthFontData()) | |
| 57 { | 55 { |
| 58 #if ENABLE(SVG_FONTS) | 56 #if ENABLE(SVG_FONTS) |
| 59 if (SVGFontFaceElement* svgFontFaceElement = svgFontData ? svgFontData->svgF
ontFaceElement() : 0) { | 57 if (SVGFontFaceElement* svgFontFaceElement = svgFontData ? svgFontData->svgF
ontFaceElement() : 0) { |
| 60 m_unitsPerEm = svgFontFaceElement->unitsPerEm(); | 58 m_unitsPerEm = svgFontFaceElement->unitsPerEm(); |
| 61 | 59 |
| 62 double scale = f.size(); | 60 double scale = f.size(); |
| 63 if (m_unitsPerEm) | 61 if (m_unitsPerEm) |
| 64 scale /= m_unitsPerEm; | 62 scale /= m_unitsPerEm; |
| 65 | 63 |
| 66 m_ascent = static_cast<int>(svgFontFaceElement->ascent() * scale); | 64 m_ascent = static_cast<int>(svgFontFaceElement->ascent() * scale); |
| 67 m_descent = static_cast<int>(svgFontFaceElement->descent() * scale); | 65 m_descent = static_cast<int>(svgFontFaceElement->descent() * scale); |
| 68 // TODO(ojan): What should SVG fonts use for avgCharWidth/maxCharWidth? | 66 // TODO(ojan): What should SVG fonts use for avgCharWidth/maxCharWidth? |
| 69 // This is currently only used in RenderTextControl.cpp for the width | 67 // This is currently only used in RenderTextControl.cpp for the width |
| 70 // of textareas with a non-fixed width. | 68 // of textareas with a non-fixed width. |
| 71 m_avgCharWidth = 0; | 69 m_avgCharWidth = 0; |
| 72 m_maxCharWidth = 0; | 70 m_maxCharWidth = 0; |
| 73 m_xHeight = static_cast<int>(svgFontFaceElement->xHeight() * scale); | 71 m_xHeight = static_cast<int>(svgFontFaceElement->xHeight() * scale); |
| 74 m_lineGap = 0.1f * f.size(); | 72 m_lineGap = 0.1f * f.size(); |
| 75 m_lineSpacing = m_ascent + m_descent + m_lineGap; | 73 m_lineSpacing = m_ascent + m_descent + m_lineGap; |
| 76 | 74 |
| 77 m_spaceGlyph = 0; | 75 m_spaceGlyph = 0; |
| 78 m_spaceWidth = 0; | 76 m_spaceWidth = 0; |
| 79 m_adjustedSpaceWidth = 0; | 77 m_adjustedSpaceWidth = 0; |
| 80 determinePitch(); | 78 determinePitch(); |
| 81 m_missingGlyphData.fontData = this; | 79 m_missingGlyphData.fontData = this; |
| 82 m_missingGlyphData.glyph = 0; | 80 m_missingGlyphData.glyph = 0; |
| 83 m_zeroWidthFontData->init(this); | |
| 84 m_cjkWidthFontData->init(this); | |
| 85 return; | 81 return; |
| 86 } | 82 } |
| 87 #endif | 83 #endif |
| 88 | 84 |
| 89 platformInit(); | 85 platformInit(); |
| 90 platformGlyphInit(); | 86 platformGlyphInit(); |
| 91 } | 87 } |
| 92 | 88 |
| 93 #if !PLATFORM(QT) | 89 #if !PLATFORM(QT) |
| 94 void SimpleFontData::platformGlyphInit() | 90 void SimpleFontData::platformGlyphInit() |
| 95 { | 91 { |
| 96 GlyphPage* glyphPageZero = GlyphPageTreeNode::getRootChild(this, 0)->page(); | 92 GlyphPage* glyphPageZero = GlyphPageTreeNode::getRootChild(this, 0)->page(); |
| 97 if (!glyphPageZero) { | 93 if (!glyphPageZero) { |
| 98 LOG_ERROR("Failed to get glyph page zero."); | 94 LOG_ERROR("Failed to get glyph page zero."); |
| 99 m_spaceGlyph = 0; | 95 m_spaceGlyph = 0; |
| 100 m_spaceWidth = 0; | 96 m_spaceWidth = 0; |
| 101 m_adjustedSpaceWidth = 0; | 97 m_adjustedSpaceWidth = 0; |
| 102 determinePitch(); | 98 determinePitch(); |
| 103 m_missingGlyphData.fontData = this; | 99 m_missingGlyphData.fontData = this; |
| 104 m_missingGlyphData.glyph = 0; | 100 m_missingGlyphData.glyph = 0; |
| 105 m_zeroWidthFontData->init(this); | |
| 106 m_cjkWidthFontData->init(this); | |
| 107 return; | 101 return; |
| 108 } | 102 } |
| 109 | 103 |
| 110 // Nasty hack to determine if we should round or ceil space widths. | 104 // Nasty hack to determine if we should round or ceil space widths. |
| 111 // If the font is monospace or fake monospace we ceil to ensure that | 105 // If the font is monospace or fake monospace we ceil to ensure that |
| 112 // every character and the space are the same width. Otherwise we round. | 106 // every character and the space are the same width. Otherwise we round. |
| 113 static const UChar32 space_char = ' '; | 107 static const UChar32 space_char = ' '; |
| 114 m_spaceGlyph = glyphPageZero->glyphDataForCharacter(space_char).glyph; | 108 m_spaceGlyph = glyphPageZero->glyphDataForCharacter(space_char).glyph; |
| 115 float width = widthForGlyph(m_spaceGlyph); | 109 float width = widthForGlyph(m_spaceGlyph); |
| 116 m_spaceWidth = width; | 110 m_spaceWidth = width; |
| 117 determinePitch(); | 111 determinePitch(); |
| 118 m_adjustedSpaceWidth = m_treatAsFixedPitch ? ceilf(width) : roundf(width); | 112 m_adjustedSpaceWidth = m_treatAsFixedPitch ? ceilf(width) : roundf(width); |
| 119 | 113 |
| 120 // TODO(dglazkov): Investigate and implement across platforms, if needed | |
| 121 #if PLATFORM(WIN_OS) | |
| 122 // ZERO WIDTH SPACES are explicitly mapped to share the glyph | |
| 123 // with SPACE (with width adjusted to 0) during GlyphPage::fill | |
| 124 // This is currently only implemented for Windows port. The FontData | |
| 125 // remapping may very well be needed for other platforms. | |
| 126 #else | |
| 127 // Force the glyph for ZERO WIDTH SPACE to have zero width, unless it is sha
red with SPACE. | 114 // Force the glyph for ZERO WIDTH SPACE to have zero width, unless it is sha
red with SPACE. |
| 128 // Helvetica is an example of a non-zero width ZERO WIDTH SPACE glyph. | 115 // Helvetica is an example of a non-zero width ZERO WIDTH SPACE glyph. |
| 129 // See <http://bugs.webkit.org/show_bug.cgi?id=13178> | 116 // See <http://bugs.webkit.org/show_bug.cgi?id=13178> |
| 130 // Ask for the glyph for 0 to avoid paging in ZERO WIDTH SPACE. Control char
acters, including 0, | 117 // Ask for the glyph for 0 to avoid paging in ZERO WIDTH SPACE. Control char
acters, including 0, |
| 131 // are mapped to the ZERO WIDTH SPACE glyph. | 118 // are mapped to the ZERO WIDTH SPACE glyph. |
| 132 Glyph zeroWidthSpaceGlyph = glyphPageZero->glyphDataForCharacter(0).glyph; | 119 Glyph zeroWidthSpaceGlyph = glyphPageZero->glyphDataForCharacter(0).glyph; |
| 133 if (zeroWidthSpaceGlyph) { | 120 if (zeroWidthSpaceGlyph) { |
| 134 if (zeroWidthSpaceGlyph != m_spaceGlyph) | 121 if (zeroWidthSpaceGlyph != m_spaceGlyph) |
| 135 m_glyphToWidthMap.setWidthForGlyph(zeroWidthSpaceGlyph, 0); | 122 m_glyphToWidthMap.setWidthForGlyph(zeroWidthSpaceGlyph, 0); |
| 136 else | 123 else |
| 137 LOG_ERROR("Font maps SPACE and ZERO WIDTH SPACE to the same glyph. G
lyph width not overridden."); | 124 LOG_ERROR("Font maps SPACE and ZERO WIDTH SPACE to the same glyph. G
lyph width not overridden."); |
| 138 } | 125 } |
| 139 #endif | |
| 140 | 126 |
| 141 m_missingGlyphData.fontData = this; | 127 m_missingGlyphData.fontData = this; |
| 142 m_missingGlyphData.glyph = 0; | 128 m_missingGlyphData.glyph = 0; |
| 143 m_zeroWidthFontData->init(this); | |
| 144 m_cjkWidthFontData->init(this); | |
| 145 } | 129 } |
| 146 #endif | 130 #endif |
| 147 | 131 |
| 148 SimpleFontData::SimpleFontData() | |
| 149 : m_treatAsFixedPitch(false) | |
| 150 #if ENABLE(SVG_FONTS) | |
| 151 , m_svgFontData(0) | |
| 152 #endif | |
| 153 , m_isCustomFont(0) | |
| 154 , m_isLoading(0) | |
| 155 , m_smallCapsFontData(0) | |
| 156 { | |
| 157 } | |
| 158 | |
| 159 SimpleFontData::~SimpleFontData() | 132 SimpleFontData::~SimpleFontData() |
| 160 { | 133 { |
| 161 if (!isCustomFont()) { | 134 if (!isCustomFont()) { |
| 162 if (m_smallCapsFontData) | 135 if (m_smallCapsFontData) |
| 163 fontCache()->releaseFontData(m_smallCapsFontData); | 136 fontCache()->releaseFontData(m_smallCapsFontData); |
| 164 GlyphPageTreeNode::pruneTreeFontData(this); | 137 GlyphPageTreeNode::pruneTreeFontData(this); |
| 165 } | 138 } |
| 166 | 139 |
| 167 #if ENABLE(SVG_FONTS) | 140 #if ENABLE(SVG_FONTS) |
| 168 if (!m_svgFontData || !m_svgFontData->svgFontFaceElement()) | 141 if (!m_svgFontData || !m_svgFontData->svgFontFaceElement()) |
| (...skipping 18 matching lines...) Expand all Loading... |
| 187 const SimpleFontData* SimpleFontData::fontDataForCharacter(UChar32) const | 160 const SimpleFontData* SimpleFontData::fontDataForCharacter(UChar32) const |
| 188 { | 161 { |
| 189 return this; | 162 return this; |
| 190 } | 163 } |
| 191 | 164 |
| 192 bool SimpleFontData::isSegmented() const | 165 bool SimpleFontData::isSegmented() const |
| 193 { | 166 { |
| 194 return false; | 167 return false; |
| 195 } | 168 } |
| 196 | 169 |
| 197 const SimpleFontData* SimpleFontData::zeroWidthFontData() const | |
| 198 { | |
| 199 return m_zeroWidthFontData.get(); | |
| 200 } | |
| 201 | |
| 202 const SimpleFontData* SimpleFontData::cjkWidthFontData() const | |
| 203 { | |
| 204 return m_cjkWidthFontData.get(); | |
| 205 } | |
| 206 | |
| 207 void ZeroWidthFontData::init(SimpleFontData* fontData) | |
| 208 { | |
| 209 m_font = fontData->m_font; | |
| 210 m_smallCapsFontData = fontData->m_smallCapsFontData; | |
| 211 m_ascent = fontData->m_ascent; | |
| 212 m_descent = fontData->m_descent; | |
| 213 m_lineSpacing = fontData->m_lineSpacing; | |
| 214 m_lineGap = fontData->m_lineGap; | |
| 215 m_maxCharWidth = 0; | |
| 216 m_avgCharWidth = 0; | |
| 217 m_xHeight = fontData->m_xHeight; | |
| 218 m_unitsPerEm = fontData->m_unitsPerEm; | |
| 219 m_spaceWidth = 0; | |
| 220 m_spaceGlyph = 0; | |
| 221 m_adjustedSpaceWidth = fontData->m_adjustedSpaceWidth; | |
| 222 #if PLATFORM(WIN) | |
| 223 m_scriptCache = 0; | |
| 224 m_scriptFontProperties = 0; | |
| 225 #elif USE(ATSUI) | |
| 226 m_ATSUStyle = 0; | |
| 227 m_ATSUStyleInitialized = false; | |
| 228 m_ATSUMirrors = false; | |
| 229 m_checkedShapesArabic = false; | |
| 230 m_shapesArabic = false; | |
| 231 #endif | |
| 232 } | |
| 233 | |
| 234 CJKWidthFontData::CJKWidthFontData() | |
| 235 : m_cjkGlyphWidth(cGlyphWidthUnknown) | |
| 236 { | |
| 237 } | |
| 238 | |
| 239 float CJKWidthFontData::widthForGlyph(Glyph glyph) const | |
| 240 { | |
| 241 if (m_cjkGlyphWidth != cGlyphWidthUnknown) | |
| 242 return m_cjkGlyphWidth; | |
| 243 | |
| 244 float width = platformWidthForGlyph(glyph); | |
| 245 m_cjkGlyphWidth = width; | |
| 246 | |
| 247 #ifndef NDEBUG | |
| 248 // Test our optimization that assuming all CGK glyphs have the same width | |
| 249 const float actual_width = platformWidthForGlyph(glyph); | |
| 250 ASSERT((cGlyphWidthUnknown == width) || (actual_width == width)); | |
| 251 #endif | |
| 252 | |
| 253 return width; | |
| 254 } | |
| 255 | |
| 256 // static | |
| 257 // TODO(dglazkov): Move to Font::isCJKCodePoint for consistency | |
| 258 bool SimpleFontData::isCJKCodePoint(UChar32 c) | |
| 259 { | |
| 260 // AC00..D7AF; Hangul Syllables | |
| 261 if ((0xAC00 <= c) && (c <= 0xD7AF)) | |
| 262 return true; | |
| 263 | |
| 264 // CJK ideographs | |
| 265 UErrorCode errorCode = U_ZERO_ERROR; // has to be initialized. | |
| 266 return uscript_getScript(c, &errorCode) == USCRIPT_HAN && | |
| 267 U_SUCCESS(errorCode); | |
| 268 } | |
| 269 | |
| 270 | |
| 271 } // namespace WebCore | 170 } // namespace WebCore |
| OLD | NEW |