⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 fontcachewin.cpp

📁 linux下开源浏览器WebKit的源码,市面上的很多商用浏览器都是移植自WebKit
💻 CPP
📖 第 1 页 / 共 2 页
字号:
/* * Copyright (C) 2006, 2007, 2008 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 "Font.h"#include "SimpleFontData.h"#include "StringHash.h"#include "UnicodeRange.h"#include <windows.h>#include <mlang.h>#if PLATFORM(CG)#include <ApplicationServices/ApplicationServices.h>#include <WebKitSystemInterface/WebKitSystemInterface.h>#endifusing std::min;namespace WebCore{void FontCache::platformInit(){#if PLATFORM(CG)    wkSetUpFontCache(1536 * 1024 * 4); // This size matches Mac.#endif}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;}static int CALLBACK metaFileEnumProc(HDC hdc, HANDLETABLE* table, CONST ENHMETARECORD* record, int tableEntries, LPARAM logFont){    if (record->iType == EMR_EXTCREATEFONTINDIRECTW) {        const EMREXTCREATEFONTINDIRECTW* createFontRecord = reinterpret_cast<const EMREXTCREATEFONTINDIRECTW*>(record);        *reinterpret_cast<LOGFONT*>(logFont) = createFontRecord->elfw.elfLogFont;    }    return true;}static int CALLBACK linkedFontEnumProc(CONST LOGFONT* logFont, CONST TEXTMETRIC* metrics, DWORD fontType, LPARAM hfont){    *reinterpret_cast<HFONT*>(hfont) = CreateFontIndirect(logFont);    return false;}static const Vector<String>* getLinkedFonts(String& family){    static HashMap<String, Vector<String>*> systemLinkMap;    Vector<String>* result = systemLinkMap.get(family);    if (result)        return result;    result = new Vector<String>;    systemLinkMap.set(family, result);    HKEY fontLinkKey;    if (FAILED(RegOpenKeyEx(HKEY_LOCAL_MACHINE, L"Software\\Microsoft\\Windows NT\\CurrentVersion\\FontLink\\SystemLink", 0, KEY_READ, &fontLinkKey)))        return result;    DWORD linkedFontsBufferSize = 0;    RegQueryValueEx(fontLinkKey, family.charactersWithNullTermination(), 0, NULL, NULL, &linkedFontsBufferSize);    WCHAR* linkedFonts = reinterpret_cast<WCHAR*>(malloc(linkedFontsBufferSize));    if (SUCCEEDED(RegQueryValueEx(fontLinkKey, family.charactersWithNullTermination(), 0, NULL, reinterpret_cast<BYTE*>(linkedFonts), &linkedFontsBufferSize))) {        unsigned i = 0;        unsigned length = linkedFontsBufferSize / sizeof(*linkedFonts);        while (i < length) {            while (i < length && linkedFonts[i] != ',')                i++;            i++;            unsigned j = i;            while (j < length && linkedFonts[j])                j++;            result->append(String(linkedFonts + i, j - i));            i = j + 1;        }    }    free(linkedFonts);    RegCloseKey(fontLinkKey);    return result;}static const Vector<DWORD, 4>& getCJKCodePageMasks(){    // The default order in which we look for a font for a CJK character. If the user's default code page is    // one of these, we will use it first.    static const UINT CJKCodePages[] = {         932, /* Japanese */        936, /* Simplified Chinese */        950, /* Traditional Chinese */        949  /* Korean */    };    static Vector<DWORD, 4> codePageMasks;    static bool initialized;    if (!initialized) {        initialized = true;        IMLangFontLink2* langFontLink = fontCache()->getFontLinkInterface();        if (!langFontLink)            return codePageMasks;        UINT defaultCodePage;        DWORD defaultCodePageMask = 0;        if (GetLocaleInfo(LOCALE_USER_DEFAULT, LOCALE_RETURN_NUMBER | LOCALE_IDEFAULTANSICODEPAGE, reinterpret_cast<LPWSTR>(&defaultCodePage), sizeof(defaultCodePage)))            langFontLink->CodePageToCodePages(defaultCodePage, &defaultCodePageMask);        if (defaultCodePage == CJKCodePages[0] || defaultCodePage == CJKCodePages[1] || defaultCodePage == CJKCodePages[2] || defaultCodePage == CJKCodePages[3])            codePageMasks.append(defaultCodePageMask);        for (unsigned i = 0; i < 4; ++i) {            if (defaultCodePage != CJKCodePages[i]) {                DWORD codePageMask;                langFontLink->CodePageToCodePages(CJKCodePages[i], &codePageMask);                codePageMasks.append(codePageMask);            }        }    }    return codePageMasks;}static bool currentFontContainsCharacter(HDC hdc, UChar character){    static Vector<char, 512> glyphsetBuffer;    glyphsetBuffer.resize(GetFontUnicodeRanges(hdc, 0));    GLYPHSET* glyphset = reinterpret_cast<GLYPHSET*>(glyphsetBuffer.data());    GetFontUnicodeRanges(hdc, glyphset);    // FIXME: Change this to a binary search.    unsigned i = 0;    while (i < glyphset->cRanges && glyphset->ranges[i].wcLow <= character)        i++;    return i && glyphset->ranges[i - 1].wcLow + glyphset->ranges[i - 1].cGlyphs > character;}static HFONT createMLangFont(IMLangFontLink2* langFontLink, HDC hdc, DWORD codePageMask, UChar character = 0){    HFONT MLangFont;    HFONT hfont = 0;    if (SUCCEEDED(langFontLink->MapFont(hdc, codePageMask, character, &MLangFont)) && MLangFont) {        LOGFONT lf;        GetObject(MLangFont, sizeof(LOGFONT), &lf);        langFontLink->ReleaseFont(MLangFont);        hfont = CreateFontIndirect(&lf);    }    return hfont;}const SimpleFontData* FontCache::getFontDataForCharacters(const Font& font, const UChar* characters, int length){    UChar character = characters[0];    SimpleFontData* fontData = 0;    HDC hdc = GetDC(0);    HFONT primaryFont = font.primaryFont()->fontDataForCharacter(character)->platformData().hfont();    HGDIOBJ oldFont = SelectObject(hdc, primaryFont);    HFONT hfont = 0;    if (IMLangFontLink2* langFontLink = getFontLinkInterface()) {        // Try MLang font linking first.        DWORD codePages = 0;        langFontLink->GetCharCodePages(character, &codePages);        if (codePages && findCharUnicodeRange(character) == cRangeSetCJK) {            // The CJK character may belong to multiple code pages. We want to            // do font linking against a single one of them, preferring the default            // code page for the user's locale.            const Vector<DWORD, 4>& CJKCodePageMasks = getCJKCodePageMasks();            unsigned numCodePages = CJKCodePageMasks.size();            for (unsigned i = 0; i < numCodePages && !hfont; ++i) {                hfont = createMLangFont(langFontLink, hdc, CJKCodePageMasks[i]);                if (hfont && !(codePages & CJKCodePageMasks[i])) {                    // We asked about a code page that is not one of the code pages                    // returned by MLang, so the font might not contain the character.                    SelectObject(hdc, hfont);                    if (!currentFontContainsCharacter(hdc, character)) {                        DeleteObject(hfont);                        hfont = 0;                    }                    SelectObject(hdc, primaryFont);                }            }        } else            hfont = createMLangFont(langFontLink, hdc, codePages, character);    }    // A font returned from MLang is trusted to contain the character.    bool containsCharacter = hfont;    if (!hfont) {        // To find out what font Uniscribe would use, we make it draw into a metafile and intercept        // calls to CreateFontIndirect().        HDC metaFileDc = CreateEnhMetaFile(hdc, NULL, NULL, NULL);        SelectObject(metaFileDc, primaryFont);        bool scriptStringOutSucceeded = false;        SCRIPT_STRING_ANALYSIS ssa;        // FIXME: If length is greater than 1, we actually return the font for the last character.        // This function should be renamed getFontDataForCharacter and take a single 32-bit character.        if (SUCCEEDED(ScriptStringAnalyse(metaFileDc, characters, length, 0, -1, SSA_METAFILE | SSA_FALLBACK | SSA_GLYPHS | SSA_LINK,            0, NULL, NULL, NULL, NULL, NULL, &ssa))) {            scriptStringOutSucceeded = SUCCEEDED(ScriptStringOut(ssa, 0, 0, 0, NULL, 0, 0, FALSE));            ScriptStringFree(&ssa);        }        HENHMETAFILE metaFile = CloseEnhMetaFile(metaFileDc);        if (scriptStringOutSucceeded) {            LOGFONT logFont;            logFont.lfFaceName[0] = 0;            EnumEnhMetaFile(0, metaFile, metaFileEnumProc, &logFont, NULL);            if (logFont.lfFaceName[0])                hfont = CreateFontIndirect(&logFont);        }        DeleteEnhMetaFile(metaFile);    }    String familyName;    const Vector<String>* linkedFonts = 0;    unsigned linkedFontIndex = 0;    while (hfont) {        SelectObject(hdc, hfont);        WCHAR name[LF_FACESIZE];        GetTextFace(hdc, LF_FACESIZE, name);        familyName = name;        if (containsCharacter || currentFontContainsCharacter(hdc, character))            break;        if (!linkedFonts)            linkedFonts = getLinkedFonts(familyName);        SelectObject(hdc, oldFont);        DeleteObject(hfont);        hfont = 0;        if (linkedFonts->size() <= linkedFontIndex)

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -