// 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 #include #include "core/fxcrt/fx_codepage.h" #include "core/fxge/apple/apple_int.h" #include "core/fxge/cfx_folderfontinfo.h" #include "core/fxge/cfx_gemodule.h" #include "core/fxge/ifx_systemfontinfo.h" #include "third_party/base/ptr_util.h" namespace { const struct { const char* m_pName; const char* m_pSubstName; } g_Base14Substs[] = { {"Courier", "Courier New"}, {"Courier-Bold", "Courier New Bold"}, {"Courier-BoldOblique", "Courier New Bold Italic"}, {"Courier-Oblique", "Courier New Italic"}, {"Helvetica", "Arial"}, {"Helvetica-Bold", "Arial Bold"}, {"Helvetica-BoldOblique", "Arial Bold Italic"}, {"Helvetica-Oblique", "Arial Italic"}, {"Times-Roman", "Times New Roman"}, {"Times-Bold", "Times New Roman Bold"}, {"Times-BoldItalic", "Times New Roman Bold Italic"}, {"Times-Italic", "Times New Roman Italic"}, }; class CFX_MacFontInfo : public CFX_FolderFontInfo { public: CFX_MacFontInfo() {} ~CFX_MacFontInfo() override {} // CFX_FolderFontInfo void* MapFont(int weight, bool bItalic, int charset, int pitch_family, const char* family) override; }; const char JAPAN_GOTHIC[] = "Hiragino Kaku Gothic Pro W6"; const char JAPAN_MINCHO[] = "Hiragino Mincho Pro W6"; void GetJapanesePreference(ByteString* face, int weight, int pitch_family) { if (face->Contains("Gothic")) { *face = JAPAN_GOTHIC; return; } *face = (FontFamilyIsRoman(pitch_family) || weight <= 400) ? JAPAN_MINCHO : JAPAN_GOTHIC; } void* CFX_MacFontInfo::MapFont(int weight, bool bItalic, int charset, int pitch_family, const char* cstr_face) { ByteString face = cstr_face; for (const auto& sub : g_Base14Substs) { if (face == ByteStringView(sub.m_pName)) { face = sub.m_pSubstName; return GetFont(face.c_str()); } } // The request may not ask for the bold and/or italic version of a font by // name. So try to construct the appropriate name. This is not 100% foolproof // as there are fonts that have "Oblique" or "BoldOblique" or "Heavy" in their // names instead. But this at least works for common fonts like Arial and // Times New Roman. A more sophisticated approach would be to find all the // fonts in |m_FontList| with |face| in the name, and examine the fonts to // see which best matches the requested characteristics. if (!face.Contains("Bold") && !face.Contains("Italic")) { ByteString new_face = face; if (weight > 400) new_face += " Bold"; if (bItalic) new_face += " Italic"; auto it = m_FontList.find(new_face); if (it != m_FontList.end()) return it->second.get(); } auto it = m_FontList.find(face); if (it != m_FontList.end()) return it->second.get(); if (charset == FX_CHARSET_ANSI && FontFamilyIsFixedPitch(pitch_family)) return GetFont("Courier New"); if (charset == FX_CHARSET_ANSI || charset == FX_CHARSET_Symbol) return nullptr; switch (charset) { case FX_CHARSET_ShiftJIS: GetJapanesePreference(&face, weight, pitch_family); break; case FX_CHARSET_ChineseSimplified: face = "STSong"; break; case FX_CHARSET_Hangul: face = "AppleMyungjo"; break; case FX_CHARSET_ChineseTraditional: face = "LiSong Pro Light"; } it = m_FontList.find(face); return it != m_FontList.end() ? it->second.get() : nullptr; } } // namespace std::unique_ptr IFX_SystemFontInfo::CreateDefault( const char** pUnused) { auto pInfo = pdfium::MakeUnique(); pInfo->AddPath("~/Library/Fonts"); pInfo->AddPath("/Library/Fonts"); pInfo->AddPath("/System/Library/Fonts"); return std::move(pInfo); } void CFX_GEModule::InitPlatform() { m_pPlatformData = new CApplePlatform; m_pFontMgr->SetSystemFontInfo(IFX_SystemFontInfo::CreateDefault(nullptr)); } void CFX_GEModule::DestroyPlatform() { delete reinterpret_cast(m_pPlatformData); m_pPlatformData = nullptr; }