blob: 22c6266ec19d9edb0e83d3a5ce448e43ba8a8190 [file] [log] [blame]
Nicolas Pena49058402017-02-14 18:26:20 -05001// Copyright 2017 PDFium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Nicolas Penaf45ade32017-05-03 10:23:49 -04005#include <map>
Nicolas Penad03ca422017-03-06 13:54:33 -05006#include <memory>
Nicolas Penabe90aae2017-02-27 10:41:41 -05007#include <utility>
Nicolas Penaf45ade32017-05-03 10:23:49 -04008#include <vector>
Nicolas Pena49058402017-02-14 18:26:20 -05009
10#include "core/fpdfapi/cpdf_modulemgr.h"
11#include "core/fpdfapi/font/cpdf_font.h"
Nicolas Penabe90aae2017-02-27 10:41:41 -050012#include "core/fpdfapi/font/cpdf_type1font.h"
Nicolas Penab3161852017-05-02 14:12:50 -040013#include "core/fpdfapi/page/cpdf_docpagedata.h"
Nicolas Pena49058402017-02-14 18:26:20 -050014#include "core/fpdfapi/page/cpdf_textobject.h"
Nicolas Penabe90aae2017-02-27 10:41:41 -050015#include "core/fpdfapi/parser/cpdf_array.h"
16#include "core/fpdfapi/parser/cpdf_dictionary.h"
17#include "core/fpdfapi/parser/cpdf_document.h"
18#include "core/fpdfapi/parser/cpdf_name.h"
19#include "core/fpdfapi/parser/cpdf_number.h"
20#include "core/fpdfapi/parser/cpdf_reference.h"
21#include "core/fpdfapi/parser/cpdf_stream.h"
Nicolas Pena54b91662017-05-05 16:49:30 -040022#include "core/fxcrt/fx_extension.h"
Nicolas Penabe90aae2017-02-27 10:41:41 -050023#include "core/fxge/cfx_fontmgr.h"
24#include "core/fxge/fx_font.h"
Nicolas Pena49058402017-02-14 18:26:20 -050025#include "fpdfsdk/fsdk_define.h"
Nicolas Penabe90aae2017-02-27 10:41:41 -050026#include "public/fpdf_edit.h"
Nicolas Pena49058402017-02-14 18:26:20 -050027
Nicolas Penad03ca422017-03-06 13:54:33 -050028namespace {
29
30CPDF_Dictionary* LoadFontDesc(CPDF_Document* pDoc,
Ryan Harrison275e2602017-09-18 14:23:18 -040031 const ByteString& font_name,
Nicolas Penad03ca422017-03-06 13:54:33 -050032 CFX_Font* pFont,
33 const uint8_t* data,
34 uint32_t size,
35 int font_type) {
36 CPDF_Dictionary* fontDesc = pDoc->NewIndirect<CPDF_Dictionary>();
37 fontDesc->SetNewFor<CPDF_Name>("Type", "FontDescriptor");
38 fontDesc->SetNewFor<CPDF_Name>("FontName", font_name);
39 int flags = 0;
40 if (FXFT_Is_Face_fixedwidth(pFont->GetFace()))
41 flags |= FXFONT_FIXED_PITCH;
Ryan Harrison12db7512017-08-23 10:39:35 -040042 if (font_name.Contains("Serif"))
Nicolas Penad03ca422017-03-06 13:54:33 -050043 flags |= FXFONT_SERIF;
44 if (FXFT_Is_Face_Italic(pFont->GetFace()))
45 flags |= FXFONT_ITALIC;
46 if (FXFT_Is_Face_Bold(pFont->GetFace()))
47 flags |= FXFONT_BOLD;
48
49 // TODO(npm): How do I know if a font is symbolic, script, allcap, smallcap
50 flags |= FXFONT_NONSYMBOLIC;
51
52 fontDesc->SetNewFor<CPDF_Number>("Flags", flags);
53 FX_RECT bbox;
54 pFont->GetBBox(bbox);
55 auto pBBox = pdfium::MakeUnique<CPDF_Array>();
56 pBBox->AddNew<CPDF_Number>(bbox.left);
Nicolas Penad03ca422017-03-06 13:54:33 -050057 pBBox->AddNew<CPDF_Number>(bbox.top);
Nicolás Peña5f95f362017-09-28 13:00:45 +090058 pBBox->AddNew<CPDF_Number>(bbox.right);
59 pBBox->AddNew<CPDF_Number>(bbox.bottom);
Nicolas Penad03ca422017-03-06 13:54:33 -050060 fontDesc->SetFor("FontBBox", std::move(pBBox));
61
62 // TODO(npm): calculate italic angle correctly
63 fontDesc->SetNewFor<CPDF_Number>("ItalicAngle", pFont->IsItalic() ? -12 : 0);
64
65 fontDesc->SetNewFor<CPDF_Number>("Ascent", pFont->GetAscent());
66 fontDesc->SetNewFor<CPDF_Number>("Descent", pFont->GetDescent());
67
68 // TODO(npm): calculate the capheight, stemV correctly
69 fontDesc->SetNewFor<CPDF_Number>("CapHeight", pFont->GetAscent());
70 fontDesc->SetNewFor<CPDF_Number>("StemV", pFont->IsBold() ? 120 : 70);
71
72 CPDF_Stream* pStream = pDoc->NewIndirect<CPDF_Stream>();
73 pStream->SetData(data, size);
Nicolás Peña79eab232017-09-28 13:29:05 +090074 // TODO(npm): Lengths for Type1 fonts.
75 if (font_type == FPDF_FONT_TRUETYPE) {
76 pStream->GetDict()->SetNewFor<CPDF_Number>("Length1",
77 static_cast<int>(size));
78 }
Ryan Harrison275e2602017-09-18 14:23:18 -040079 ByteString fontFile = font_type == FPDF_FONT_TYPE1 ? "FontFile" : "FontFile2";
Nicolas Penad03ca422017-03-06 13:54:33 -050080 fontDesc->SetNewFor<CPDF_Reference>(fontFile, pDoc, pStream->GetObjNum());
81 return fontDesc;
82}
83
Nicolas Penaf45ade32017-05-03 10:23:49 -040084const char ToUnicodeStart[] =
85 "/CIDInit /ProcSet findresource begin\n"
86 "12 dict begin\n"
87 "begincmap\n"
88 "/CIDSystemInfo\n"
89 "<</Registry (Adobe)\n"
90 "/Ordering (Identity)\n"
91 "/Supplement 0\n"
92 ">> def\n"
93 "/CMapName /Adobe-Identity-H def\n"
94 "CMapType 2 def\n"
95 "1 begincodespacerange\n"
Nicolas Penab48912f2017-05-17 14:45:52 -040096 "<0000> <FFFFF>\n"
97 "endcodespacerange\n";
98
99const char ToUnicodeEnd[] =
100 "endcmap\n"
101 "CMapName currentdict /CMap defineresource pop\n"
102 "end\n"
103 "end\n";
Nicolas Penaf45ade32017-05-03 10:23:49 -0400104
Henrique Nakashima0ba3c6d2017-07-05 16:21:17 -0400105void AddCharcode(std::ostringstream* pBuffer, uint32_t number) {
Nicolas Pena54b91662017-05-05 16:49:30 -0400106 ASSERT(number <= 0xFFFF);
Nicolas Penaf45ade32017-05-03 10:23:49 -0400107 *pBuffer << "<";
108 char ans[4];
Nicolas Pena54b91662017-05-05 16:49:30 -0400109 FXSYS_IntToFourHexChars(number, ans);
Nicolas Penaf45ade32017-05-03 10:23:49 -0400110 for (size_t i = 0; i < 4; ++i)
Henrique Nakashima0ba3c6d2017-07-05 16:21:17 -0400111 *pBuffer << ans[i];
Nicolas Penaf45ade32017-05-03 10:23:49 -0400112 *pBuffer << ">";
113}
114
Nicolas Pena54b91662017-05-05 16:49:30 -0400115// PDF spec 1.7 Section 5.9.2: "Unicode character sequences as expressed in
116// UTF-16BE encoding." See https://en.wikipedia.org/wiki/UTF-16#Description
Henrique Nakashima0ba3c6d2017-07-05 16:21:17 -0400117void AddUnicode(std::ostringstream* pBuffer, uint32_t unicode) {
Nicolas Pena217644c2017-07-06 10:53:01 -0400118 if (unicode >= 0xD800 && unicode <= 0xDFFF)
119 unicode = 0;
120
Nicolas Pena54b91662017-05-05 16:49:30 -0400121 char ans[8];
122 *pBuffer << "<";
123 size_t numChars = FXSYS_ToUTF16BE(unicode, ans);
124 for (size_t i = 0; i < numChars; ++i)
Henrique Nakashima0ba3c6d2017-07-05 16:21:17 -0400125 *pBuffer << ans[i];
Nicolas Pena54b91662017-05-05 16:49:30 -0400126 *pBuffer << ">";
127}
128
Nicolas Penaf45ade32017-05-03 10:23:49 -0400129// Loads the charcode to unicode mapping into a stream
130CPDF_Stream* LoadUnicode(CPDF_Document* pDoc,
131 const std::map<uint32_t, uint32_t>& to_unicode) {
Nicolas Penaf45ade32017-05-03 10:23:49 -0400132 // A map charcode->unicode
133 std::map<uint32_t, uint32_t> char_to_uni;
134 // A map <char_start, char_end> to vector v of unicode characters of size (end
135 // - start + 1). This abbreviates: start->v[0], start+1->v[1], etc. PDF spec
136 // 1.7 Section 5.9.2 says that only the last byte of the unicode may change.
137 std::map<std::pair<uint32_t, uint32_t>, std::vector<uint32_t>>
138 map_range_vector;
139 // A map <start, end> -> unicode
140 // This abbreviates: start->unicode, start+1->unicode+1, etc.
141 // PDF spec 1.7 Section 5.9.2 says that only the last byte of the unicode may
142 // change.
143 std::map<std::pair<uint32_t, uint32_t>, uint32_t> map_range;
144
145 // Calculate the maps
146 for (auto iter = to_unicode.begin(); iter != to_unicode.end(); ++iter) {
147 uint32_t firstCharcode = iter->first;
148 uint32_t firstUnicode = iter->second;
149 if (std::next(iter) == to_unicode.end() ||
150 firstCharcode + 1 != std::next(iter)->first) {
151 char_to_uni[firstCharcode] = firstUnicode;
152 continue;
153 }
154 ++iter;
155 uint32_t curCharcode = iter->first;
156 uint32_t curUnicode = iter->second;
157 if (curCharcode % 256 == 0) {
158 char_to_uni[firstCharcode] = firstUnicode;
159 char_to_uni[curCharcode] = curUnicode;
160 continue;
161 }
162 const size_t maxExtra = 255 - (curCharcode % 256);
163 auto next_it = std::next(iter);
164 if (firstUnicode + 1 != curUnicode) {
165 // Consecutive charcodes mapping to non-consecutive unicodes
166 std::vector<uint32_t> unicodes;
167 unicodes.push_back(firstUnicode);
168 unicodes.push_back(curUnicode);
169 for (size_t i = 0; i < maxExtra; ++i) {
170 if (next_it == to_unicode.end() || curCharcode + 1 != next_it->first)
171 break;
172 ++iter;
173 ++curCharcode;
174 unicodes.push_back(iter->second);
175 next_it = std::next(iter);
176 }
177 ASSERT(iter->first - firstCharcode + 1 == unicodes.size());
178 map_range_vector[std::make_pair(firstCharcode, iter->first)] = unicodes;
179 continue;
180 }
181 // Consecutive charcodes mapping to consecutive unicodes
182 for (size_t i = 0; i < maxExtra; ++i) {
183 if (next_it == to_unicode.end() || curCharcode + 1 != next_it->first ||
184 curUnicode + 1 != next_it->second) {
185 break;
186 }
187 ++iter;
188 ++curCharcode;
189 ++curUnicode;
190 next_it = std::next(iter);
191 }
192 map_range[std::make_pair(firstCharcode, curCharcode)] = firstUnicode;
193 }
Henrique Nakashima0ba3c6d2017-07-05 16:21:17 -0400194 std::ostringstream buffer;
Nicolas Penab48912f2017-05-17 14:45:52 -0400195 buffer << ToUnicodeStart;
Nicolas Penaf45ade32017-05-03 10:23:49 -0400196 // Add maps to buffer
197 buffer << static_cast<uint32_t>(char_to_uni.size()) << " beginbfchar\n";
Nicolas Pena54b91662017-05-05 16:49:30 -0400198 for (const auto& iter : char_to_uni) {
199 AddCharcode(&buffer, iter.first);
Nicolas Penaf45ade32017-05-03 10:23:49 -0400200 buffer << " ";
Nicolas Pena54b91662017-05-05 16:49:30 -0400201 AddUnicode(&buffer, iter.second);
Nicolas Penaf45ade32017-05-03 10:23:49 -0400202 buffer << "\n";
203 }
204 buffer << "endbfchar\n"
205 << static_cast<uint32_t>(map_range_vector.size() + map_range.size())
206 << " beginbfrange\n";
Nicolas Pena54b91662017-05-05 16:49:30 -0400207 for (const auto& iter : map_range_vector) {
Nicolas Penaf45ade32017-05-03 10:23:49 -0400208 const std::pair<uint32_t, uint32_t>& charcodeRange = iter.first;
Nicolas Pena54b91662017-05-05 16:49:30 -0400209 AddCharcode(&buffer, charcodeRange.first);
Nicolas Penaf45ade32017-05-03 10:23:49 -0400210 buffer << " ";
Nicolas Pena54b91662017-05-05 16:49:30 -0400211 AddCharcode(&buffer, charcodeRange.second);
Nicolas Penaf45ade32017-05-03 10:23:49 -0400212 buffer << " [";
213 const std::vector<uint32_t>& unicodes = iter.second;
214 for (size_t i = 0; i < unicodes.size(); ++i) {
215 uint32_t uni = unicodes[i];
Nicolas Pena54b91662017-05-05 16:49:30 -0400216 AddUnicode(&buffer, uni);
Nicolas Penaf45ade32017-05-03 10:23:49 -0400217 if (i != unicodes.size() - 1)
218 buffer << " ";
219 }
220 buffer << "]\n";
221 }
Nicolas Pena54b91662017-05-05 16:49:30 -0400222 for (const auto& iter : map_range) {
Nicolas Penaf45ade32017-05-03 10:23:49 -0400223 const std::pair<uint32_t, uint32_t>& charcodeRange = iter.first;
Nicolas Pena54b91662017-05-05 16:49:30 -0400224 AddCharcode(&buffer, charcodeRange.first);
Nicolas Penaf45ade32017-05-03 10:23:49 -0400225 buffer << " ";
Nicolas Pena54b91662017-05-05 16:49:30 -0400226 AddCharcode(&buffer, charcodeRange.second);
Nicolas Penaf45ade32017-05-03 10:23:49 -0400227 buffer << " ";
Nicolas Pena54b91662017-05-05 16:49:30 -0400228 AddUnicode(&buffer, iter.second);
Nicolas Penaf45ade32017-05-03 10:23:49 -0400229 buffer << "\n";
230 }
Nicolas Penab48912f2017-05-17 14:45:52 -0400231 buffer << "endbfrange\n";
232 buffer << ToUnicodeEnd;
Nicolas Penaf45ade32017-05-03 10:23:49 -0400233 // TODO(npm): Encrypt / Compress?
Henrique Nakashima0ba3c6d2017-07-05 16:21:17 -0400234 CPDF_Stream* stream = pDoc->NewIndirect<CPDF_Stream>();
235 stream->SetData(&buffer);
236 return stream;
Nicolas Penaf45ade32017-05-03 10:23:49 -0400237}
238
Nicolas Penab83d8702017-06-09 17:55:51 -0400239const uint32_t kMaxSimpleFontChar = 0xFF;
240
Nicolas Penad03ca422017-03-06 13:54:33 -0500241void* LoadSimpleFont(CPDF_Document* pDoc,
242 std::unique_ptr<CFX_Font> pFont,
243 const uint8_t* data,
244 uint32_t size,
245 int font_type) {
246 CPDF_Dictionary* fontDict = pDoc->NewIndirect<CPDF_Dictionary>();
247 fontDict->SetNewFor<CPDF_Name>("Type", "Font");
248 fontDict->SetNewFor<CPDF_Name>(
249 "Subtype", font_type == FPDF_FONT_TYPE1 ? "Type1" : "TrueType");
Ryan Harrison275e2602017-09-18 14:23:18 -0400250 ByteString name = pFont->GetFaceName();
Nicolas Penad03ca422017-03-06 13:54:33 -0500251 if (name.IsEmpty())
252 name = "Unnamed";
253 fontDict->SetNewFor<CPDF_Name>("BaseFont", name);
254
255 uint32_t glyphIndex;
Nicolas Penab83d8702017-06-09 17:55:51 -0400256 uint32_t currentChar = FXFT_Get_First_Char(pFont->GetFace(), &glyphIndex);
257 if (currentChar > kMaxSimpleFontChar || glyphIndex == 0)
258 return nullptr;
259 fontDict->SetNewFor<CPDF_Number>("FirstChar", static_cast<int>(currentChar));
Nicolas Penad03ca422017-03-06 13:54:33 -0500260 CPDF_Array* widthsArray = pDoc->NewIndirect<CPDF_Array>();
261 while (true) {
Nicolas Penaf45ade32017-05-03 10:23:49 -0400262 widthsArray->AddNew<CPDF_Number>(pFont->GetGlyphWidth(glyphIndex));
Nicolas Penab83d8702017-06-09 17:55:51 -0400263 uint32_t nextChar =
Nicolas Penad03ca422017-03-06 13:54:33 -0500264 FXFT_Get_Next_Char(pFont->GetFace(), currentChar, &glyphIndex);
265 // Simple fonts have 1-byte charcodes only.
Nicolas Penab83d8702017-06-09 17:55:51 -0400266 if (nextChar > kMaxSimpleFontChar || glyphIndex == 0)
Nicolas Penad03ca422017-03-06 13:54:33 -0500267 break;
Nicolas Penab83d8702017-06-09 17:55:51 -0400268 for (uint32_t i = currentChar + 1; i < nextChar; i++)
Nicolas Penad03ca422017-03-06 13:54:33 -0500269 widthsArray->AddNew<CPDF_Number>(0);
270 currentChar = nextChar;
271 }
Nicolas Penab83d8702017-06-09 17:55:51 -0400272 fontDict->SetNewFor<CPDF_Number>("LastChar", static_cast<int>(currentChar));
Nicolas Penad03ca422017-03-06 13:54:33 -0500273 fontDict->SetNewFor<CPDF_Reference>("Widths", pDoc, widthsArray->GetObjNum());
274 CPDF_Dictionary* fontDesc =
275 LoadFontDesc(pDoc, name, pFont.get(), data, size, font_type);
276
277 fontDict->SetNewFor<CPDF_Reference>("FontDescriptor", pDoc,
278 fontDesc->GetObjNum());
279 return pDoc->LoadFont(fontDict);
280}
281
Nicolas Penab83d8702017-06-09 17:55:51 -0400282const uint32_t kMaxUnicode = 0x10FFFF;
283
Nicolas Penad03ca422017-03-06 13:54:33 -0500284void* LoadCompositeFont(CPDF_Document* pDoc,
285 std::unique_ptr<CFX_Font> pFont,
286 const uint8_t* data,
287 uint32_t size,
288 int font_type) {
289 CPDF_Dictionary* fontDict = pDoc->NewIndirect<CPDF_Dictionary>();
290 fontDict->SetNewFor<CPDF_Name>("Type", "Font");
291 fontDict->SetNewFor<CPDF_Name>("Subtype", "Type0");
292 // TODO(npm): Get the correct encoding, if it's not identity.
Ryan Harrison275e2602017-09-18 14:23:18 -0400293 ByteString encoding = "Identity-H";
Nicolas Penad03ca422017-03-06 13:54:33 -0500294 fontDict->SetNewFor<CPDF_Name>("Encoding", encoding);
Ryan Harrison275e2602017-09-18 14:23:18 -0400295 ByteString name = pFont->GetFaceName();
Nicolas Penad03ca422017-03-06 13:54:33 -0500296 if (name.IsEmpty())
297 name = "Unnamed";
298 fontDict->SetNewFor<CPDF_Name>(
299 "BaseFont", font_type == FPDF_FONT_TYPE1 ? name + "-" + encoding : name);
300
301 CPDF_Dictionary* pCIDFont = pDoc->NewIndirect<CPDF_Dictionary>();
302 pCIDFont->SetNewFor<CPDF_Name>("Type", "Font");
303 pCIDFont->SetNewFor<CPDF_Name>("Subtype", font_type == FPDF_FONT_TYPE1
304 ? "CIDFontType0"
305 : "CIDFontType2");
306 pCIDFont->SetNewFor<CPDF_Name>("BaseFont", name);
307
308 // TODO(npm): Maybe use FT_Get_CID_Registry_Ordering_Supplement to get the
309 // CIDSystemInfo
310 CPDF_Dictionary* pCIDSystemInfo = pDoc->NewIndirect<CPDF_Dictionary>();
311 pCIDSystemInfo->SetNewFor<CPDF_Name>("Registry", "Adobe");
312 pCIDSystemInfo->SetNewFor<CPDF_Name>("Ordering", "Identity");
313 pCIDSystemInfo->SetNewFor<CPDF_Number>("Supplement", 0);
314 pCIDFont->SetNewFor<CPDF_Reference>("CIDSystemInfo", pDoc,
315 pCIDSystemInfo->GetObjNum());
316
317 CPDF_Dictionary* fontDesc =
318 LoadFontDesc(pDoc, name, pFont.get(), data, size, font_type);
319 pCIDFont->SetNewFor<CPDF_Reference>("FontDescriptor", pDoc,
320 fontDesc->GetObjNum());
321
322 uint32_t glyphIndex;
Nicolas Penab83d8702017-06-09 17:55:51 -0400323 uint32_t currentChar = FXFT_Get_First_Char(pFont->GetFace(), &glyphIndex);
Nicolas Penaf45ade32017-05-03 10:23:49 -0400324 // If it doesn't have a single char, just fail
Nicolas Penab83d8702017-06-09 17:55:51 -0400325 if (glyphIndex == 0 || currentChar > kMaxUnicode)
Nicolas Penaf45ade32017-05-03 10:23:49 -0400326 return nullptr;
327
328 std::map<uint32_t, uint32_t> to_unicode;
329 std::map<uint32_t, uint32_t> widths;
Nicolas Penad03ca422017-03-06 13:54:33 -0500330 while (true) {
Nicolas Penab83d8702017-06-09 17:55:51 -0400331 if (currentChar > kMaxUnicode)
Nicolas Penafb71fbb2017-05-23 13:16:09 -0400332 break;
333
Nicolas Penaf45ade32017-05-03 10:23:49 -0400334 widths[glyphIndex] = pFont->GetGlyphWidth(glyphIndex);
335 to_unicode[glyphIndex] = currentChar;
336 currentChar =
Nicolas Penad03ca422017-03-06 13:54:33 -0500337 FXFT_Get_Next_Char(pFont->GetFace(), currentChar, &glyphIndex);
Nicolas Penaf45ade32017-05-03 10:23:49 -0400338 if (glyphIndex == 0)
339 break;
340 }
341 CPDF_Array* widthsArray = pDoc->NewIndirect<CPDF_Array>();
342 for (auto it = widths.begin(); it != widths.end(); ++it) {
343 int ch = it->first;
344 int w = it->second;
345 if (std::next(it) == widths.end()) {
Nicolas Penad03ca422017-03-06 13:54:33 -0500346 // Only one char left, use format c [w]
347 auto oneW = pdfium::MakeUnique<CPDF_Array>();
Nicolas Penaf45ade32017-05-03 10:23:49 -0400348 oneW->AddNew<CPDF_Number>(w);
349 widthsArray->AddNew<CPDF_Number>(ch);
Nicolas Penad03ca422017-03-06 13:54:33 -0500350 widthsArray->Add(std::move(oneW));
351 break;
352 }
Nicolas Penaf45ade32017-05-03 10:23:49 -0400353 ++it;
354 int next_ch = it->first;
355 int next_w = it->second;
356 if (next_ch == ch + 1 && next_w == w) {
Nicolas Penad03ca422017-03-06 13:54:33 -0500357 // The array can have a group c_first c_last w: all CIDs in the range from
358 // c_first to c_last will have width w
Nicolas Penaf45ade32017-05-03 10:23:49 -0400359 widthsArray->AddNew<CPDF_Number>(ch);
360 ch = next_ch;
Nicolas Penad03ca422017-03-06 13:54:33 -0500361 while (true) {
Nicolas Penaf45ade32017-05-03 10:23:49 -0400362 auto next_it = std::next(it);
363 if (next_it == widths.end() || next_it->first != it->first + 1 ||
364 next_it->second != it->second) {
Nicolas Penad03ca422017-03-06 13:54:33 -0500365 break;
Nicolas Penaf45ade32017-05-03 10:23:49 -0400366 }
367 ++it;
368 ch = it->first;
Nicolas Penad03ca422017-03-06 13:54:33 -0500369 }
Nicolas Penaf45ade32017-05-03 10:23:49 -0400370 widthsArray->AddNew<CPDF_Number>(ch);
371 widthsArray->AddNew<CPDF_Number>(w);
372 continue;
Nicolas Penad03ca422017-03-06 13:54:33 -0500373 }
Nicolas Penaf45ade32017-05-03 10:23:49 -0400374 // Otherwise we can have a group of the form c [w1 w2 ...]: c has width
375 // w1, c+1 has width w2, etc.
376 widthsArray->AddNew<CPDF_Number>(ch);
377 auto curWidthArray = pdfium::MakeUnique<CPDF_Array>();
378 curWidthArray->AddNew<CPDF_Number>(w);
379 curWidthArray->AddNew<CPDF_Number>(next_w);
380 while (true) {
381 auto next_it = std::next(it);
382 if (next_it == widths.end() || next_it->first != it->first + 1)
383 break;
384 ++it;
385 curWidthArray->AddNew<CPDF_Number>(static_cast<int>(it->second));
386 }
387 widthsArray->Add(std::move(curWidthArray));
Nicolas Penad03ca422017-03-06 13:54:33 -0500388 }
389 pCIDFont->SetNewFor<CPDF_Reference>("W", pDoc, widthsArray->GetObjNum());
390 // TODO(npm): Support vertical writing
391
392 auto pDescendant = pdfium::MakeUnique<CPDF_Array>();
393 pDescendant->AddNew<CPDF_Reference>(pDoc, pCIDFont->GetObjNum());
394 fontDict->SetFor("DescendantFonts", std::move(pDescendant));
Nicolas Penaf45ade32017-05-03 10:23:49 -0400395 CPDF_Stream* toUnicodeStream = LoadUnicode(pDoc, to_unicode);
396 fontDict->SetNewFor<CPDF_Reference>("ToUnicode", pDoc,
397 toUnicodeStream->GetObjNum());
Nicolas Penad03ca422017-03-06 13:54:33 -0500398 return pDoc->LoadFont(fontDict);
399}
400
401} // namespace
402
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400403FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV
404FPDFPageObj_NewTextObj(FPDF_DOCUMENT document,
405 FPDF_BYTESTRING font,
406 float font_size) {
Nicolas Pena49058402017-02-14 18:26:20 -0500407 CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document);
408 if (!pDoc)
409 return nullptr;
410
Ryan Harrison275e2602017-09-18 14:23:18 -0400411 CPDF_Font* pFont = CPDF_Font::GetStockFont(pDoc, ByteStringView(font));
Nicolas Pena49058402017-02-14 18:26:20 -0500412 if (!pFont)
413 return nullptr;
414
Tom Sepezfe91c6c2017-05-16 15:33:20 -0700415 auto pTextObj = pdfium::MakeUnique<CPDF_TextObject>();
Nicolas Pena49058402017-02-14 18:26:20 -0500416 pTextObj->m_TextState.SetFont(pFont);
417 pTextObj->m_TextState.SetFontSize(font_size);
418 pTextObj->DefaultStates();
Tom Sepezfe91c6c2017-05-16 15:33:20 -0700419 return pTextObj.release(); // Caller takes ownership.
Nicolas Pena49058402017-02-14 18:26:20 -0500420}
421
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400422FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
423FPDFText_SetText(FPDF_PAGEOBJECT text_object, FPDF_WIDESTRING text) {
Nicolas Pena54b91662017-05-05 16:49:30 -0400424 auto* pTextObj = static_cast<CPDF_TextObject*>(text_object);
425 if (!pTextObj)
Nicolas Pena49058402017-02-14 18:26:20 -0500426 return false;
427
Ryan Harrison875e98c2017-09-27 10:53:11 -0400428 size_t len = WideString::WStringLength(text);
Ryan Harrison275e2602017-09-18 14:23:18 -0400429 WideString encodedText = WideString::FromUTF16LE(text, len);
430 ByteString byteText;
Nicolas Penaf45ade32017-05-03 10:23:49 -0400431 for (wchar_t wc : encodedText) {
432 pTextObj->GetFont()->AppendChar(
433 &byteText, pTextObj->GetFont()->CharCodeFromUnicode(wc));
Nicolas Penab3161852017-05-02 14:12:50 -0400434 }
435 pTextObj->SetText(byteText);
Nicolas Pena49058402017-02-14 18:26:20 -0500436 return true;
Nicolas Penabe90aae2017-02-27 10:41:41 -0500437}
438
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400439FPDF_EXPORT FPDF_FONT FPDF_CALLCONV FPDFText_LoadFont(FPDF_DOCUMENT document,
440 const uint8_t* data,
441 uint32_t size,
442 int font_type,
443 FPDF_BOOL cid) {
Nicolas Penabe90aae2017-02-27 10:41:41 -0500444 CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document);
Nicolas Penad03ca422017-03-06 13:54:33 -0500445 if (!pDoc || !data || size == 0 ||
446 (font_type != FPDF_FONT_TYPE1 && font_type != FPDF_FONT_TRUETYPE)) {
Nicolas Penabe90aae2017-02-27 10:41:41 -0500447 return nullptr;
Nicolas Penad03ca422017-03-06 13:54:33 -0500448 }
Nicolas Penabe90aae2017-02-27 10:41:41 -0500449
450 auto pFont = pdfium::MakeUnique<CFX_Font>();
451
Nicolas Penad03ca422017-03-06 13:54:33 -0500452 // TODO(npm): Maybe use FT_Get_X11_Font_Format to check format? Otherwise, we
453 // are allowing giving any font that can be loaded on freetype and setting it
454 // as any font type.
Nicolas Penabe90aae2017-02-27 10:41:41 -0500455 if (!pFont->LoadEmbedded(data, size))
456 return nullptr;
457
Nicolas Penad03ca422017-03-06 13:54:33 -0500458 return cid ? LoadCompositeFont(pDoc, std::move(pFont), data, size, font_type)
459 : LoadSimpleFont(pDoc, std::move(pFont), data, size, font_type);
Nicolas Penabe90aae2017-02-27 10:41:41 -0500460}
Nicolas Penab3161852017-05-02 14:12:50 -0400461
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400462FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
463FPDFText_SetFillColor(FPDF_PAGEOBJECT text_object,
464 unsigned int R,
465 unsigned int G,
466 unsigned int B,
467 unsigned int A) {
wileyrya864e9fb2017-05-26 11:38:14 -0500468 return FPDFPageObj_SetFillColor(text_object, R, G, B, A);
469}
470
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400471FPDF_EXPORT void FPDF_CALLCONV FPDFFont_Close(FPDF_FONT font) {
Nicolas Pena54b91662017-05-05 16:49:30 -0400472 CPDF_Font* pFont = static_cast<CPDF_Font*>(font);
473 if (!pFont)
Nicolas Penab3161852017-05-02 14:12:50 -0400474 return;
475
Lei Zhangb8a8c432017-05-04 14:10:50 -0700476 CPDF_Document* pDoc = pFont->GetDocument();
477 if (!pDoc)
478 return;
479
480 CPDF_DocPageData* pPageData = pDoc->GetPageData();
481 if (!pPageData->IsForceClear())
482 pPageData->ReleaseFont(pFont->GetFontDict());
Nicolas Penab3161852017-05-02 14:12:50 -0400483}
484
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400485FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV
Nicolas Penab3161852017-05-02 14:12:50 -0400486FPDFPageObj_CreateTextObj(FPDF_DOCUMENT document,
487 FPDF_FONT font,
488 float font_size) {
489 CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document);
Nicolas Pena54b91662017-05-05 16:49:30 -0400490 CPDF_Font* pFont = static_cast<CPDF_Font*>(font);
491 if (!pDoc || !pFont)
Nicolas Penab3161852017-05-02 14:12:50 -0400492 return nullptr;
493
Nicolas Penab3161852017-05-02 14:12:50 -0400494 auto pTextObj = pdfium::MakeUnique<CPDF_TextObject>();
495 pTextObj->m_TextState.SetFont(pDoc->LoadFont(pFont->GetFontDict()));
496 pTextObj->m_TextState.SetFontSize(font_size);
497 pTextObj->DefaultStates();
498 return pTextObj.release();
499}