blob: 0d5cb699499850a740c3c399c514d2f1544ff790 [file] [log] [blame]
// Copyright 2017 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 "core/fpdfapi/font/cpdf_fontglobals.h"
#include "core/fpdfapi/cmaps/CNS1/cmaps_cns1.h"
#include "core/fpdfapi/cmaps/GB1/cmaps_gb1.h"
#include "core/fpdfapi/cmaps/Japan1/cmaps_japan1.h"
#include "core/fpdfapi/cmaps/Korea1/cmaps_korea1.h"
#include "core/fpdfapi/font/cfx_stockfontarray.h"
#include "core/fpdfapi/font/cpdf_cid2unicodemap.h"
#include "core/fpdfapi/font/cpdf_cmap.h"
#include "core/fpdfapi/parser/cpdf_document.h"
#include "third_party/base/check.h"
#include "third_party/base/containers/contains.h"
namespace {
CPDF_FontGlobals* g_FontGlobals = nullptr;
RetainPtr<const CPDF_CMap> LoadPredefinedCMap(ByteStringView name) {
if (!name.IsEmpty() && name[0] == '/')
name = name.Last(name.GetLength() - 1);
return pdfium::MakeRetain<CPDF_CMap>(name);
}
} // namespace
// static
void CPDF_FontGlobals::Create() {
DCHECK(!g_FontGlobals);
g_FontGlobals = new CPDF_FontGlobals();
}
// static
void CPDF_FontGlobals::Destroy() {
DCHECK(g_FontGlobals);
delete g_FontGlobals;
g_FontGlobals = nullptr;
}
// static
CPDF_FontGlobals* CPDF_FontGlobals::GetInstance() {
DCHECK(g_FontGlobals);
return g_FontGlobals;
}
CPDF_FontGlobals::CPDF_FontGlobals() {
memset(m_EmbeddedCharsets, 0, sizeof(m_EmbeddedCharsets));
memset(m_EmbeddedToUnicodes, 0, sizeof(m_EmbeddedToUnicodes));
}
CPDF_FontGlobals::~CPDF_FontGlobals() = default;
void CPDF_FontGlobals::LoadEmbeddedMaps() {
LoadEmbeddedGB1CMaps();
LoadEmbeddedCNS1CMaps();
LoadEmbeddedJapan1CMaps();
LoadEmbeddedKorea1CMaps();
}
RetainPtr<CPDF_Font> CPDF_FontGlobals::Find(
CPDF_Document* pDoc,
CFX_FontMapper::StandardFont index) {
auto it = m_StockMap.find(pDoc);
if (it == m_StockMap.end() || !it->second)
return nullptr;
return it->second->GetFont(index);
}
void CPDF_FontGlobals::Set(CPDF_Document* pDoc,
CFX_FontMapper::StandardFont index,
const RetainPtr<CPDF_Font>& pFont) {
if (!pdfium::Contains(m_StockMap, pDoc))
m_StockMap[pDoc] = std::make_unique<CFX_StockFontArray>();
m_StockMap[pDoc]->SetFont(index, pFont);
}
void CPDF_FontGlobals::Clear(CPDF_Document* pDoc) {
m_StockMap.erase(pDoc);
}
void CPDF_FontGlobals::LoadEmbeddedGB1CMaps() {
SetEmbeddedCharset(
CIDSET_GB1, pdfium::make_span(kFXCMAP_GB1_cmaps, kFXCMAP_GB1_cmaps_size));
SetEmbeddedToUnicode(CIDSET_GB1, kFXCMAP_GB1CID2Unicode_5);
}
void CPDF_FontGlobals::LoadEmbeddedCNS1CMaps() {
SetEmbeddedCharset(CIDSET_CNS1, pdfium::make_span(kFXCMAP_CNS1_cmaps,
kFXCMAP_CNS1_cmaps_size));
SetEmbeddedToUnicode(CIDSET_CNS1, kFXCMAP_CNS1CID2Unicode_5);
}
void CPDF_FontGlobals::LoadEmbeddedJapan1CMaps() {
SetEmbeddedCharset(
CIDSET_JAPAN1,
pdfium::make_span(kFXCMAP_Japan1_cmaps, kFXCMAP_Japan1_cmaps_size));
SetEmbeddedToUnicode(CIDSET_JAPAN1, kFXCMAP_Japan1CID2Unicode_4);
}
void CPDF_FontGlobals::LoadEmbeddedKorea1CMaps() {
SetEmbeddedCharset(
CIDSET_KOREA1,
pdfium::make_span(kFXCMAP_Korea1_cmaps, kFXCMAP_Korea1_cmaps_size));
SetEmbeddedToUnicode(CIDSET_KOREA1, kFXCMAP_Korea1CID2Unicode_2);
}
RetainPtr<const CPDF_CMap> CPDF_FontGlobals::GetPredefinedCMap(
const ByteString& name) {
auto it = m_CMaps.find(name);
if (it != m_CMaps.end())
return it->second;
RetainPtr<const CPDF_CMap> pCMap = LoadPredefinedCMap(name.AsStringView());
if (!name.IsEmpty())
m_CMaps[name] = pCMap;
return pCMap;
}
CPDF_CID2UnicodeMap* CPDF_FontGlobals::GetCID2UnicodeMap(CIDSet charset) {
if (!m_CID2UnicodeMaps[charset]) {
m_CID2UnicodeMaps[charset] = std::make_unique<CPDF_CID2UnicodeMap>(charset);
}
return m_CID2UnicodeMaps[charset].get();
}