Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(75)

Side by Side Diff: Source/core/platform/graphics/win/FontPlatformDataWin.cpp

Issue 99103006: Moving GraphicsContext and dependencies from core to platform. (Closed) Base URL: svn://svn.chromium.org/blink/trunk
Patch Set: Final patch - fixes Android Created 7 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
(Empty)
1 /*
2 * Copyright (C) 2006, 2007 Apple Computer, Inc.
3 * Copyright (c) 2006, 2007, 2008, 2009, 2012 Google Inc. All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are
7 * met:
8 *
9 * * Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * * Redistributions in binary form must reproduce the above
12 * copyright notice, this list of conditions and the following disclaimer
13 * in the documentation and/or other materials provided with the
14 * distribution.
15 * * Neither the name of Google Inc. nor the names of its
16 * contributors may be used to endorse or promote products derived from
17 * this software without specific prior written permission.
18 *
19 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30 */
31
32 #include "config.h"
33 #include "core/platform/graphics/FontPlatformData.h"
34
35 #include "core/platform/graphics/FontCache.h"
36 #include "core/platform/graphics/GraphicsContext.h"
37 #if USE(HARFBUZZ)
38 #include "core/platform/graphics/harfbuzz/HarfBuzzFace.h"
39 #endif
40 #include "core/platform/graphics/skia/SkiaFontWin.h"
41 #include "platform/LayoutTestSupport.h"
42 #include "platform/win/HWndDC.h"
43 #include "public/platform/Platform.h"
44 #include "public/platform/win/WebSandboxSupport.h"
45 #include "wtf/PassOwnPtr.h"
46 #include "wtf/StdLibExtras.h"
47 #include <mlang.h>
48 #include <objidl.h>
49 #include <windows.h>
50
51 namespace WebCore {
52
53 void FontPlatformData::setupPaint(SkPaint* paint, GraphicsContext* context) cons t
54 {
55 const float ts = m_textSize >= 0 ? m_textSize : 12;
56 paint->setTextSize(SkFloatToScalar(m_textSize));
57 paint->setTypeface(typeface());
58 paint->setFakeBoldText(m_fakeBold);
59 paint->setTextSkewX(m_fakeItalic ? -SK_Scalar1 / 4 : 0);
60 paint->setSubpixelText(m_useSubpixelPositioning);
61
62 int textFlags = paintTextFlags();
63 // Only set painting flags when we're actually painting.
64 if (context && !context->couldUseLCDRenderedText()) {
65 textFlags &= ~SkPaint::kLCDRenderText_Flag;
66 // If we *just* clear our request for LCD, then GDI seems to
67 // sometimes give us AA text, and sometimes give us BW text. Since the
68 // original intent was LCD, we want to force AA (rather than BW), so we
69 // add a special bit to tell Skia to do its best to avoid the BW: by
70 // drawing LCD offscreen and downsampling that to AA.
71 textFlags |= SkPaint::kGenA8FromLCD_Flag;
72 }
73
74 static const uint32_t textFlagsMask = SkPaint::kAntiAlias_Flag |
75 SkPaint::kLCDRenderText_Flag |
76 SkPaint::kGenA8FromLCD_Flag;
77
78 SkASSERT(!(textFlags & ~textFlagsMask));
79 uint32_t flags = paint->getFlags();
80 flags &= ~textFlagsMask;
81 flags |= textFlags;
82 paint->setFlags(flags);
83 }
84
85 // Lookup the current system settings for font smoothing.
86 // We cache these values for performance, but if the browser has a way to be
87 // notified when these change, we could re-query them at that time.
88 static uint32_t getDefaultGDITextFlags()
89 {
90 static bool gInited;
91 static uint32_t gFlags;
92 if (!gInited) {
93 BOOL enabled;
94 gFlags = 0;
95 if (SystemParametersInfo(SPI_GETFONTSMOOTHING, 0, &enabled, 0) && enable d) {
96 gFlags |= SkPaint::kAntiAlias_Flag;
97
98 UINT smoothType;
99 if (SystemParametersInfo(SPI_GETFONTSMOOTHINGTYPE, 0, &smoothType, 0 )) {
100 if (FE_FONTSMOOTHINGCLEARTYPE == smoothType)
101 gFlags |= SkPaint::kLCDRenderText_Flag;
102 }
103 }
104 gInited = true;
105 }
106 return gFlags;
107 }
108
109 static bool isWebFont(const LOGFONT& lf)
110 {
111 // web-fonts have artifical names constructed to always be
112 // 1. 24 characters, followed by a '\0'
113 // 2. the last two characters are '=='
114 return '=' == lf.lfFaceName[22] && '=' == lf.lfFaceName[23] && '\0' == lf.lf FaceName[24];
115 }
116
117 static int computePaintTextFlags(const LOGFONT& lf)
118 {
119 int textFlags = 0;
120 switch (lf.lfQuality) {
121 case NONANTIALIASED_QUALITY:
122 textFlags = 0;
123 break;
124 case ANTIALIASED_QUALITY:
125 textFlags = SkPaint::kAntiAlias_Flag;
126 break;
127 case CLEARTYPE_QUALITY:
128 textFlags = (SkPaint::kAntiAlias_Flag | SkPaint::kLCDRenderText_Flag);
129 break;
130 default:
131 textFlags = getDefaultGDITextFlags();
132 break;
133 }
134
135 // only allow features that SystemParametersInfo allows
136 textFlags &= getDefaultGDITextFlags();
137
138 /*
139 * FontPlatformData(...) will read our logfont, and try to honor the the lf Quality
140 * setting (computing the corresponding SkPaint flags for AA and LCD). Howe ver, it
141 * will limit the quality based on its query of SPI_GETFONTSMOOTHING. This could mean
142 * we end up drawing the text in BW, even though our lfQuality requested an tialiasing.
143 *
144 * Many web-fonts are so poorly hinted that they are terrible to read when drawn in BW.
145 * In these cases, we have decided to FORCE these fonts to be drawn with at least grayscale AA,
146 * even when the System (getDefaultGDITextFlags) tells us to draw only in B W.
147 */
148 if (isWebFont(lf) && !isRunningLayoutTest())
149 textFlags |= SkPaint::kAntiAlias_Flag;
150 return textFlags;
151 }
152
153 #if !USE(HARFBUZZ)
154 PassRefPtr<SkTypeface> CreateTypefaceFromHFont(HFONT hfont, int* size, int* pain tTextFlags)
155 {
156 LOGFONT info;
157 GetObject(hfont, sizeof(info), &info);
158 if (size) {
159 int height = info.lfHeight;
160 if (height < 0)
161 height = -height;
162 *size = height;
163 }
164 if (paintTextFlags)
165 *paintTextFlags = computePaintTextFlags(info);
166 return adoptRef(SkCreateTypefaceFromLOGFONT(info));
167 }
168 #endif
169
170 FontPlatformData::FontPlatformData(WTF::HashTableDeletedValueType)
171 : m_textSize(-1)
172 , m_fakeBold(false)
173 , m_fakeItalic(false)
174 , m_orientation(Horizontal)
175 , m_typeface(adoptRef(SkTypeface::RefDefault()))
176 , m_paintTextFlags(0)
177 , m_isHashTableDeletedValue(true)
178 , m_useSubpixelPositioning(false)
179 {
180 #if !USE(HARFBUZZ)
181 m_font = 0;
182 m_scriptCache = 0;
183 #endif
184 }
185
186 FontPlatformData::FontPlatformData()
187 : m_textSize(0)
188 , m_fakeBold(false)
189 , m_fakeItalic(false)
190 , m_orientation(Horizontal)
191 , m_typeface(adoptRef(SkTypeface::RefDefault()))
192 , m_paintTextFlags(0)
193 , m_isHashTableDeletedValue(false)
194 , m_useSubpixelPositioning(false)
195 {
196 #if !USE(HARFBUZZ)
197 m_font = 0;
198 m_scriptCache = 0;
199 #endif
200 }
201
202 #if ENABLE(GDI_FONTS_ON_WINDOWS) && !USE(HARFBUZZ)
203 FontPlatformData::FontPlatformData(HFONT font, float size, FontOrientation orien tation)
204 : m_font(RefCountedHFONT::create(font))
205 , m_textSize(size)
206 , m_fakeBold(false)
207 , m_fakeItalic(false)
208 , m_orientation(orientation)
209 , m_scriptCache(0)
210 , m_typeface(CreateTypefaceFromHFont(font, 0, &m_paintTextFlags))
211 , m_isHashTableDeletedValue(false)
212 , m_useSubpixelPositioning(false)
213 {
214 }
215 #endif
216
217 // FIXME: this constructor is needed for SVG fonts but doesn't seem to do much
218 FontPlatformData::FontPlatformData(float size, bool bold, bool oblique)
219 : m_textSize(size)
220 , m_fakeBold(false)
221 , m_fakeItalic(false)
222 , m_orientation(Horizontal)
223 , m_typeface(adoptRef(SkTypeface::RefDefault()))
224 , m_paintTextFlags(0)
225 , m_isHashTableDeletedValue(false)
226 , m_useSubpixelPositioning(false)
227 {
228 #if !USE(HARFBUZZ)
229 m_font = 0;
230 m_scriptCache = 0;
231 #endif
232 }
233
234 FontPlatformData::FontPlatformData(const FontPlatformData& data)
235 : m_textSize(data.m_textSize)
236 , m_fakeBold(data.m_fakeBold)
237 , m_fakeItalic(data.m_fakeItalic)
238 , m_orientation(data.m_orientation)
239 , m_typeface(data.m_typeface)
240 , m_paintTextFlags(data.m_paintTextFlags)
241 , m_isHashTableDeletedValue(false)
242 , m_useSubpixelPositioning(data.m_useSubpixelPositioning)
243 {
244 #if !USE(HARFBUZZ)
245 m_font = data.m_font;
246 m_scriptCache = 0;
247 #endif
248 }
249
250 FontPlatformData::FontPlatformData(const FontPlatformData& data, float textSize)
251 : m_textSize(textSize)
252 , m_fakeBold(data.m_fakeBold)
253 , m_fakeItalic(data.m_fakeItalic)
254 , m_orientation(data.m_orientation)
255 , m_typeface(data.m_typeface)
256 , m_paintTextFlags(data.m_paintTextFlags)
257 , m_isHashTableDeletedValue(false)
258 , m_useSubpixelPositioning(data.m_useSubpixelPositioning)
259 {
260 #if !USE(HARFBUZZ)
261 m_font = data.m_font;
262 m_scriptCache = 0;
263 #endif
264 }
265
266 FontPlatformData::FontPlatformData(PassRefPtr<SkTypeface> tf, const char* family ,
267 float textSize, bool fakeBold, bool fakeItalic, FontOrientation orientation,
268 bool useSubpixelPositioning)
269 : m_textSize(textSize)
270 , m_fakeBold(fakeBold)
271 , m_fakeItalic(fakeItalic)
272 , m_orientation(orientation)
273 , m_typeface(tf)
274 , m_isHashTableDeletedValue(false)
275 , m_useSubpixelPositioning(useSubpixelPositioning)
276 {
277 // FIXME: This can be removed together with m_font once the last few
278 // uses of hfont() has been eliminated.
279 LOGFONT logFont;
280 SkLOGFONTFromTypeface(m_typeface.get(), &logFont);
281 logFont.lfHeight = -textSize;
282 m_paintTextFlags = computePaintTextFlags(logFont);
283
284 #if !USE(HARFBUZZ)
285 HFONT hFont = CreateFontIndirect(&logFont);
286 m_font = hFont ? RefCountedHFONT::create(hFont) : 0;
287 m_scriptCache = 0;
288 #endif
289 }
290
291 FontPlatformData& FontPlatformData::operator=(const FontPlatformData& data)
292 {
293 if (this != &data) {
294 m_textSize = data.m_textSize;
295 m_fakeBold = data.m_fakeBold;
296 m_fakeItalic = data.m_fakeItalic;
297 m_orientation = data.m_orientation;
298 m_typeface = data.m_typeface;
299 m_paintTextFlags = data.m_paintTextFlags;
300
301 #if !USE(HARFBUZZ)
302 m_font = data.m_font;
303 // The following fields will get re-computed if necessary.
304 ScriptFreeCache(&m_scriptCache);
305 m_scriptCache = 0;
306 m_scriptFontProperties.clear();
307 #endif
308 }
309 return *this;
310 }
311
312 FontPlatformData::~FontPlatformData()
313 {
314 #if !USE(HARFBUZZ)
315 ScriptFreeCache(&m_scriptCache);
316 m_scriptCache = 0;
317 #endif
318 }
319
320 String FontPlatformData::fontFamilyName() const
321 {
322 #if ENABLE(GDI_FONTS_ON_WINDOWS)
323 HWndDC dc(0);
324 HGDIOBJ oldFont = static_cast<HFONT>(SelectObject(dc, hfont()));
325 WCHAR name[LF_FACESIZE];
326 unsigned resultLength = GetTextFace(dc, LF_FACESIZE, name);
327 if (resultLength > 0)
328 resultLength--; // ignore the null terminator
329 SelectObject(dc, oldFont);
330 return String(name, resultLength);
331 #else
332 // FIXME: This returns the requested name, perhaps a better solution would b e to
333 // return the list of names provided by SkTypeface::createFamilyNameIterator .
334 ASSERT(typeface());
335 SkString familyName;
336 typeface()->getFamilyName(&familyName);
337 return String::fromUTF8(familyName.c_str());
338 #endif
339 }
340
341 bool FontPlatformData::isFixedPitch() const
342 {
343 #if ENABLE(GDI_FONTS_ON_WINDOWS)
344 // TEXTMETRICS have this. Set m_treatAsFixedPitch based off that.
345 HWndDC dc(0);
346 HGDIOBJ oldFont = SelectObject(dc, hfont());
347
348 // Yes, this looks backwards, but the fixed pitch bit is actually set if the font
349 // is *not* fixed pitch. Unbelievable but true.
350 TEXTMETRIC textMetric = { 0 };
351 if (!GetTextMetrics(dc, &textMetric)) {
352 if (ensureFontLoaded(hfont())) {
353 // Retry GetTextMetrics.
354 // FIXME: Handle gracefully the error if this call also fails.
355 // See http://crbug.com/6401.
356 if (!GetTextMetrics(dc, &textMetric))
357 WTF_LOG_ERROR("Unable to get the text metrics after second attem pt");
358 }
359 }
360
361 bool treatAsFixedPitch = !(textMetric.tmPitchAndFamily & TMPF_FIXED_PITCH);
362
363 SelectObject(dc, oldFont);
364
365 return treatAsFixedPitch;
366 #else
367 return typeface() && typeface()->isFixedPitch();
368 #endif
369 }
370
371 bool FontPlatformData::operator==(const FontPlatformData& a) const
372 {
373 return SkTypeface::Equal(m_typeface.get(), a.m_typeface.get())
374 && m_textSize == a.m_textSize
375 && m_fakeBold == a.m_fakeBold
376 && m_fakeItalic == a.m_fakeItalic
377 && m_orientation == a.m_orientation
378 && m_isHashTableDeletedValue == a.m_isHashTableDeletedValue;
379 }
380
381 #if USE(HARFBUZZ)
382 HarfBuzzFace* FontPlatformData::harfBuzzFace() const
383 {
384 if (!m_harfBuzzFace)
385 m_harfBuzzFace = HarfBuzzFace::create(const_cast<FontPlatformData*>(this ), uniqueID());
386
387 return m_harfBuzzFace.get();
388 }
389
390 #else
391 FontPlatformData::RefCountedHFONT::~RefCountedHFONT()
392 {
393 DeleteObject(m_hfont);
394 }
395
396 SCRIPT_FONTPROPERTIES* FontPlatformData::scriptFontProperties() const
397 {
398 if (!m_scriptFontProperties) {
399 m_scriptFontProperties = adoptPtr(new SCRIPT_FONTPROPERTIES);
400 memset(m_scriptFontProperties.get(), 0, sizeof(SCRIPT_FONTPROPERTIES));
401 m_scriptFontProperties->cBytes = sizeof(SCRIPT_FONTPROPERTIES);
402 HRESULT result = ScriptGetFontProperties(0, scriptCache(), m_scriptFontP roperties.get());
403 if (result == E_PENDING) {
404 HWndDC dc(0);
405 HGDIOBJ oldFont = SelectObject(dc, hfont());
406 HRESULT hr = ScriptGetFontProperties(dc, scriptCache(), m_scriptFont Properties.get());
407 if (S_OK != hr) {
408 if (FontPlatformData::ensureFontLoaded(hfont())) {
409 // FIXME: Handle gracefully the error if this call also fail s.
410 hr = ScriptGetFontProperties(dc, scriptCache(), m_scriptFont Properties.get());
411 if (S_OK != hr) {
412 WTF_LOG_ERROR("Unable to get the font properties after s econd attempt");
413 }
414 }
415 }
416
417 SelectObject(dc, oldFont);
418 }
419 }
420 return m_scriptFontProperties.get();
421 }
422
423 bool FontPlatformData::ensureFontLoaded(HFONT font)
424 {
425 blink::WebSandboxSupport* sandboxSupport = blink::Platform::current()->sandb oxSupport();
426 // if there is no sandbox, then we can assume the font
427 // was able to be loaded successfully already
428 return sandboxSupport ? sandboxSupport->ensureFontLoaded(font) : true;
429 }
430 #endif
431
432 #ifndef NDEBUG
433 String FontPlatformData::description() const
434 {
435 return String();
436 }
437 #endif
438
439 }
OLDNEW
« no previous file with comments | « Source/core/platform/graphics/win/FontPlatformDataWin.h ('k') | Source/core/platform/graphics/win/FontWin.cpp » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698