blob: 5ce4be65a1556ea322445f01df39d9bb295d84dd [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.
#include <memory>
#include <utility>
#include "core/fpdfapi/cpdf_modulemgr.h"
#include "core/fpdfapi/font/cpdf_font.h"
#include "core/fpdfapi/font/cpdf_type1font.h"
#include "core/fpdfapi/page/cpdf_textobject.h"
#include "core/fpdfapi/parser/cpdf_array.h"
#include "core/fpdfapi/parser/cpdf_dictionary.h"
#include "core/fpdfapi/parser/cpdf_document.h"
#include "core/fpdfapi/parser/cpdf_name.h"
#include "core/fpdfapi/parser/cpdf_number.h"
#include "core/fpdfapi/parser/cpdf_reference.h"
#include "core/fpdfapi/parser/cpdf_stream.h"
#include "core/fxge/cfx_fontmgr.h"
#include "core/fxge/fx_font.h"
#include "fpdfsdk/fsdk_define.h"
#include "public/fpdf_edit.h"
namespace {
CPDF_Dictionary* LoadFontDesc(CPDF_Document* pDoc,
const CFX_ByteString& font_name,
CFX_Font* pFont,
const uint8_t* data,
uint32_t size,
int font_type) {
CPDF_Dictionary* fontDesc = pDoc->NewIndirect<CPDF_Dictionary>();
fontDesc->SetNewFor<CPDF_Name>("Type", "FontDescriptor");
fontDesc->SetNewFor<CPDF_Name>("FontName", font_name);
int flags = 0;
if (FXFT_Is_Face_fixedwidth(pFont->GetFace()))
flags |= FXFONT_FIXED_PITCH;
if (font_name.Find("Serif") > -1)
flags |= FXFONT_SERIF;
if (FXFT_Is_Face_Italic(pFont->GetFace()))
flags |= FXFONT_ITALIC;
if (FXFT_Is_Face_Bold(pFont->GetFace()))
flags |= FXFONT_BOLD;
// TODO(npm): How do I know if a font is symbolic, script, allcap, smallcap
flags |= FXFONT_NONSYMBOLIC;
fontDesc->SetNewFor<CPDF_Number>("Flags", flags);
FX_RECT bbox;
pFont->GetBBox(bbox);
auto pBBox = pdfium::MakeUnique<CPDF_Array>();
pBBox->AddNew<CPDF_Number>(bbox.left);
pBBox->AddNew<CPDF_Number>(bbox.bottom);
pBBox->AddNew<CPDF_Number>(bbox.right);
pBBox->AddNew<CPDF_Number>(bbox.top);
fontDesc->SetFor("FontBBox", std::move(pBBox));
// TODO(npm): calculate italic angle correctly
fontDesc->SetNewFor<CPDF_Number>("ItalicAngle", pFont->IsItalic() ? -12 : 0);
fontDesc->SetNewFor<CPDF_Number>("Ascent", pFont->GetAscent());
fontDesc->SetNewFor<CPDF_Number>("Descent", pFont->GetDescent());
// TODO(npm): calculate the capheight, stemV correctly
fontDesc->SetNewFor<CPDF_Number>("CapHeight", pFont->GetAscent());
fontDesc->SetNewFor<CPDF_Number>("StemV", pFont->IsBold() ? 120 : 70);
CPDF_Stream* pStream = pDoc->NewIndirect<CPDF_Stream>();
pStream->SetData(data, size);
CFX_ByteString fontFile =
font_type == FPDF_FONT_TYPE1 ? "FontFile" : "FontFile2";
fontDesc->SetNewFor<CPDF_Reference>(fontFile, pDoc, pStream->GetObjNum());
return fontDesc;
}
void* LoadSimpleFont(CPDF_Document* pDoc,
std::unique_ptr<CFX_Font> pFont,
const uint8_t* data,
uint32_t size,
int font_type) {
CPDF_Dictionary* fontDict = pDoc->NewIndirect<CPDF_Dictionary>();
fontDict->SetNewFor<CPDF_Name>("Type", "Font");
fontDict->SetNewFor<CPDF_Name>(
"Subtype", font_type == FPDF_FONT_TYPE1 ? "Type1" : "TrueType");
CFX_ByteString name = pFont->GetFaceName();
if (name.IsEmpty())
name = "Unnamed";
fontDict->SetNewFor<CPDF_Name>("BaseFont", name);
uint32_t glyphIndex;
int currentChar = FXFT_Get_First_Char(pFont->GetFace(), &glyphIndex);
fontDict->SetNewFor<CPDF_Number>("FirstChar", currentChar);
CPDF_Array* widthsArray = pDoc->NewIndirect<CPDF_Array>();
while (true) {
int width = pFont->GetGlyphWidth(glyphIndex);
widthsArray->AddNew<CPDF_Number>(width);
int nextChar =
FXFT_Get_Next_Char(pFont->GetFace(), currentChar, &glyphIndex);
// Simple fonts have 1-byte charcodes only.
if (nextChar > 0xff || glyphIndex == 0)
break;
for (int i = currentChar + 1; i < nextChar; i++)
widthsArray->AddNew<CPDF_Number>(0);
currentChar = nextChar;
}
fontDict->SetNewFor<CPDF_Number>("LastChar", currentChar);
fontDict->SetNewFor<CPDF_Reference>("Widths", pDoc, widthsArray->GetObjNum());
CPDF_Dictionary* fontDesc =
LoadFontDesc(pDoc, name, pFont.get(), data, size, font_type);
fontDict->SetNewFor<CPDF_Reference>("FontDescriptor", pDoc,
fontDesc->GetObjNum());
return pDoc->LoadFont(fontDict);
}
void* LoadCompositeFont(CPDF_Document* pDoc,
std::unique_ptr<CFX_Font> pFont,
const uint8_t* data,
uint32_t size,
int font_type) {
CPDF_Dictionary* fontDict = pDoc->NewIndirect<CPDF_Dictionary>();
fontDict->SetNewFor<CPDF_Name>("Type", "Font");
fontDict->SetNewFor<CPDF_Name>("Subtype", "Type0");
// TODO(npm): Get the correct encoding, if it's not identity.
CFX_ByteString encoding = "Identity-H";
fontDict->SetNewFor<CPDF_Name>("Encoding", encoding);
CFX_ByteString name = pFont->GetFaceName();
if (name.IsEmpty())
name = "Unnamed";
fontDict->SetNewFor<CPDF_Name>(
"BaseFont", font_type == FPDF_FONT_TYPE1 ? name + "-" + encoding : name);
CPDF_Dictionary* pCIDFont = pDoc->NewIndirect<CPDF_Dictionary>();
pCIDFont->SetNewFor<CPDF_Name>("Type", "Font");
pCIDFont->SetNewFor<CPDF_Name>("Subtype", font_type == FPDF_FONT_TYPE1
? "CIDFontType0"
: "CIDFontType2");
pCIDFont->SetNewFor<CPDF_Name>("BaseFont", name);
// TODO(npm): Maybe use FT_Get_CID_Registry_Ordering_Supplement to get the
// CIDSystemInfo
CPDF_Dictionary* pCIDSystemInfo = pDoc->NewIndirect<CPDF_Dictionary>();
pCIDSystemInfo->SetNewFor<CPDF_Name>("Registry", "Adobe");
pCIDSystemInfo->SetNewFor<CPDF_Name>("Ordering", "Identity");
pCIDSystemInfo->SetNewFor<CPDF_Number>("Supplement", 0);
pCIDFont->SetNewFor<CPDF_Reference>("CIDSystemInfo", pDoc,
pCIDSystemInfo->GetObjNum());
CPDF_Dictionary* fontDesc =
LoadFontDesc(pDoc, name, pFont.get(), data, size, font_type);
pCIDFont->SetNewFor<CPDF_Reference>("FontDescriptor", pDoc,
fontDesc->GetObjNum());
uint32_t glyphIndex;
int currentChar = FXFT_Get_First_Char(pFont->GetFace(), &glyphIndex);
CPDF_Array* widthsArray = pDoc->NewIndirect<CPDF_Array>();
while (true) {
int width = pFont->GetGlyphWidth(glyphIndex);
int nextChar =
FXFT_Get_Next_Char(pFont->GetFace(), currentChar, &glyphIndex);
if (glyphIndex == 0) {
// Only one char left, use format c [w]
auto oneW = pdfium::MakeUnique<CPDF_Array>();
oneW->AddNew<CPDF_Number>(width);
widthsArray->AddNew<CPDF_Number>(currentChar);
widthsArray->Add(std::move(oneW));
break;
}
int nextWidth = pFont->GetGlyphWidth(glyphIndex);
if (nextChar == currentChar + 1 && nextWidth == width) {
// The array can have a group c_first c_last w: all CIDs in the range from
// c_first to c_last will have width w
widthsArray->AddNew<CPDF_Number>(currentChar);
currentChar = nextChar;
while (true) {
nextChar =
FXFT_Get_Next_Char(pFont->GetFace(), currentChar, &glyphIndex);
if (glyphIndex == 0)
break;
nextWidth = pFont->GetGlyphWidth(glyphIndex);
if (nextChar != currentChar + 1 || nextWidth != width)
break;
currentChar = nextChar;
}
widthsArray->AddNew<CPDF_Number>(currentChar);
widthsArray->AddNew<CPDF_Number>(width);
} else {
// Otherwise we can have a group of the form c [w1 w2 ...]: c has width
// w1, c+1 has width w2, etc.
widthsArray->AddNew<CPDF_Number>(currentChar);
auto curWidthArray = pdfium::MakeUnique<CPDF_Array>();
curWidthArray->AddNew<CPDF_Number>(width);
while (nextChar == currentChar + 1) {
curWidthArray->AddNew<CPDF_Number>(nextWidth);
currentChar = nextChar;
nextChar =
FXFT_Get_Next_Char(pFont->GetFace(), currentChar, &glyphIndex);
if (glyphIndex == 0)
break;
nextWidth = pFont->GetGlyphWidth(glyphIndex);
}
widthsArray->Add(std::move(curWidthArray));
}
if (glyphIndex == 0)
break;
currentChar = nextChar;
}
pCIDFont->SetNewFor<CPDF_Reference>("W", pDoc, widthsArray->GetObjNum());
// TODO(npm): Support vertical writing
auto pDescendant = pdfium::MakeUnique<CPDF_Array>();
pDescendant->AddNew<CPDF_Reference>(pDoc, pCIDFont->GetObjNum());
fontDict->SetFor("DescendantFonts", std::move(pDescendant));
// TODO(npm): do we need a ToUnicode?
return pDoc->LoadFont(fontDict);
}
} // namespace
DLLEXPORT FPDF_PAGEOBJECT STDCALL FPDFPageObj_NewTextObj(FPDF_DOCUMENT document,
FPDF_BYTESTRING font,
float font_size) {
CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document);
if (!pDoc)
return nullptr;
CPDF_Font* pFont = CPDF_Font::GetStockFont(pDoc, CFX_ByteStringC(font));
if (!pFont)
return nullptr;
CPDF_TextObject* pTextObj = new CPDF_TextObject;
pTextObj->m_TextState.SetFont(pFont);
pTextObj->m_TextState.SetFontSize(font_size);
pTextObj->DefaultStates();
return pTextObj;
}
DLLEXPORT FPDF_BOOL STDCALL FPDFText_SetText(FPDF_PAGEOBJECT text_object,
FPDF_BYTESTRING text) {
if (!text_object)
return false;
auto pTextObj = reinterpret_cast<CPDF_TextObject*>(text_object);
pTextObj->SetText(CFX_ByteString(text));
return true;
}
DLLEXPORT FPDF_FONT STDCALL FPDFText_LoadFont(FPDF_DOCUMENT document,
const uint8_t* data,
uint32_t size,
int font_type,
FPDF_BOOL cid) {
CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document);
if (!pDoc || !data || size == 0 ||
(font_type != FPDF_FONT_TYPE1 && font_type != FPDF_FONT_TRUETYPE)) {
return nullptr;
}
auto pFont = pdfium::MakeUnique<CFX_Font>();
// TODO(npm): Maybe use FT_Get_X11_Font_Format to check format? Otherwise, we
// are allowing giving any font that can be loaded on freetype and setting it
// as any font type.
if (!pFont->LoadEmbedded(data, size))
return nullptr;
return cid ? LoadCompositeFont(pDoc, std::move(pFont), data, size, font_type)
: LoadSimpleFont(pDoc, std::move(pFont), data, size, font_type);
}