• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // 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 
5 #include <map>
6 #include <memory>
7 #include <utility>
8 #include <vector>
9 
10 #include "core/fpdfapi/cpdf_modulemgr.h"
11 #include "core/fpdfapi/font/cpdf_font.h"
12 #include "core/fpdfapi/font/cpdf_type1font.h"
13 #include "core/fpdfapi/page/cpdf_docpagedata.h"
14 #include "core/fpdfapi/page/cpdf_textobject.h"
15 #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"
22 #include "core/fxcrt/fx_extension.h"
23 #include "core/fxge/cfx_fontmgr.h"
24 #include "core/fxge/fx_font.h"
25 #include "fpdfsdk/fsdk_define.h"
26 #include "public/fpdf_edit.h"
27 
28 namespace {
29 
LoadFontDesc(CPDF_Document * pDoc,const ByteString & font_name,CFX_Font * pFont,const uint8_t * data,uint32_t size,int font_type)30 CPDF_Dictionary* LoadFontDesc(CPDF_Document* pDoc,
31                               const ByteString& font_name,
32                               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;
42   if (font_name.Contains("Serif"))
43     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);
57   pBBox->AddNew<CPDF_Number>(bbox.top);
58   pBBox->AddNew<CPDF_Number>(bbox.right);
59   pBBox->AddNew<CPDF_Number>(bbox.bottom);
60   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);
74   // 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   }
79   ByteString fontFile = font_type == FPDF_FONT_TYPE1 ? "FontFile" : "FontFile2";
80   fontDesc->SetNewFor<CPDF_Reference>(fontFile, pDoc, pStream->GetObjNum());
81   return fontDesc;
82 }
83 
84 const 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"
96     "<0000> <FFFFF>\n"
97     "endcodespacerange\n";
98 
99 const char ToUnicodeEnd[] =
100     "endcmap\n"
101     "CMapName currentdict /CMap defineresource pop\n"
102     "end\n"
103     "end\n";
104 
AddCharcode(std::ostringstream * pBuffer,uint32_t number)105 void AddCharcode(std::ostringstream* pBuffer, uint32_t number) {
106   ASSERT(number <= 0xFFFF);
107   *pBuffer << "<";
108   char ans[4];
109   FXSYS_IntToFourHexChars(number, ans);
110   for (size_t i = 0; i < 4; ++i)
111     *pBuffer << ans[i];
112   *pBuffer << ">";
113 }
114 
115 // 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
AddUnicode(std::ostringstream * pBuffer,uint32_t unicode)117 void AddUnicode(std::ostringstream* pBuffer, uint32_t unicode) {
118   if (unicode >= 0xD800 && unicode <= 0xDFFF)
119     unicode = 0;
120 
121   char ans[8];
122   *pBuffer << "<";
123   size_t numChars = FXSYS_ToUTF16BE(unicode, ans);
124   for (size_t i = 0; i < numChars; ++i)
125     *pBuffer << ans[i];
126   *pBuffer << ">";
127 }
128 
129 // Loads the charcode to unicode mapping into a stream
LoadUnicode(CPDF_Document * pDoc,const std::map<uint32_t,uint32_t> & to_unicode)130 CPDF_Stream* LoadUnicode(CPDF_Document* pDoc,
131                          const std::map<uint32_t, uint32_t>& to_unicode) {
132   // 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   }
194   std::ostringstream buffer;
195   buffer << ToUnicodeStart;
196   // Add maps to buffer
197   buffer << static_cast<uint32_t>(char_to_uni.size()) << " beginbfchar\n";
198   for (const auto& iter : char_to_uni) {
199     AddCharcode(&buffer, iter.first);
200     buffer << " ";
201     AddUnicode(&buffer, iter.second);
202     buffer << "\n";
203   }
204   buffer << "endbfchar\n"
205          << static_cast<uint32_t>(map_range_vector.size() + map_range.size())
206          << " beginbfrange\n";
207   for (const auto& iter : map_range_vector) {
208     const std::pair<uint32_t, uint32_t>& charcodeRange = iter.first;
209     AddCharcode(&buffer, charcodeRange.first);
210     buffer << " ";
211     AddCharcode(&buffer, charcodeRange.second);
212     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];
216       AddUnicode(&buffer, uni);
217       if (i != unicodes.size() - 1)
218         buffer << " ";
219     }
220     buffer << "]\n";
221   }
222   for (const auto& iter : map_range) {
223     const std::pair<uint32_t, uint32_t>& charcodeRange = iter.first;
224     AddCharcode(&buffer, charcodeRange.first);
225     buffer << " ";
226     AddCharcode(&buffer, charcodeRange.second);
227     buffer << " ";
228     AddUnicode(&buffer, iter.second);
229     buffer << "\n";
230   }
231   buffer << "endbfrange\n";
232   buffer << ToUnicodeEnd;
233   // TODO(npm): Encrypt / Compress?
234   CPDF_Stream* stream = pDoc->NewIndirect<CPDF_Stream>();
235   stream->SetData(&buffer);
236   return stream;
237 }
238 
239 const uint32_t kMaxSimpleFontChar = 0xFF;
240 
LoadSimpleFont(CPDF_Document * pDoc,std::unique_ptr<CFX_Font> pFont,const uint8_t * data,uint32_t size,int font_type)241 void* 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");
250   ByteString name = pFont->GetFaceName();
251   if (name.IsEmpty())
252     name = "Unnamed";
253   fontDict->SetNewFor<CPDF_Name>("BaseFont", name);
254 
255   uint32_t glyphIndex;
256   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));
260   CPDF_Array* widthsArray = pDoc->NewIndirect<CPDF_Array>();
261   while (true) {
262     widthsArray->AddNew<CPDF_Number>(pFont->GetGlyphWidth(glyphIndex));
263     uint32_t nextChar =
264         FXFT_Get_Next_Char(pFont->GetFace(), currentChar, &glyphIndex);
265     // Simple fonts have 1-byte charcodes only.
266     if (nextChar > kMaxSimpleFontChar || glyphIndex == 0)
267       break;
268     for (uint32_t i = currentChar + 1; i < nextChar; i++)
269       widthsArray->AddNew<CPDF_Number>(0);
270     currentChar = nextChar;
271   }
272   fontDict->SetNewFor<CPDF_Number>("LastChar", static_cast<int>(currentChar));
273   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 
282 const uint32_t kMaxUnicode = 0x10FFFF;
283 
LoadCompositeFont(CPDF_Document * pDoc,std::unique_ptr<CFX_Font> pFont,const uint8_t * data,uint32_t size,int font_type)284 void* 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.
293   ByteString encoding = "Identity-H";
294   fontDict->SetNewFor<CPDF_Name>("Encoding", encoding);
295   ByteString name = pFont->GetFaceName();
296   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;
323   uint32_t currentChar = FXFT_Get_First_Char(pFont->GetFace(), &glyphIndex);
324   // If it doesn't have a single char, just fail
325   if (glyphIndex == 0 || currentChar > kMaxUnicode)
326     return nullptr;
327 
328   std::map<uint32_t, uint32_t> to_unicode;
329   std::map<uint32_t, uint32_t> widths;
330   while (true) {
331     if (currentChar > kMaxUnicode)
332       break;
333 
334     widths[glyphIndex] = pFont->GetGlyphWidth(glyphIndex);
335     to_unicode[glyphIndex] = currentChar;
336     currentChar =
337         FXFT_Get_Next_Char(pFont->GetFace(), currentChar, &glyphIndex);
338     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()) {
346       // Only one char left, use format c [w]
347       auto oneW = pdfium::MakeUnique<CPDF_Array>();
348       oneW->AddNew<CPDF_Number>(w);
349       widthsArray->AddNew<CPDF_Number>(ch);
350       widthsArray->Add(std::move(oneW));
351       break;
352     }
353     ++it;
354     int next_ch = it->first;
355     int next_w = it->second;
356     if (next_ch == ch + 1 && next_w == w) {
357       // 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
359       widthsArray->AddNew<CPDF_Number>(ch);
360       ch = next_ch;
361       while (true) {
362         auto next_it = std::next(it);
363         if (next_it == widths.end() || next_it->first != it->first + 1 ||
364             next_it->second != it->second) {
365           break;
366         }
367         ++it;
368         ch = it->first;
369       }
370       widthsArray->AddNew<CPDF_Number>(ch);
371       widthsArray->AddNew<CPDF_Number>(w);
372       continue;
373     }
374     // 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));
388   }
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));
395   CPDF_Stream* toUnicodeStream = LoadUnicode(pDoc, to_unicode);
396   fontDict->SetNewFor<CPDF_Reference>("ToUnicode", pDoc,
397                                       toUnicodeStream->GetObjNum());
398   return pDoc->LoadFont(fontDict);
399 }
400 
401 }  // namespace
402 
403 FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV
FPDFPageObj_NewTextObj(FPDF_DOCUMENT document,FPDF_BYTESTRING font,float font_size)404 FPDFPageObj_NewTextObj(FPDF_DOCUMENT document,
405                        FPDF_BYTESTRING font,
406                        float font_size) {
407   CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document);
408   if (!pDoc)
409     return nullptr;
410 
411   CPDF_Font* pFont = CPDF_Font::GetStockFont(pDoc, ByteStringView(font));
412   if (!pFont)
413     return nullptr;
414 
415   auto pTextObj = pdfium::MakeUnique<CPDF_TextObject>();
416   pTextObj->m_TextState.SetFont(pFont);
417   pTextObj->m_TextState.SetFontSize(font_size);
418   pTextObj->DefaultStates();
419   return pTextObj.release();  // Caller takes ownership.
420 }
421 
422 FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
FPDFText_SetText(FPDF_PAGEOBJECT text_object,FPDF_WIDESTRING text)423 FPDFText_SetText(FPDF_PAGEOBJECT text_object, FPDF_WIDESTRING text) {
424   auto* pTextObj = static_cast<CPDF_TextObject*>(text_object);
425   if (!pTextObj)
426     return false;
427 
428   size_t len = WideString::WStringLength(text);
429   WideString encodedText = WideString::FromUTF16LE(text, len);
430   ByteString byteText;
431   for (wchar_t wc : encodedText) {
432     pTextObj->GetFont()->AppendChar(
433         &byteText, pTextObj->GetFont()->CharCodeFromUnicode(wc));
434   }
435   pTextObj->SetText(byteText);
436   return true;
437 }
438 
FPDFText_LoadFont(FPDF_DOCUMENT document,const uint8_t * data,uint32_t size,int font_type,FPDF_BOOL cid)439 FPDF_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) {
444   CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document);
445   if (!pDoc || !data || size == 0 ||
446       (font_type != FPDF_FONT_TYPE1 && font_type != FPDF_FONT_TRUETYPE)) {
447     return nullptr;
448   }
449 
450   auto pFont = pdfium::MakeUnique<CFX_Font>();
451 
452   // 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.
455   if (!pFont->LoadEmbedded(data, size))
456     return nullptr;
457 
458   return cid ? LoadCompositeFont(pDoc, std::move(pFont), data, size, font_type)
459              : LoadSimpleFont(pDoc, std::move(pFont), data, size, font_type);
460 }
461 
462 FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
FPDFText_SetFillColor(FPDF_PAGEOBJECT text_object,unsigned int R,unsigned int G,unsigned int B,unsigned int A)463 FPDFText_SetFillColor(FPDF_PAGEOBJECT text_object,
464                       unsigned int R,
465                       unsigned int G,
466                       unsigned int B,
467                       unsigned int A) {
468   return FPDFPageObj_SetFillColor(text_object, R, G, B, A);
469 }
470 
FPDFFont_Close(FPDF_FONT font)471 FPDF_EXPORT void FPDF_CALLCONV FPDFFont_Close(FPDF_FONT font) {
472   CPDF_Font* pFont = static_cast<CPDF_Font*>(font);
473   if (!pFont)
474     return;
475 
476   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());
483 }
484 
485 FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV
FPDFPageObj_CreateTextObj(FPDF_DOCUMENT document,FPDF_FONT font,float font_size)486 FPDFPageObj_CreateTextObj(FPDF_DOCUMENT document,
487                           FPDF_FONT font,
488                           float font_size) {
489   CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document);
490   CPDF_Font* pFont = static_cast<CPDF_Font*>(font);
491   if (!pDoc || !pFont)
492     return nullptr;
493 
494   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 }
500