| // Copyright 2014 PDFium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| |
| #include "public/fpdf_sysfontinfo.h" |
| |
| #include <memory> |
| |
| #include "core/fxcrt/fx_codepage.h" |
| #include "core/fxge/cfx_fontmapper.h" |
| #include "core/fxge/cfx_gemodule.h" |
| #include "core/fxge/fx_font.h" |
| #include "core/fxge/ifx_systemfontinfo.h" |
| #include "fpdfsdk/fsdk_define.h" |
| #include "fpdfsdk/pdfwindow/PWL_FontMap.h" |
| #include "third_party/base/ptr_util.h" |
| |
| static_assert(FXFONT_ANSI_CHARSET == FX_CHARSET_ANSI, "Charset must match"); |
| static_assert(FXFONT_DEFAULT_CHARSET == FX_CHARSET_Default, |
| "Charset must match"); |
| static_assert(FXFONT_SYMBOL_CHARSET == FX_CHARSET_Symbol, "Charset must match"); |
| static_assert(FXFONT_SHIFTJIS_CHARSET == FX_CHARSET_ShiftJIS, |
| "Charset must match"); |
| static_assert(FXFONT_HANGEUL_CHARSET == FX_CHARSET_Hangul, |
| "Charset must match"); |
| static_assert(FXFONT_GB2312_CHARSET == FX_CHARSET_ChineseSimplified, |
| "Charset must match"); |
| static_assert(FXFONT_CHINESEBIG5_CHARSET == FX_CHARSET_ChineseTraditional, |
| "Charset must match"); |
| |
| class CFX_ExternalFontInfo final : public IFX_SystemFontInfo { |
| public: |
| explicit CFX_ExternalFontInfo(FPDF_SYSFONTINFO* pInfo) : m_pInfo(pInfo) {} |
| ~CFX_ExternalFontInfo() override { |
| if (m_pInfo->Release) |
| m_pInfo->Release(m_pInfo); |
| } |
| |
| bool EnumFontList(CFX_FontMapper* pMapper) override { |
| if (m_pInfo->EnumFonts) { |
| m_pInfo->EnumFonts(m_pInfo, pMapper); |
| return true; |
| } |
| return false; |
| } |
| |
| void* MapFont(int weight, |
| bool bItalic, |
| int charset, |
| int pitch_family, |
| const char* family, |
| int& iExact) override { |
| if (!m_pInfo->MapFont) |
| return nullptr; |
| return m_pInfo->MapFont(m_pInfo, weight, bItalic, charset, pitch_family, |
| family, &iExact); |
| } |
| |
| void* GetFont(const char* family) override { |
| if (!m_pInfo->GetFont) |
| return nullptr; |
| return m_pInfo->GetFont(m_pInfo, family); |
| } |
| |
| uint32_t GetFontData(void* hFont, |
| uint32_t table, |
| uint8_t* buffer, |
| uint32_t size) override { |
| if (!m_pInfo->GetFontData) |
| return 0; |
| return m_pInfo->GetFontData(m_pInfo, hFont, table, buffer, size); |
| } |
| |
| bool GetFaceName(void* hFont, CFX_ByteString* name) override { |
| if (!m_pInfo->GetFaceName) |
| return false; |
| uint32_t size = m_pInfo->GetFaceName(m_pInfo, hFont, nullptr, 0); |
| if (size == 0) |
| return false; |
| char* buffer = FX_Alloc(char, size); |
| size = m_pInfo->GetFaceName(m_pInfo, hFont, buffer, size); |
| *name = CFX_ByteString(buffer, size); |
| FX_Free(buffer); |
| return true; |
| } |
| |
| bool GetFontCharset(void* hFont, int* charset) override { |
| if (!m_pInfo->GetFontCharset) |
| return false; |
| |
| *charset = m_pInfo->GetFontCharset(m_pInfo, hFont); |
| return true; |
| } |
| |
| void DeleteFont(void* hFont) override { |
| if (m_pInfo->DeleteFont) |
| m_pInfo->DeleteFont(m_pInfo, hFont); |
| } |
| |
| private: |
| FPDF_SYSFONTINFO* const m_pInfo; |
| }; |
| |
| DLLEXPORT void STDCALL FPDF_AddInstalledFont(void* mapper, |
| const char* name, |
| int charset) { |
| CFX_FontMapper* pMapper = static_cast<CFX_FontMapper*>(mapper); |
| pMapper->AddInstalledFont(name, charset); |
| } |
| |
| DLLEXPORT void STDCALL FPDF_SetSystemFontInfo(FPDF_SYSFONTINFO* pFontInfoExt) { |
| if (pFontInfoExt->version != 1) |
| return; |
| |
| CFX_GEModule::Get()->GetFontMgr()->SetSystemFontInfo( |
| pdfium::MakeUnique<CFX_ExternalFontInfo>(pFontInfoExt)); |
| } |
| |
| DLLEXPORT const FPDF_CharsetFontMap* STDCALL FPDF_GetDefaultTTFMap() { |
| return CPWL_FontMap::defaultTTFMap; |
| } |
| |
| struct FPDF_SYSFONTINFO_DEFAULT : public FPDF_SYSFONTINFO { |
| IFX_SystemFontInfo* m_pFontInfo; |
| }; |
| |
| static void DefaultRelease(struct _FPDF_SYSFONTINFO* pThis) { |
| auto* pDefault = static_cast<FPDF_SYSFONTINFO_DEFAULT*>(pThis); |
| // TODO(thestig): Should this be set to nullptr too? |
| delete pDefault->m_pFontInfo; |
| } |
| |
| static void DefaultEnumFonts(struct _FPDF_SYSFONTINFO* pThis, void* pMapper) { |
| auto* pDefault = static_cast<FPDF_SYSFONTINFO_DEFAULT*>(pThis); |
| pDefault->m_pFontInfo->EnumFontList((CFX_FontMapper*)pMapper); |
| } |
| |
| static void* DefaultMapFont(struct _FPDF_SYSFONTINFO* pThis, |
| int weight, |
| int bItalic, |
| int charset, |
| int pitch_family, |
| const char* family, |
| int* bExact) { |
| auto* pDefault = static_cast<FPDF_SYSFONTINFO_DEFAULT*>(pThis); |
| return pDefault->m_pFontInfo->MapFont(weight, !!bItalic, charset, |
| pitch_family, family, *bExact); |
| } |
| |
| void* DefaultGetFont(struct _FPDF_SYSFONTINFO* pThis, const char* family) { |
| auto* pDefault = static_cast<FPDF_SYSFONTINFO_DEFAULT*>(pThis); |
| return pDefault->m_pFontInfo->GetFont(family); |
| } |
| |
| static unsigned long DefaultGetFontData(struct _FPDF_SYSFONTINFO* pThis, |
| void* hFont, |
| unsigned int table, |
| unsigned char* buffer, |
| unsigned long buf_size) { |
| auto* pDefault = static_cast<FPDF_SYSFONTINFO_DEFAULT*>(pThis); |
| return pDefault->m_pFontInfo->GetFontData(hFont, table, buffer, buf_size); |
| } |
| |
| static unsigned long DefaultGetFaceName(struct _FPDF_SYSFONTINFO* pThis, |
| void* hFont, |
| char* buffer, |
| unsigned long buf_size) { |
| CFX_ByteString name; |
| auto* pDefault = static_cast<FPDF_SYSFONTINFO_DEFAULT*>(pThis); |
| if (!pDefault->m_pFontInfo->GetFaceName(hFont, &name)) |
| return 0; |
| if (name.GetLength() >= (long)buf_size) |
| return name.GetLength() + 1; |
| |
| strncpy(buffer, name.c_str(), |
| (name.GetLength() + 1) * sizeof(CFX_ByteString::CharType)); |
| return name.GetLength() + 1; |
| } |
| |
| static int DefaultGetFontCharset(struct _FPDF_SYSFONTINFO* pThis, void* hFont) { |
| int charset; |
| auto* pDefault = static_cast<FPDF_SYSFONTINFO_DEFAULT*>(pThis); |
| if (!pDefault->m_pFontInfo->GetFontCharset(hFont, &charset)) |
| return 0; |
| return charset; |
| } |
| |
| static void DefaultDeleteFont(struct _FPDF_SYSFONTINFO* pThis, void* hFont) { |
| auto* pDefault = static_cast<FPDF_SYSFONTINFO_DEFAULT*>(pThis); |
| pDefault->m_pFontInfo->DeleteFont(hFont); |
| } |
| |
| DLLEXPORT FPDF_SYSFONTINFO* STDCALL FPDF_GetDefaultSystemFontInfo() { |
| std::unique_ptr<IFX_SystemFontInfo> pFontInfo = |
| IFX_SystemFontInfo::CreateDefault(nullptr); |
| if (!pFontInfo) |
| return nullptr; |
| |
| FPDF_SYSFONTINFO_DEFAULT* pFontInfoExt = |
| FX_Alloc(FPDF_SYSFONTINFO_DEFAULT, 1); |
| pFontInfoExt->DeleteFont = DefaultDeleteFont; |
| pFontInfoExt->EnumFonts = DefaultEnumFonts; |
| pFontInfoExt->GetFaceName = DefaultGetFaceName; |
| pFontInfoExt->GetFont = DefaultGetFont; |
| pFontInfoExt->GetFontCharset = DefaultGetFontCharset; |
| pFontInfoExt->GetFontData = DefaultGetFontData; |
| pFontInfoExt->MapFont = DefaultMapFont; |
| pFontInfoExt->Release = DefaultRelease; |
| pFontInfoExt->version = 1; |
| pFontInfoExt->m_pFontInfo = pFontInfo.release(); |
| return pFontInfoExt; |
| } |
| |
| DLLEXPORT void FPDF_FreeDefaultSystemFontInfo( |
| FPDF_SYSFONTINFO* pDefaultFontInfo) { |
| FX_Free(static_cast<FPDF_SYSFONTINFO_DEFAULT*>(pDefaultFontInfo)); |
| } |