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

📄 fontcustomplatformdata.cpp

📁 linux下开源浏览器WebKit的源码,市面上的很多商用浏览器都是移植自WebKit
💻 CPP
字号:
/* * Copyright (C) 2007, 2008, 2009 Apple Inc. All rights reserved. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public License * along with this library; see the file COPYING.LIB.  If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. * */#include "config.h"#include "FontCustomPlatformData.h"#include "Base64.h"#include "FontPlatformData.h"#include "OpenTypeUtilities.h"#include "SharedBuffer.h"#include "SoftLinking.h"#include <ApplicationServices/ApplicationServices.h>#include <WebKitSystemInterface/WebKitSystemInterface.h>#include <wtf/RetainPtr.h>// From t2embapi.h, which is missing from the Microsoft Platform SDK.typedef unsigned long(WINAPIV *READEMBEDPROC) (void*, void*, unsigned long);struct TTLOADINFO;#define TTLOAD_PRIVATE 0x00000001#define LICENSE_PREVIEWPRINT 0x0004#define E_NONE 0x0000Lnamespace WebCore {using namespace std;SOFT_LINK_LIBRARY(T2embed);SOFT_LINK(T2embed, TTLoadEmbeddedFont, LONG, __stdcall, (HANDLE* phFontReference, ULONG ulFlags, ULONG* pulPrivStatus, ULONG ulPrivs, ULONG* pulStatus, READEMBEDPROC lpfnReadFromStream, LPVOID lpvReadStream, LPWSTR szWinFamilyName, LPSTR szMacFamilyName, TTLOADINFO* pTTLoadInfo), (phFontReference, ulFlags,pulPrivStatus, ulPrivs, pulStatus, lpfnReadFromStream, lpvReadStream, szWinFamilyName, szMacFamilyName, pTTLoadInfo));SOFT_LINK(T2embed, TTGetNewFontName, LONG, __stdcall, (HANDLE* phFontReference, LPWSTR szWinFamilyName, long cchMaxWinName, LPSTR szMacFamilyName, long cchMaxMacName), (phFontReference, szWinFamilyName, cchMaxWinName, szMacFamilyName, cchMaxMacName));SOFT_LINK(T2embed, TTDeleteEmbeddedFont, LONG, __stdcall, (HANDLE hFontReference, ULONG ulFlags, ULONG* pulStatus), (hFontReference, ulFlags, pulStatus));FontCustomPlatformData::~FontCustomPlatformData(){    CGFontRelease(m_cgFont);    if (m_fontReference) {        if (m_name.isNull()) {            ASSERT(T2embedLibrary());            ULONG status;            TTDeleteEmbeddedFont(m_fontReference, 0, &status);        } else            RemoveFontMemResourceEx(m_fontReference);    }}FontPlatformData FontCustomPlatformData::fontPlatformData(int size, bool bold, bool italic, FontRenderingMode renderingMode){    ASSERT(m_cgFont);    ASSERT(m_fontReference);    ASSERT(T2embedLibrary());    LOGFONT& logFont = *static_cast<LOGFONT*>(malloc(sizeof(LOGFONT)));    if (m_name.isNull())        TTGetNewFontName(&m_fontReference, logFont.lfFaceName, LF_FACESIZE, 0, 0);    else        memcpy(logFont.lfFaceName, m_name.charactersWithNullTermination(), sizeof(logFont.lfFaceName[0]) * min(static_cast<size_t>(LF_FACESIZE), 1 + m_name.length()));    logFont.lfHeight = -size;    if (renderingMode == NormalRenderingMode)        logFont.lfHeight *= 32;    logFont.lfWidth = 0;    logFont.lfEscapement = 0;    logFont.lfOrientation = 0;    logFont.lfUnderline = false;    logFont.lfStrikeOut = false;    logFont.lfCharSet = DEFAULT_CHARSET;    logFont.lfOutPrecision = OUT_TT_ONLY_PRECIS;    logFont.lfQuality = CLEARTYPE_QUALITY;    logFont.lfPitchAndFamily = DEFAULT_PITCH | FF_DONTCARE;    logFont.lfItalic = italic;    logFont.lfWeight = bold ? 700 : 400;    HFONT hfont = CreateFontIndirect(&logFont);    wkSetFontPlatformInfo(m_cgFont, &logFont, free);    return FontPlatformData(hfont, m_cgFont, size, bold, italic, renderingMode == AlternateRenderingMode);}const void* getData(void* info){    SharedBuffer* buffer = static_cast<SharedBuffer*>(info);    buffer->ref();    return (void*)buffer->data();}void releaseData(void* info, const void* data){    static_cast<SharedBuffer*>(info)->deref();}size_t getBytesWithOffset(void *info, void* buffer, size_t offset, size_t count){    SharedBuffer* sharedBuffer = static_cast<SharedBuffer*>(info);    size_t availBytes = count;    if (offset + count > sharedBuffer->size())        availBytes -= (offset + count) - sharedBuffer->size();    memcpy(buffer, sharedBuffer->data() + offset, availBytes);    return availBytes;}// Streams the concatenation of a header and font data.class EOTStream {public:    EOTStream(const Vector<uint8_t, 512>& eotHeader, const SharedBuffer* fontData, size_t overlayDst, size_t overlaySrc, size_t overlayLength)        : m_eotHeader(eotHeader)        , m_fontData(fontData)        , m_overlayDst(overlayDst)        , m_overlaySrc(overlaySrc)        , m_overlayLength(overlayLength)        , m_offset(0)        , m_inHeader(true)    {    }    size_t read(void* buffer, size_t count);private:    const Vector<uint8_t, 512>& m_eotHeader;    const SharedBuffer* m_fontData;    size_t m_overlayDst;    size_t m_overlaySrc;    size_t m_overlayLength;    size_t m_offset;    bool m_inHeader;};size_t EOTStream::read(void* buffer, size_t count){    size_t bytesToRead = count;    if (m_inHeader) {        size_t bytesFromHeader = min(m_eotHeader.size() - m_offset, count);        memcpy(buffer, m_eotHeader.data() + m_offset, bytesFromHeader);        m_offset += bytesFromHeader;        bytesToRead -= bytesFromHeader;        if (m_offset == m_eotHeader.size()) {            m_inHeader = false;            m_offset = 0;        }    }    if (bytesToRead && !m_inHeader) {        size_t bytesFromData = min(m_fontData->size() - m_offset, bytesToRead);        memcpy(buffer, m_fontData->data() + m_offset, bytesFromData);        if (m_offset < m_overlayDst + m_overlayLength && m_offset + bytesFromData >= m_overlayDst) {            size_t dstOffset = max<int>(m_overlayDst - m_offset, 0);            size_t srcOffset = max<int>(0, m_offset - m_overlayDst);            size_t bytesToCopy = min(bytesFromData - dstOffset, m_overlayLength - srcOffset);            memcpy(reinterpret_cast<char*>(buffer) + dstOffset, m_fontData->data() + m_overlaySrc + srcOffset, bytesToCopy);        }        m_offset += bytesFromData;        bytesToRead -= bytesFromData;    }    return count - bytesToRead;}static unsigned long WINAPIV readEmbedProc(void* stream, void* buffer, unsigned long length){    return static_cast<EOTStream*>(stream)->read(buffer, length);}// Creates a unique and unpredictable font name, in order to avoid collisions and to// not allow access from CSS.static String createUniqueFontName(){    Vector<char> fontUuid(sizeof(GUID));    CoCreateGuid(reinterpret_cast<GUID*>(fontUuid.data()));    Vector<char> fontNameVector;    base64Encode(fontUuid, fontNameVector);    ASSERT(fontNameVector.size() < LF_FACESIZE);    return String(fontNameVector.data(), fontNameVector.size());}FontCustomPlatformData* createFontCustomPlatformData(SharedBuffer* buffer){    ASSERT_ARG(buffer, buffer);    ASSERT(T2embedLibrary());    // Get CG to create the font.    CGDataProviderDirectAccessCallbacks callbacks = { &getData, &releaseData, &getBytesWithOffset, NULL };    RetainPtr<CGDataProviderRef> dataProvider(AdoptCF, CGDataProviderCreateDirectAccess(buffer, buffer->size(), &callbacks));    CGFontRef cgFont = CGFontCreateWithDataProvider(dataProvider.get());    if (!cgFont)        return 0;    // Introduce the font to GDI. AddFontMemResourceEx cannot be used, because it will pollute the process's    // font namespace (Windows has no API for creating an HFONT from data without exposing the font to the    // entire process first). TTLoadEmbeddedFont lets us override the font family name, so using a unique name    // we avoid namespace collisions.    String fontName = createUniqueFontName();    // TTLoadEmbeddedFont works only with Embedded OpenType (.eot) data, so we need to create an EOT header    // and prepend it to the font data.    Vector<uint8_t, 512> eotHeader;    size_t overlayDst;    size_t overlaySrc;    size_t overlayLength;    if (!getEOTHeader(buffer, eotHeader, overlayDst, overlaySrc, overlayLength)) {        CGFontRelease(cgFont);        return 0;    }    HANDLE fontReference;    ULONG privStatus;    ULONG status;    EOTStream eotStream(eotHeader, buffer, overlayDst, overlaySrc, overlayLength);    LONG loadEmbeddedFontResult = TTLoadEmbeddedFont(&fontReference, TTLOAD_PRIVATE, &privStatus, LICENSE_PREVIEWPRINT, &status, readEmbedProc, &eotStream, const_cast<LPWSTR>(fontName.charactersWithNullTermination()), 0, 0);    if (loadEmbeddedFontResult == E_NONE)        fontName = String();    else {        fontReference = renameAndActivateFont(buffer, fontName);        if (!fontReference) {            CGFontRelease(cgFont);            return 0;        }    }    return new FontCustomPlatformData(cgFont, fontReference, fontName);}}

⌨️ 快捷键说明

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