blob: a0ef0ed27d3141fff5249b3be80193c9a6d973f3 [file] [log] [blame]
/*
* 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 <winsock2.h>
#include "FontCache.h"
#include "FontData.h"
#include "Font.h"
#include <windows.h>
#include <mlang.h>
#include <ApplicationServices/ApplicationServices.h>
#include <WebKitSystemInterface/WebKitSystemInterface.h>
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(&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);
HFONT hfont = 0;
DWORD acpCodePages;
langFontLink->CodePageToCodePages(CP_ACP, &acpCodePages);
DWORD actualCodePages;
long cchActual;
langFontLink->GetStrCodePages(characters, length, acpCodePages, &actualCodePages, &cchActual);
if (cchActual) {
HFONT result;
if (langFontLink->MapFont(hdc, actualCodePages, characters[0], &result) == S_OK) {
// Fill in a log font with the returned font from MLang, and then use that to create a new font.
LOGFONT lf;
GetObject(result, sizeof(LOGFONT), &lf);
langFontLink->ReleaseFont(result);
hfont = CreateFontIndirect(&lf);
SelectObject(hdc, hfont);
WCHAR name[LF_FACESIZE];
GetTextFace(hdc, LF_FACESIZE, name);
String familyName(name);
if (!familyName.isEmpty()) {
FontPlatformData* result = getCachedFontPlatformData(font.fontDescription(), familyName);
if (result)
fontData = getCachedFontData(result);
}
}
}
SelectObject(hdc, oldFont);
if (hfont)
DeleteObject(hfont);
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;
}
}