blob: b90accd8d7bc2ad4bb1629852745226339241c9e [file] [log] [blame]
// Copyright 2016 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_simplefont.h"
#include "core/fpdfapi/parser/cpdf_array.h"
#include "core/fpdfapi/parser/cpdf_dictionary.h"
#include "core/fpdfapi/parser/cpdf_name.h"
#include "core/fxge/fx_font.h"
#include "core/fxge/fx_freetype.h"
#include "third_party/base/numerics/safe_math.h"
#include "third_party/base/stl_util.h"
namespace {
void GetPredefinedEncoding(const ByteString& value, int* basemap) {
if (value == "WinAnsiEncoding")
*basemap = PDFFONT_ENCODING_WINANSI;
else if (value == "MacRomanEncoding")
*basemap = PDFFONT_ENCODING_MACROMAN;
else if (value == "MacExpertEncoding")
*basemap = PDFFONT_ENCODING_MACEXPERT;
else if (value == "PDFDocEncoding")
*basemap = PDFFONT_ENCODING_PDFDOC;
}
} // namespace
CPDF_SimpleFont::CPDF_SimpleFont(CPDF_Document* pDocument,
CPDF_Dictionary* pFontDict)
: CPDF_Font(pDocument, pFontDict) {
memset(m_CharWidth, 0xff, sizeof(m_CharWidth));
memset(m_GlyphIndex, 0xff, sizeof(m_GlyphIndex));
for (size_t i = 0; i < pdfium::size(m_CharBBox); ++i)
m_CharBBox[i] = FX_RECT(-1, -1, -1, -1);
}
CPDF_SimpleFont::~CPDF_SimpleFont() = default;
int CPDF_SimpleFont::GlyphFromCharCode(uint32_t charcode, bool* pVertGlyph) {
if (pVertGlyph)
*pVertGlyph = false;
if (charcode > 0xff)
return -1;
int index = m_GlyphIndex[charcode];
if (index == 0xffff)
return -1;
return index;
}
void CPDF_SimpleFont::LoadCharMetrics(int charcode) {
if (!m_Font.GetFaceRec())
return;
if (charcode < 0 || charcode > 0xff) {
return;
}
int glyph_index = m_GlyphIndex[charcode];
if (glyph_index == 0xffff) {
if (!m_pFontFile && charcode != 32) {
LoadCharMetrics(32);
m_CharBBox[charcode] = m_CharBBox[32];
if (m_bUseFontWidth) {
m_CharWidth[charcode] = m_CharWidth[32];
}
}
return;
}
FXFT_FaceRec* face = m_Font.GetFaceRec();
int err =
FT_Load_Glyph(face, glyph_index,
FT_LOAD_NO_SCALE | FT_LOAD_IGNORE_GLOBAL_ADVANCE_WIDTH);
if (err)
return;
int iHoriBearingX = FXFT_Get_Glyph_HoriBearingX(face);
int iHoriBearingY = FXFT_Get_Glyph_HoriBearingY(face);
m_CharBBox[charcode] =
FX_RECT(TT2PDF(iHoriBearingX, face), TT2PDF(iHoriBearingY, face),
TT2PDF(iHoriBearingX + FXFT_Get_Glyph_Width(face), face),
TT2PDF(iHoriBearingY - FXFT_Get_Glyph_Height(face), face));
if (m_bUseFontWidth) {
int TT_Width = TT2PDF(FXFT_Get_Glyph_HoriAdvance(face), face);
if (m_CharWidth[charcode] == 0xffff) {
m_CharWidth[charcode] = TT_Width;
} else if (TT_Width && !IsEmbedded()) {
m_CharBBox[charcode].right =
m_CharBBox[charcode].right * m_CharWidth[charcode] / TT_Width;
m_CharBBox[charcode].left =
m_CharBBox[charcode].left * m_CharWidth[charcode] / TT_Width;
}
}
}
void CPDF_SimpleFont::LoadPDFEncoding(bool bEmbedded, bool bTrueType) {
const CPDF_Object* pEncoding = m_pFontDict->GetDirectObjectFor("Encoding");
if (!pEncoding) {
if (m_BaseFontName == "Symbol") {
m_BaseEncoding = bTrueType ? PDFFONT_ENCODING_MS_SYMBOL
: PDFFONT_ENCODING_ADOBE_SYMBOL;
} else if (!bEmbedded && m_BaseEncoding == PDFFONT_ENCODING_BUILTIN) {
m_BaseEncoding = PDFFONT_ENCODING_WINANSI;
}
return;
}
if (pEncoding->IsName()) {
if (m_BaseEncoding == PDFFONT_ENCODING_ADOBE_SYMBOL ||
m_BaseEncoding == PDFFONT_ENCODING_ZAPFDINGBATS) {
return;
}
if (FontStyleIsSymbolic(m_Flags) && m_BaseFontName == "Symbol") {
if (!bTrueType)
m_BaseEncoding = PDFFONT_ENCODING_ADOBE_SYMBOL;
return;
}
ByteString bsEncoding = pEncoding->GetString();
if (bsEncoding.Compare("MacExpertEncoding") == 0) {
bsEncoding = "WinAnsiEncoding";
}
GetPredefinedEncoding(bsEncoding, &m_BaseEncoding);
return;
}
const CPDF_Dictionary* pDict = pEncoding->AsDictionary();
if (!pDict)
return;
if (m_BaseEncoding != PDFFONT_ENCODING_ADOBE_SYMBOL &&
m_BaseEncoding != PDFFONT_ENCODING_ZAPFDINGBATS) {
ByteString bsEncoding = pDict->GetStringFor("BaseEncoding");
if (bTrueType && bsEncoding.Compare("MacExpertEncoding") == 0)
bsEncoding = "WinAnsiEncoding";
GetPredefinedEncoding(bsEncoding, &m_BaseEncoding);
}
if ((!bEmbedded || bTrueType) && m_BaseEncoding == PDFFONT_ENCODING_BUILTIN)
m_BaseEncoding = PDFFONT_ENCODING_STANDARD;
const CPDF_Array* pDiffs = pDict->GetArrayFor("Differences");
if (!pDiffs)
return;
m_CharNames.resize(256);
uint32_t cur_code = 0;
for (uint32_t i = 0; i < pDiffs->size(); i++) {
const CPDF_Object* pElement = pDiffs->GetDirectObjectAt(i);
if (!pElement)
continue;
const CPDF_Name* pName = pElement->AsName();
if (pName) {
if (cur_code < 256)
m_CharNames[cur_code] = pName->GetString();
cur_code++;
} else {
cur_code = pElement->GetInteger();
}
}
}
int CPDF_SimpleFont::GetCharWidthF(uint32_t charcode) {
if (charcode > 0xff)
charcode = 0;
if (m_CharWidth[charcode] == 0xffff) {
LoadCharMetrics(charcode);
if (m_CharWidth[charcode] == 0xffff) {
m_CharWidth[charcode] = 0;
}
}
return m_CharWidth[charcode];
}
FX_RECT CPDF_SimpleFont::GetCharBBox(uint32_t charcode) {
if (charcode > 0xff)
charcode = 0;
if (m_CharBBox[charcode].left == -1)
LoadCharMetrics(charcode);
return m_CharBBox[charcode];
}
bool CPDF_SimpleFont::LoadCommon() {
const CPDF_Dictionary* pFontDesc = m_pFontDict->GetDictFor("FontDescriptor");
if (pFontDesc) {
LoadFontDescriptor(pFontDesc);
}
const CPDF_Array* pWidthArray = m_pFontDict->GetArrayFor("Widths");
m_bUseFontWidth = !pWidthArray;
if (pWidthArray) {
if (pFontDesc && pFontDesc->KeyExist("MissingWidth")) {
int MissingWidth = pFontDesc->GetIntegerFor("MissingWidth");
for (int i = 0; i < 256; i++) {
m_CharWidth[i] = MissingWidth;
}
}
size_t width_start = m_pFontDict->GetIntegerFor("FirstChar", 0);
size_t width_end = m_pFontDict->GetIntegerFor("LastChar", 0);
if (width_start <= 255) {
if (width_end == 0 || width_end >= width_start + pWidthArray->size())
width_end = width_start + pWidthArray->size() - 1;
if (width_end > 255)
width_end = 255;
for (size_t i = width_start; i <= width_end; i++)
m_CharWidth[i] = pWidthArray->GetIntegerAt(i - width_start);
}
}
if (m_pFontFile) {
if (m_BaseFontName.GetLength() > 8 && m_BaseFontName[7] == '+')
m_BaseFontName = m_BaseFontName.Last(m_BaseFontName.GetLength() - 8);
} else {
LoadSubstFont();
}
if (!FontStyleIsSymbolic(m_Flags))
m_BaseEncoding = PDFFONT_ENCODING_STANDARD;
LoadPDFEncoding(!!m_pFontFile, m_Font.IsTTFont());
LoadGlyphMap();
m_CharNames.clear();
if (!m_Font.GetFaceRec())
return true;
if (FontStyleIsAllCaps(m_Flags)) {
static const unsigned char kLowercases[][2] = {
{'a', 'z'}, {0xe0, 0xf6}, {0xf8, 0xfd}};
for (size_t range = 0; range < pdfium::size(kLowercases); ++range) {
const auto& lower = kLowercases[range];
for (int i = lower[0]; i <= lower[1]; ++i) {
if (m_GlyphIndex[i] != 0xffff && m_pFontFile)
continue;
int j = i - 32;
m_GlyphIndex[i] = m_GlyphIndex[j];
if (m_CharWidth[j]) {
m_CharWidth[i] = m_CharWidth[j];
m_CharBBox[i] = m_CharBBox[j];
}
}
}
}
CheckFontMetrics();
return true;
}
void CPDF_SimpleFont::LoadSubstFont() {
if (!m_bUseFontWidth && !FontStyleIsFixedPitch(m_Flags)) {
int width = 0, i;
for (i = 0; i < 256; i++) {
if (m_CharWidth[i] == 0 || m_CharWidth[i] == 0xffff)
continue;
if (width == 0)
width = m_CharWidth[i];
else if (width != m_CharWidth[i])
break;
}
if (i == 256 && width)
m_Flags |= FXFONT_FIXED_PITCH;
}
m_Font.LoadSubst(m_BaseFontName, IsTrueTypeFont(), m_Flags, GetFontWeight(),
m_ItalicAngle, 0, false);
}
bool CPDF_SimpleFont::IsUnicodeCompatible() const {
return m_BaseEncoding != PDFFONT_ENCODING_BUILTIN &&
m_BaseEncoding != PDFFONT_ENCODING_ADOBE_SYMBOL &&
m_BaseEncoding != PDFFONT_ENCODING_ZAPFDINGBATS;
}
WideString CPDF_SimpleFont::UnicodeFromCharCode(uint32_t charcode) const {
WideString unicode = CPDF_Font::UnicodeFromCharCode(charcode);
if (!unicode.IsEmpty())
return unicode;
wchar_t ret = m_Encoding.UnicodeFromCharCode((uint8_t)charcode);
if (ret == 0)
return WideString();
return ret;
}
uint32_t CPDF_SimpleFont::CharCodeFromUnicode(wchar_t unicode) const {
uint32_t ret = CPDF_Font::CharCodeFromUnicode(unicode);
if (ret)
return ret;
return m_Encoding.CharCodeFromUnicode(unicode);
}
bool CPDF_SimpleFont::HasFontWidths() const {
return !m_bUseFontWidth;
}