/* * Copyright (C) 2006, 2007 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of * its contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #include #include "FontCache.h" #include "FontData.h" #include "Font.h" #include #include #include #include using std::min; namespace WebCore { void FontCache::platformInit() { wkSetUpFontCache(1536 * 1024 * 4); // This size matches Mac. } IMLangFontLink2* FontCache::getFontLinkInterface() { static IMultiLanguage *multiLanguage; if (!multiLanguage) { if (CoCreateInstance(CLSID_CMultiLanguage, 0, CLSCTX_ALL, IID_IMultiLanguage, (void**)&multiLanguage) != S_OK) return 0; } static IMLangFontLink2* langFontLink; if (!langFontLink) { if (multiLanguage->QueryInterface(IID_IMLangFontLink2, (void**)&langFontLink) != S_OK) return 0; } return langFontLink; } const FontData* FontCache::getFontDataForCharacters(const Font& font, const UChar* characters, int length) { // IMLangFontLink::MapFont Method does what we want. IMLangFontLink2* langFontLink = getFontLinkInterface(); if (!langFontLink) return 0; FontData* fontData = 0; HDC hdc = GetDC(0); HFONT primaryFont = font.primaryFont()->m_font.hfont(); HGDIOBJ oldFont = SelectObject(hdc, primaryFont); DWORD fontCodePages; langFontLink->GetFontCodePages(hdc, primaryFont, &fontCodePages); DWORD actualCodePages; long cchActual; langFontLink->GetStrCodePages(characters, length, fontCodePages, &actualCodePages, &cchActual); if (cchActual) { HFONT result; if (langFontLink->MapFont(hdc, actualCodePages, characters[0], &result) == S_OK) { // MLang has an internal cache, which means it really will hand back the same HFONT pointer if it can. We can // therefore safely use our hash on HFONT pointers (FontPlatformData -> FontData) and actually expect it to work. FontPlatformData platformData(result, font.fontDescription().computedPixelSize(), font.fontDescription().bold(), font.fontDescription().italic()); fontData = getCachedFontData(&platformData); fontData->setIsMLangFont(); } } SelectObject(hdc, oldFont); ReleaseDC(0, hdc); return fontData; } FontPlatformData* FontCache::getSimilarFontPlatformData(const Font& font) { return 0; } FontPlatformData* FontCache::getLastResortFallbackFont(const Font& font) { // FIXME: Would be even better to somehow get the user's default font here. For now we'll pick // the default that the user would get without changing any prefs. static AtomicString timesStr("Times New Roman"); return getCachedFontPlatformData(font.fontDescription(), timesStr); } FontPlatformData* FontCache::createFontPlatformData(const FontDescription& fontDescription, const AtomicString& family) { LOGFONT winfont; // The size here looks unusual. The negative number is intentional. The logical size constant is 32. winfont.lfHeight = -fontDescription.computedPixelSize() * 32; winfont.lfWidth = 0; winfont.lfEscapement = 0; winfont.lfOrientation = 0; winfont.lfUnderline = false; winfont.lfStrikeOut = false; winfont.lfCharSet = DEFAULT_CHARSET; #if PLATFORM(CG) winfont.lfOutPrecision = OUT_TT_ONLY_PRECIS; #else winfont.lfOutPrecision = OUT_TT_PRECIS; #endif winfont.lfQuality = 5; // Force cleartype. winfont.lfPitchAndFamily = DEFAULT_PITCH | FF_DONTCARE; winfont.lfItalic = fontDescription.italic(); // FIXME: Support weights for real. Do our own enumeration of the available weights. // We can't rely on Windows here, since we need to follow the CSS2 algorithm for how to fill in // gaps in the weight list. // FIXME: Hardcoding Lucida Grande for now. It uses different weights than typical Win32 fonts // (500/600 instead of 400/700). static AtomicString lucidaStr("Lucida Grande"); if (equalIgnoringCase(family, lucidaStr)) winfont.lfWeight = fontDescription.bold() ? 600 : 500; else winfont.lfWeight = fontDescription.bold() ? 700 : 400; int len = min(family.length(), (unsigned int)LF_FACESIZE - 1); memcpy(winfont.lfFaceName, family.characters(), len * sizeof(WORD)); winfont.lfFaceName[len] = '\0'; HFONT hfont = CreateFontIndirect(&winfont); // Windows will always give us a valid pointer here, even if the face name is non-existent. We have to double-check // and see if the family name was really used. HDC dc = GetDC((HWND)0); SaveDC(dc); SelectObject(dc, hfont); WCHAR name[LF_FACESIZE]; GetTextFace(dc, LF_FACESIZE, name); RestoreDC(dc, -1); ReleaseDC(0, dc); if (_wcsicmp(winfont.lfFaceName, name)) { DeleteObject(hfont); return 0; } FontPlatformData* result = new FontPlatformData(hfont, fontDescription.computedPixelSize(), fontDescription.bold(), fontDescription.italic()); if (!result->cgFont()) { // The creation of the CGFontRef failed for some reason. We already asserted in debug builds, but to make // absolutely sure that we don't use this font, go ahead and return 0 so that we can fall back to the next // font. delete result; DeleteObject(hfont); return 0; } return result; } }