xref: /aosp_15_r20/external/pdfium/fpdfsdk/fpdf_edittext.cpp (revision 3ac0a46f773bac49fa9476ec2b1cf3f8da5ec3a4)
1*3ac0a46fSAndroid Build Coastguard Worker // Copyright 2017 The PDFium Authors
2*3ac0a46fSAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*3ac0a46fSAndroid Build Coastguard Worker // found in the LICENSE file.
4*3ac0a46fSAndroid Build Coastguard Worker 
5*3ac0a46fSAndroid Build Coastguard Worker #include <map>
6*3ac0a46fSAndroid Build Coastguard Worker #include <memory>
7*3ac0a46fSAndroid Build Coastguard Worker #include <sstream>
8*3ac0a46fSAndroid Build Coastguard Worker #include <utility>
9*3ac0a46fSAndroid Build Coastguard Worker #include <vector>
10*3ac0a46fSAndroid Build Coastguard Worker 
11*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/font/cpdf_cidfont.h"
12*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/font/cpdf_font.h"
13*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/page/cpdf_docpagedata.h"
14*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/page/cpdf_textobject.h"
15*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/page/cpdf_textstate.h"
16*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_array.h"
17*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_dictionary.h"
18*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_document.h"
19*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_name.h"
20*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_number.h"
21*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_reference.h"
22*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_stream.h"
23*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/parser/cpdf_string.h"
24*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/render/charposlist.h"
25*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/render/cpdf_pagerendercontext.h"
26*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/render/cpdf_rendercontext.h"
27*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/render/cpdf_renderstatus.h"
28*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/render/cpdf_textrenderer.h"
29*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdftext/cpdf_textpage.h"
30*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/fx_extension.h"
31*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/fx_string_wrappers.h"
32*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/span_util.h"
33*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/stl_util.h"
34*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/utf16.h"
35*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/cfx_defaultrenderdevice.h"
36*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/cfx_fontmgr.h"
37*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/dib/cfx_dibitmap.h"
38*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/fx_font.h"
39*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/text_char_pos.h"
40*3ac0a46fSAndroid Build Coastguard Worker #include "fpdfsdk/cpdfsdk_helpers.h"
41*3ac0a46fSAndroid Build Coastguard Worker #include "public/fpdf_edit.h"
42*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/check.h"
43*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/check_op.h"
44*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/containers/contains.h"
45*3ac0a46fSAndroid Build Coastguard Worker #include "third_party/base/numerics/safe_conversions.h"
46*3ac0a46fSAndroid Build Coastguard Worker 
47*3ac0a46fSAndroid Build Coastguard Worker // These checks are here because core/ and public/ cannot depend on each other.
48*3ac0a46fSAndroid Build Coastguard Worker static_assert(static_cast<int>(TextRenderingMode::MODE_UNKNOWN) ==
49*3ac0a46fSAndroid Build Coastguard Worker                   FPDF_TEXTRENDERMODE_UNKNOWN,
50*3ac0a46fSAndroid Build Coastguard Worker               "TextRenderingMode::MODE_UNKNOWN value mismatch");
51*3ac0a46fSAndroid Build Coastguard Worker static_assert(static_cast<int>(TextRenderingMode::MODE_FILL) ==
52*3ac0a46fSAndroid Build Coastguard Worker                   FPDF_TEXTRENDERMODE_FILL,
53*3ac0a46fSAndroid Build Coastguard Worker               "TextRenderingMode::MODE_FILL value mismatch");
54*3ac0a46fSAndroid Build Coastguard Worker static_assert(static_cast<int>(TextRenderingMode::MODE_STROKE) ==
55*3ac0a46fSAndroid Build Coastguard Worker                   FPDF_TEXTRENDERMODE_STROKE,
56*3ac0a46fSAndroid Build Coastguard Worker               "TextRenderingMode::MODE_STROKE value mismatch");
57*3ac0a46fSAndroid Build Coastguard Worker static_assert(static_cast<int>(TextRenderingMode::MODE_FILL_STROKE) ==
58*3ac0a46fSAndroid Build Coastguard Worker                   FPDF_TEXTRENDERMODE_FILL_STROKE,
59*3ac0a46fSAndroid Build Coastguard Worker               "TextRenderingMode::MODE_FILL_STROKE value mismatch");
60*3ac0a46fSAndroid Build Coastguard Worker static_assert(static_cast<int>(TextRenderingMode::MODE_INVISIBLE) ==
61*3ac0a46fSAndroid Build Coastguard Worker                   FPDF_TEXTRENDERMODE_INVISIBLE,
62*3ac0a46fSAndroid Build Coastguard Worker               "TextRenderingMode::MODE_INVISIBLE value mismatch");
63*3ac0a46fSAndroid Build Coastguard Worker static_assert(static_cast<int>(TextRenderingMode::MODE_FILL_CLIP) ==
64*3ac0a46fSAndroid Build Coastguard Worker                   FPDF_TEXTRENDERMODE_FILL_CLIP,
65*3ac0a46fSAndroid Build Coastguard Worker               "TextRenderingMode::MODE_FILL_CLIP value mismatch");
66*3ac0a46fSAndroid Build Coastguard Worker static_assert(static_cast<int>(TextRenderingMode::MODE_STROKE_CLIP) ==
67*3ac0a46fSAndroid Build Coastguard Worker                   FPDF_TEXTRENDERMODE_STROKE_CLIP,
68*3ac0a46fSAndroid Build Coastguard Worker               "TextRenderingMode::MODE_STROKE_CLIP value mismatch");
69*3ac0a46fSAndroid Build Coastguard Worker static_assert(static_cast<int>(TextRenderingMode::MODE_FILL_STROKE_CLIP) ==
70*3ac0a46fSAndroid Build Coastguard Worker                   FPDF_TEXTRENDERMODE_FILL_STROKE_CLIP,
71*3ac0a46fSAndroid Build Coastguard Worker               "TextRenderingMode::MODE_FILL_STROKE_CLIP value mismatch");
72*3ac0a46fSAndroid Build Coastguard Worker static_assert(static_cast<int>(TextRenderingMode::MODE_CLIP) ==
73*3ac0a46fSAndroid Build Coastguard Worker                   FPDF_TEXTRENDERMODE_CLIP,
74*3ac0a46fSAndroid Build Coastguard Worker               "TextRenderingMode::MODE_CLIP value mismatch");
75*3ac0a46fSAndroid Build Coastguard Worker static_assert(static_cast<int>(TextRenderingMode::MODE_LAST) ==
76*3ac0a46fSAndroid Build Coastguard Worker                   FPDF_TEXTRENDERMODE_LAST,
77*3ac0a46fSAndroid Build Coastguard Worker               "TextRenderingMode::MODE_LAST value mismatch");
78*3ac0a46fSAndroid Build Coastguard Worker 
79*3ac0a46fSAndroid Build Coastguard Worker namespace {
80*3ac0a46fSAndroid Build Coastguard Worker 
BaseFontNameForType(CFX_Font * pFont,int font_type)81*3ac0a46fSAndroid Build Coastguard Worker ByteString BaseFontNameForType(CFX_Font* pFont, int font_type) {
82*3ac0a46fSAndroid Build Coastguard Worker   ByteString name = font_type == FPDF_FONT_TYPE1 ? pFont->GetPsName()
83*3ac0a46fSAndroid Build Coastguard Worker                                                  : pFont->GetBaseFontName();
84*3ac0a46fSAndroid Build Coastguard Worker   if (!name.IsEmpty())
85*3ac0a46fSAndroid Build Coastguard Worker     return name;
86*3ac0a46fSAndroid Build Coastguard Worker 
87*3ac0a46fSAndroid Build Coastguard Worker   return CFX_Font::kUntitledFontName;
88*3ac0a46fSAndroid Build Coastguard Worker }
89*3ac0a46fSAndroid Build Coastguard Worker 
LoadFontDesc(CPDF_Document * pDoc,const ByteString & font_name,CFX_Font * pFont,pdfium::span<const uint8_t> span,int font_type)90*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Dictionary> LoadFontDesc(CPDF_Document* pDoc,
91*3ac0a46fSAndroid Build Coastguard Worker                                         const ByteString& font_name,
92*3ac0a46fSAndroid Build Coastguard Worker                                         CFX_Font* pFont,
93*3ac0a46fSAndroid Build Coastguard Worker                                         pdfium::span<const uint8_t> span,
94*3ac0a46fSAndroid Build Coastguard Worker                                         int font_type) {
95*3ac0a46fSAndroid Build Coastguard Worker   auto pFontDesc = pDoc->NewIndirect<CPDF_Dictionary>();
96*3ac0a46fSAndroid Build Coastguard Worker   pFontDesc->SetNewFor<CPDF_Name>("Type", "FontDescriptor");
97*3ac0a46fSAndroid Build Coastguard Worker   pFontDesc->SetNewFor<CPDF_Name>("FontName", font_name);
98*3ac0a46fSAndroid Build Coastguard Worker   int flags = 0;
99*3ac0a46fSAndroid Build Coastguard Worker   if (FXFT_Is_Face_fixedwidth(pFont->GetFaceRec()))
100*3ac0a46fSAndroid Build Coastguard Worker     flags |= FXFONT_FIXED_PITCH;
101*3ac0a46fSAndroid Build Coastguard Worker   if (font_name.Contains("Serif"))
102*3ac0a46fSAndroid Build Coastguard Worker     flags |= FXFONT_SERIF;
103*3ac0a46fSAndroid Build Coastguard Worker   if (FXFT_Is_Face_Italic(pFont->GetFaceRec()))
104*3ac0a46fSAndroid Build Coastguard Worker     flags |= FXFONT_ITALIC;
105*3ac0a46fSAndroid Build Coastguard Worker   if (FXFT_Is_Face_Bold(pFont->GetFaceRec()))
106*3ac0a46fSAndroid Build Coastguard Worker     flags |= FXFONT_FORCE_BOLD;
107*3ac0a46fSAndroid Build Coastguard Worker 
108*3ac0a46fSAndroid Build Coastguard Worker   // TODO(npm): How do I know if a  font is symbolic, script, allcap, smallcap
109*3ac0a46fSAndroid Build Coastguard Worker   flags |= FXFONT_NONSYMBOLIC;
110*3ac0a46fSAndroid Build Coastguard Worker 
111*3ac0a46fSAndroid Build Coastguard Worker   pFontDesc->SetNewFor<CPDF_Number>("Flags", flags);
112*3ac0a46fSAndroid Build Coastguard Worker   FX_RECT bbox = pFont->GetBBox().value_or(FX_RECT());
113*3ac0a46fSAndroid Build Coastguard Worker   pFontDesc->SetRectFor("FontBBox", CFX_FloatRect(bbox));
114*3ac0a46fSAndroid Build Coastguard Worker 
115*3ac0a46fSAndroid Build Coastguard Worker   // TODO(npm): calculate italic angle correctly
116*3ac0a46fSAndroid Build Coastguard Worker   pFontDesc->SetNewFor<CPDF_Number>("ItalicAngle", pFont->IsItalic() ? -12 : 0);
117*3ac0a46fSAndroid Build Coastguard Worker 
118*3ac0a46fSAndroid Build Coastguard Worker   pFontDesc->SetNewFor<CPDF_Number>("Ascent", pFont->GetAscent());
119*3ac0a46fSAndroid Build Coastguard Worker   pFontDesc->SetNewFor<CPDF_Number>("Descent", pFont->GetDescent());
120*3ac0a46fSAndroid Build Coastguard Worker 
121*3ac0a46fSAndroid Build Coastguard Worker   // TODO(npm): calculate the capheight, stemV correctly
122*3ac0a46fSAndroid Build Coastguard Worker   pFontDesc->SetNewFor<CPDF_Number>("CapHeight", pFont->GetAscent());
123*3ac0a46fSAndroid Build Coastguard Worker   pFontDesc->SetNewFor<CPDF_Number>("StemV", pFont->IsBold() ? 120 : 70);
124*3ac0a46fSAndroid Build Coastguard Worker 
125*3ac0a46fSAndroid Build Coastguard Worker   auto pStream = pDoc->NewIndirect<CPDF_Stream>();
126*3ac0a46fSAndroid Build Coastguard Worker   pStream->SetData(span);
127*3ac0a46fSAndroid Build Coastguard Worker   // TODO(npm): Lengths for Type1 fonts.
128*3ac0a46fSAndroid Build Coastguard Worker   if (font_type == FPDF_FONT_TRUETYPE) {
129*3ac0a46fSAndroid Build Coastguard Worker     pStream->GetMutableDict()->SetNewFor<CPDF_Number>(
130*3ac0a46fSAndroid Build Coastguard Worker         "Length1", static_cast<int>(span.size()));
131*3ac0a46fSAndroid Build Coastguard Worker   }
132*3ac0a46fSAndroid Build Coastguard Worker   ByteString fontFile = font_type == FPDF_FONT_TYPE1 ? "FontFile" : "FontFile2";
133*3ac0a46fSAndroid Build Coastguard Worker   pFontDesc->SetNewFor<CPDF_Reference>(fontFile, pDoc, pStream->GetObjNum());
134*3ac0a46fSAndroid Build Coastguard Worker   return pFontDesc;
135*3ac0a46fSAndroid Build Coastguard Worker }
136*3ac0a46fSAndroid Build Coastguard Worker 
137*3ac0a46fSAndroid Build Coastguard Worker const char ToUnicodeStart[] =
138*3ac0a46fSAndroid Build Coastguard Worker     "/CIDInit /ProcSet findresource begin\n"
139*3ac0a46fSAndroid Build Coastguard Worker     "12 dict begin\n"
140*3ac0a46fSAndroid Build Coastguard Worker     "begincmap\n"
141*3ac0a46fSAndroid Build Coastguard Worker     "/CIDSystemInfo\n"
142*3ac0a46fSAndroid Build Coastguard Worker     "<</Registry (Adobe)\n"
143*3ac0a46fSAndroid Build Coastguard Worker     "/Ordering (Identity)\n"
144*3ac0a46fSAndroid Build Coastguard Worker     "/Supplement 0\n"
145*3ac0a46fSAndroid Build Coastguard Worker     ">> def\n"
146*3ac0a46fSAndroid Build Coastguard Worker     "/CMapName /Adobe-Identity-H def\n"
147*3ac0a46fSAndroid Build Coastguard Worker     "CMapType 2 def\n"
148*3ac0a46fSAndroid Build Coastguard Worker     "1 begincodespacerange\n"
149*3ac0a46fSAndroid Build Coastguard Worker     "<0000> <FFFFF>\n"
150*3ac0a46fSAndroid Build Coastguard Worker     "endcodespacerange\n";
151*3ac0a46fSAndroid Build Coastguard Worker 
152*3ac0a46fSAndroid Build Coastguard Worker const char ToUnicodeEnd[] =
153*3ac0a46fSAndroid Build Coastguard Worker     "endcmap\n"
154*3ac0a46fSAndroid Build Coastguard Worker     "CMapName currentdict /CMap defineresource pop\n"
155*3ac0a46fSAndroid Build Coastguard Worker     "end\n"
156*3ac0a46fSAndroid Build Coastguard Worker     "end\n";
157*3ac0a46fSAndroid Build Coastguard Worker 
AddCharcode(fxcrt::ostringstream * pBuffer,uint32_t number)158*3ac0a46fSAndroid Build Coastguard Worker void AddCharcode(fxcrt::ostringstream* pBuffer, uint32_t number) {
159*3ac0a46fSAndroid Build Coastguard Worker   DCHECK(number <= 0xFFFF);
160*3ac0a46fSAndroid Build Coastguard Worker   *pBuffer << "<";
161*3ac0a46fSAndroid Build Coastguard Worker   char ans[4];
162*3ac0a46fSAndroid Build Coastguard Worker   FXSYS_IntToFourHexChars(number, ans);
163*3ac0a46fSAndroid Build Coastguard Worker   for (size_t i = 0; i < 4; ++i)
164*3ac0a46fSAndroid Build Coastguard Worker     *pBuffer << ans[i];
165*3ac0a46fSAndroid Build Coastguard Worker   *pBuffer << ">";
166*3ac0a46fSAndroid Build Coastguard Worker }
167*3ac0a46fSAndroid Build Coastguard Worker 
168*3ac0a46fSAndroid Build Coastguard Worker // PDF spec 1.7 Section 5.9.2: "Unicode character sequences as expressed in
169*3ac0a46fSAndroid Build Coastguard Worker // UTF-16BE encoding." See https://en.wikipedia.org/wiki/UTF-16#Description
AddUnicode(fxcrt::ostringstream * pBuffer,uint32_t unicode)170*3ac0a46fSAndroid Build Coastguard Worker void AddUnicode(fxcrt::ostringstream* pBuffer, uint32_t unicode) {
171*3ac0a46fSAndroid Build Coastguard Worker   if (pdfium::IsHighSurrogate(unicode) || pdfium::IsLowSurrogate(unicode)) {
172*3ac0a46fSAndroid Build Coastguard Worker     unicode = 0;
173*3ac0a46fSAndroid Build Coastguard Worker   }
174*3ac0a46fSAndroid Build Coastguard Worker 
175*3ac0a46fSAndroid Build Coastguard Worker   char ans[8];
176*3ac0a46fSAndroid Build Coastguard Worker   *pBuffer << "<";
177*3ac0a46fSAndroid Build Coastguard Worker   size_t numChars = FXSYS_ToUTF16BE(unicode, ans);
178*3ac0a46fSAndroid Build Coastguard Worker   for (size_t i = 0; i < numChars; ++i)
179*3ac0a46fSAndroid Build Coastguard Worker     *pBuffer << ans[i];
180*3ac0a46fSAndroid Build Coastguard Worker   *pBuffer << ">";
181*3ac0a46fSAndroid Build Coastguard Worker }
182*3ac0a46fSAndroid Build Coastguard Worker 
183*3ac0a46fSAndroid Build Coastguard Worker // Loads the charcode to unicode mapping into a stream
LoadUnicode(CPDF_Document * pDoc,const std::multimap<uint32_t,uint32_t> & to_unicode)184*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Stream> LoadUnicode(
185*3ac0a46fSAndroid Build Coastguard Worker     CPDF_Document* pDoc,
186*3ac0a46fSAndroid Build Coastguard Worker     const std::multimap<uint32_t, uint32_t>& to_unicode) {
187*3ac0a46fSAndroid Build Coastguard Worker   // A map charcode->unicode
188*3ac0a46fSAndroid Build Coastguard Worker   std::map<uint32_t, uint32_t> char_to_uni;
189*3ac0a46fSAndroid Build Coastguard Worker   // A map <char_start, char_end> to vector v of unicode characters of size (end
190*3ac0a46fSAndroid Build Coastguard Worker   // - start + 1). This abbreviates: start->v[0], start+1->v[1], etc. PDF spec
191*3ac0a46fSAndroid Build Coastguard Worker   // 1.7 Section 5.9.2 says that only the last byte of the unicode may change.
192*3ac0a46fSAndroid Build Coastguard Worker   std::map<std::pair<uint32_t, uint32_t>, std::vector<uint32_t>>
193*3ac0a46fSAndroid Build Coastguard Worker       map_range_vector;
194*3ac0a46fSAndroid Build Coastguard Worker   // A map <start, end> -> unicode
195*3ac0a46fSAndroid Build Coastguard Worker   // This abbreviates: start->unicode, start+1->unicode+1, etc.
196*3ac0a46fSAndroid Build Coastguard Worker   // PDF spec 1.7 Section 5.9.2 says that only the last byte of the unicode may
197*3ac0a46fSAndroid Build Coastguard Worker   // change.
198*3ac0a46fSAndroid Build Coastguard Worker   std::map<std::pair<uint32_t, uint32_t>, uint32_t> map_range;
199*3ac0a46fSAndroid Build Coastguard Worker 
200*3ac0a46fSAndroid Build Coastguard Worker   // Calculate the maps
201*3ac0a46fSAndroid Build Coastguard Worker   for (auto iter = to_unicode.begin(); iter != to_unicode.end(); ++iter) {
202*3ac0a46fSAndroid Build Coastguard Worker     uint32_t firstCharcode = iter->first;
203*3ac0a46fSAndroid Build Coastguard Worker     uint32_t firstUnicode = iter->second;
204*3ac0a46fSAndroid Build Coastguard Worker     if (std::next(iter) == to_unicode.end() ||
205*3ac0a46fSAndroid Build Coastguard Worker         firstCharcode + 1 != std::next(iter)->first) {
206*3ac0a46fSAndroid Build Coastguard Worker       char_to_uni[firstCharcode] = firstUnicode;
207*3ac0a46fSAndroid Build Coastguard Worker       continue;
208*3ac0a46fSAndroid Build Coastguard Worker     }
209*3ac0a46fSAndroid Build Coastguard Worker     ++iter;
210*3ac0a46fSAndroid Build Coastguard Worker     uint32_t curCharcode = iter->first;
211*3ac0a46fSAndroid Build Coastguard Worker     uint32_t curUnicode = iter->second;
212*3ac0a46fSAndroid Build Coastguard Worker     if (curCharcode % 256 == 0) {
213*3ac0a46fSAndroid Build Coastguard Worker       char_to_uni[firstCharcode] = firstUnicode;
214*3ac0a46fSAndroid Build Coastguard Worker       char_to_uni[curCharcode] = curUnicode;
215*3ac0a46fSAndroid Build Coastguard Worker       continue;
216*3ac0a46fSAndroid Build Coastguard Worker     }
217*3ac0a46fSAndroid Build Coastguard Worker     const size_t maxExtra = 255 - (curCharcode % 256);
218*3ac0a46fSAndroid Build Coastguard Worker     auto next_it = std::next(iter);
219*3ac0a46fSAndroid Build Coastguard Worker     if (firstUnicode + 1 != curUnicode) {
220*3ac0a46fSAndroid Build Coastguard Worker       // Consecutive charcodes mapping to non-consecutive unicodes
221*3ac0a46fSAndroid Build Coastguard Worker       std::vector<uint32_t> unicodes;
222*3ac0a46fSAndroid Build Coastguard Worker       unicodes.push_back(firstUnicode);
223*3ac0a46fSAndroid Build Coastguard Worker       unicodes.push_back(curUnicode);
224*3ac0a46fSAndroid Build Coastguard Worker       for (size_t i = 0; i < maxExtra; ++i) {
225*3ac0a46fSAndroid Build Coastguard Worker         if (next_it == to_unicode.end() || curCharcode + 1 != next_it->first)
226*3ac0a46fSAndroid Build Coastguard Worker           break;
227*3ac0a46fSAndroid Build Coastguard Worker         ++iter;
228*3ac0a46fSAndroid Build Coastguard Worker         ++curCharcode;
229*3ac0a46fSAndroid Build Coastguard Worker         unicodes.push_back(iter->second);
230*3ac0a46fSAndroid Build Coastguard Worker         next_it = std::next(iter);
231*3ac0a46fSAndroid Build Coastguard Worker       }
232*3ac0a46fSAndroid Build Coastguard Worker       DCHECK_EQ(iter->first - firstCharcode + 1, unicodes.size());
233*3ac0a46fSAndroid Build Coastguard Worker       map_range_vector[std::make_pair(firstCharcode, iter->first)] = unicodes;
234*3ac0a46fSAndroid Build Coastguard Worker       continue;
235*3ac0a46fSAndroid Build Coastguard Worker     }
236*3ac0a46fSAndroid Build Coastguard Worker     // Consecutive charcodes mapping to consecutive unicodes
237*3ac0a46fSAndroid Build Coastguard Worker     for (size_t i = 0; i < maxExtra; ++i) {
238*3ac0a46fSAndroid Build Coastguard Worker       if (next_it == to_unicode.end() || curCharcode + 1 != next_it->first ||
239*3ac0a46fSAndroid Build Coastguard Worker           curUnicode + 1 != next_it->second) {
240*3ac0a46fSAndroid Build Coastguard Worker         break;
241*3ac0a46fSAndroid Build Coastguard Worker       }
242*3ac0a46fSAndroid Build Coastguard Worker       ++iter;
243*3ac0a46fSAndroid Build Coastguard Worker       ++curCharcode;
244*3ac0a46fSAndroid Build Coastguard Worker       ++curUnicode;
245*3ac0a46fSAndroid Build Coastguard Worker       next_it = std::next(iter);
246*3ac0a46fSAndroid Build Coastguard Worker     }
247*3ac0a46fSAndroid Build Coastguard Worker     map_range[std::make_pair(firstCharcode, curCharcode)] = firstUnicode;
248*3ac0a46fSAndroid Build Coastguard Worker   }
249*3ac0a46fSAndroid Build Coastguard Worker   fxcrt::ostringstream buffer;
250*3ac0a46fSAndroid Build Coastguard Worker   buffer << ToUnicodeStart;
251*3ac0a46fSAndroid Build Coastguard Worker   // Add maps to buffer
252*3ac0a46fSAndroid Build Coastguard Worker   buffer << static_cast<uint32_t>(char_to_uni.size()) << " beginbfchar\n";
253*3ac0a46fSAndroid Build Coastguard Worker   for (const auto& iter : char_to_uni) {
254*3ac0a46fSAndroid Build Coastguard Worker     AddCharcode(&buffer, iter.first);
255*3ac0a46fSAndroid Build Coastguard Worker     buffer << " ";
256*3ac0a46fSAndroid Build Coastguard Worker     AddUnicode(&buffer, iter.second);
257*3ac0a46fSAndroid Build Coastguard Worker     buffer << "\n";
258*3ac0a46fSAndroid Build Coastguard Worker   }
259*3ac0a46fSAndroid Build Coastguard Worker   buffer << "endbfchar\n"
260*3ac0a46fSAndroid Build Coastguard Worker          << static_cast<uint32_t>(map_range_vector.size() + map_range.size())
261*3ac0a46fSAndroid Build Coastguard Worker          << " beginbfrange\n";
262*3ac0a46fSAndroid Build Coastguard Worker   for (const auto& iter : map_range_vector) {
263*3ac0a46fSAndroid Build Coastguard Worker     const std::pair<uint32_t, uint32_t>& charcodeRange = iter.first;
264*3ac0a46fSAndroid Build Coastguard Worker     AddCharcode(&buffer, charcodeRange.first);
265*3ac0a46fSAndroid Build Coastguard Worker     buffer << " ";
266*3ac0a46fSAndroid Build Coastguard Worker     AddCharcode(&buffer, charcodeRange.second);
267*3ac0a46fSAndroid Build Coastguard Worker     buffer << " [";
268*3ac0a46fSAndroid Build Coastguard Worker     const std::vector<uint32_t>& unicodes = iter.second;
269*3ac0a46fSAndroid Build Coastguard Worker     for (size_t i = 0; i < unicodes.size(); ++i) {
270*3ac0a46fSAndroid Build Coastguard Worker       uint32_t uni = unicodes[i];
271*3ac0a46fSAndroid Build Coastguard Worker       AddUnicode(&buffer, uni);
272*3ac0a46fSAndroid Build Coastguard Worker       if (i != unicodes.size() - 1)
273*3ac0a46fSAndroid Build Coastguard Worker         buffer << " ";
274*3ac0a46fSAndroid Build Coastguard Worker     }
275*3ac0a46fSAndroid Build Coastguard Worker     buffer << "]\n";
276*3ac0a46fSAndroid Build Coastguard Worker   }
277*3ac0a46fSAndroid Build Coastguard Worker   for (const auto& iter : map_range) {
278*3ac0a46fSAndroid Build Coastguard Worker     const std::pair<uint32_t, uint32_t>& charcodeRange = iter.first;
279*3ac0a46fSAndroid Build Coastguard Worker     AddCharcode(&buffer, charcodeRange.first);
280*3ac0a46fSAndroid Build Coastguard Worker     buffer << " ";
281*3ac0a46fSAndroid Build Coastguard Worker     AddCharcode(&buffer, charcodeRange.second);
282*3ac0a46fSAndroid Build Coastguard Worker     buffer << " ";
283*3ac0a46fSAndroid Build Coastguard Worker     AddUnicode(&buffer, iter.second);
284*3ac0a46fSAndroid Build Coastguard Worker     buffer << "\n";
285*3ac0a46fSAndroid Build Coastguard Worker   }
286*3ac0a46fSAndroid Build Coastguard Worker   buffer << "endbfrange\n";
287*3ac0a46fSAndroid Build Coastguard Worker   buffer << ToUnicodeEnd;
288*3ac0a46fSAndroid Build Coastguard Worker   // TODO(npm): Encrypt / Compress?
289*3ac0a46fSAndroid Build Coastguard Worker   auto stream = pDoc->NewIndirect<CPDF_Stream>();
290*3ac0a46fSAndroid Build Coastguard Worker   stream->SetDataFromStringstream(&buffer);
291*3ac0a46fSAndroid Build Coastguard Worker   return stream;
292*3ac0a46fSAndroid Build Coastguard Worker }
293*3ac0a46fSAndroid Build Coastguard Worker 
LoadSimpleFont(CPDF_Document * pDoc,std::unique_ptr<CFX_Font> pFont,pdfium::span<const uint8_t> span,int font_type)294*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Font> LoadSimpleFont(CPDF_Document* pDoc,
295*3ac0a46fSAndroid Build Coastguard Worker                                     std::unique_ptr<CFX_Font> pFont,
296*3ac0a46fSAndroid Build Coastguard Worker                                     pdfium::span<const uint8_t> span,
297*3ac0a46fSAndroid Build Coastguard Worker                                     int font_type) {
298*3ac0a46fSAndroid Build Coastguard Worker   auto pFontDict = pDoc->NewIndirect<CPDF_Dictionary>();
299*3ac0a46fSAndroid Build Coastguard Worker   pFontDict->SetNewFor<CPDF_Name>("Type", "Font");
300*3ac0a46fSAndroid Build Coastguard Worker   pFontDict->SetNewFor<CPDF_Name>(
301*3ac0a46fSAndroid Build Coastguard Worker       "Subtype", font_type == FPDF_FONT_TYPE1 ? "Type1" : "TrueType");
302*3ac0a46fSAndroid Build Coastguard Worker   ByteString name = BaseFontNameForType(pFont.get(), font_type);
303*3ac0a46fSAndroid Build Coastguard Worker   pFontDict->SetNewFor<CPDF_Name>("BaseFont", name);
304*3ac0a46fSAndroid Build Coastguard Worker 
305*3ac0a46fSAndroid Build Coastguard Worker   uint32_t dwGlyphIndex;
306*3ac0a46fSAndroid Build Coastguard Worker   uint32_t dwCurrentChar = static_cast<uint32_t>(
307*3ac0a46fSAndroid Build Coastguard Worker       FT_Get_First_Char(pFont->GetFaceRec(), &dwGlyphIndex));
308*3ac0a46fSAndroid Build Coastguard Worker   static constexpr uint32_t kMaxSimpleFontChar = 0xFF;
309*3ac0a46fSAndroid Build Coastguard Worker   if (dwCurrentChar > kMaxSimpleFontChar || dwGlyphIndex == 0)
310*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
311*3ac0a46fSAndroid Build Coastguard Worker   pFontDict->SetNewFor<CPDF_Number>("FirstChar",
312*3ac0a46fSAndroid Build Coastguard Worker                                     static_cast<int>(dwCurrentChar));
313*3ac0a46fSAndroid Build Coastguard Worker   auto widthsArray = pDoc->NewIndirect<CPDF_Array>();
314*3ac0a46fSAndroid Build Coastguard Worker   while (true) {
315*3ac0a46fSAndroid Build Coastguard Worker     widthsArray->AppendNew<CPDF_Number>(pFont->GetGlyphWidth(dwGlyphIndex));
316*3ac0a46fSAndroid Build Coastguard Worker     uint32_t nextChar = static_cast<uint32_t>(
317*3ac0a46fSAndroid Build Coastguard Worker         FT_Get_Next_Char(pFont->GetFaceRec(), dwCurrentChar, &dwGlyphIndex));
318*3ac0a46fSAndroid Build Coastguard Worker     // Simple fonts have 1-byte charcodes only.
319*3ac0a46fSAndroid Build Coastguard Worker     if (nextChar > kMaxSimpleFontChar || dwGlyphIndex == 0)
320*3ac0a46fSAndroid Build Coastguard Worker       break;
321*3ac0a46fSAndroid Build Coastguard Worker     for (uint32_t i = dwCurrentChar + 1; i < nextChar; i++)
322*3ac0a46fSAndroid Build Coastguard Worker       widthsArray->AppendNew<CPDF_Number>(0);
323*3ac0a46fSAndroid Build Coastguard Worker     dwCurrentChar = nextChar;
324*3ac0a46fSAndroid Build Coastguard Worker   }
325*3ac0a46fSAndroid Build Coastguard Worker   pFontDict->SetNewFor<CPDF_Number>("LastChar",
326*3ac0a46fSAndroid Build Coastguard Worker                                     static_cast<int>(dwCurrentChar));
327*3ac0a46fSAndroid Build Coastguard Worker   pFontDict->SetNewFor<CPDF_Reference>("Widths", pDoc,
328*3ac0a46fSAndroid Build Coastguard Worker                                        widthsArray->GetObjNum());
329*3ac0a46fSAndroid Build Coastguard Worker   RetainPtr<CPDF_Dictionary> pFontDesc =
330*3ac0a46fSAndroid Build Coastguard Worker       LoadFontDesc(pDoc, name, pFont.get(), span, font_type);
331*3ac0a46fSAndroid Build Coastguard Worker 
332*3ac0a46fSAndroid Build Coastguard Worker   pFontDict->SetNewFor<CPDF_Reference>("FontDescriptor", pDoc,
333*3ac0a46fSAndroid Build Coastguard Worker                                        pFontDesc->GetObjNum());
334*3ac0a46fSAndroid Build Coastguard Worker   return CPDF_DocPageData::FromDocument(pDoc)->GetFont(std::move(pFontDict));
335*3ac0a46fSAndroid Build Coastguard Worker }
336*3ac0a46fSAndroid Build Coastguard Worker 
LoadCompositeFont(CPDF_Document * pDoc,std::unique_ptr<CFX_Font> pFont,pdfium::span<const uint8_t> span,int font_type)337*3ac0a46fSAndroid Build Coastguard Worker RetainPtr<CPDF_Font> LoadCompositeFont(CPDF_Document* pDoc,
338*3ac0a46fSAndroid Build Coastguard Worker                                        std::unique_ptr<CFX_Font> pFont,
339*3ac0a46fSAndroid Build Coastguard Worker                                        pdfium::span<const uint8_t> span,
340*3ac0a46fSAndroid Build Coastguard Worker                                        int font_type) {
341*3ac0a46fSAndroid Build Coastguard Worker   auto pFontDict = pDoc->NewIndirect<CPDF_Dictionary>();
342*3ac0a46fSAndroid Build Coastguard Worker   pFontDict->SetNewFor<CPDF_Name>("Type", "Font");
343*3ac0a46fSAndroid Build Coastguard Worker   pFontDict->SetNewFor<CPDF_Name>("Subtype", "Type0");
344*3ac0a46fSAndroid Build Coastguard Worker   // TODO(npm): Get the correct encoding, if it's not identity.
345*3ac0a46fSAndroid Build Coastguard Worker   ByteString encoding = "Identity-H";
346*3ac0a46fSAndroid Build Coastguard Worker   pFontDict->SetNewFor<CPDF_Name>("Encoding", encoding);
347*3ac0a46fSAndroid Build Coastguard Worker   ByteString name = BaseFontNameForType(pFont.get(), font_type);
348*3ac0a46fSAndroid Build Coastguard Worker   pFontDict->SetNewFor<CPDF_Name>(
349*3ac0a46fSAndroid Build Coastguard Worker       "BaseFont", font_type == FPDF_FONT_TYPE1 ? name + "-" + encoding : name);
350*3ac0a46fSAndroid Build Coastguard Worker 
351*3ac0a46fSAndroid Build Coastguard Worker   auto pCIDFont = pDoc->NewIndirect<CPDF_Dictionary>();
352*3ac0a46fSAndroid Build Coastguard Worker   pCIDFont->SetNewFor<CPDF_Name>("Type", "Font");
353*3ac0a46fSAndroid Build Coastguard Worker   pCIDFont->SetNewFor<CPDF_Name>("Subtype", font_type == FPDF_FONT_TYPE1
354*3ac0a46fSAndroid Build Coastguard Worker                                                 ? "CIDFontType0"
355*3ac0a46fSAndroid Build Coastguard Worker                                                 : "CIDFontType2");
356*3ac0a46fSAndroid Build Coastguard Worker   pCIDFont->SetNewFor<CPDF_Name>("BaseFont", name);
357*3ac0a46fSAndroid Build Coastguard Worker 
358*3ac0a46fSAndroid Build Coastguard Worker   // TODO(npm): Maybe use FT_Get_CID_Registry_Ordering_Supplement to get the
359*3ac0a46fSAndroid Build Coastguard Worker   // CIDSystemInfo
360*3ac0a46fSAndroid Build Coastguard Worker   auto pCIDSystemInfo = pDoc->NewIndirect<CPDF_Dictionary>();
361*3ac0a46fSAndroid Build Coastguard Worker   pCIDSystemInfo->SetNewFor<CPDF_String>("Registry", "Adobe", false);
362*3ac0a46fSAndroid Build Coastguard Worker   pCIDSystemInfo->SetNewFor<CPDF_String>("Ordering", "Identity", false);
363*3ac0a46fSAndroid Build Coastguard Worker   pCIDSystemInfo->SetNewFor<CPDF_Number>("Supplement", 0);
364*3ac0a46fSAndroid Build Coastguard Worker   pCIDFont->SetNewFor<CPDF_Reference>("CIDSystemInfo", pDoc,
365*3ac0a46fSAndroid Build Coastguard Worker                                       pCIDSystemInfo->GetObjNum());
366*3ac0a46fSAndroid Build Coastguard Worker 
367*3ac0a46fSAndroid Build Coastguard Worker   RetainPtr<CPDF_Dictionary> pFontDesc =
368*3ac0a46fSAndroid Build Coastguard Worker       LoadFontDesc(pDoc, name, pFont.get(), span, font_type);
369*3ac0a46fSAndroid Build Coastguard Worker   pCIDFont->SetNewFor<CPDF_Reference>("FontDescriptor", pDoc,
370*3ac0a46fSAndroid Build Coastguard Worker                                       pFontDesc->GetObjNum());
371*3ac0a46fSAndroid Build Coastguard Worker 
372*3ac0a46fSAndroid Build Coastguard Worker   uint32_t dwGlyphIndex;
373*3ac0a46fSAndroid Build Coastguard Worker   uint32_t dwCurrentChar = static_cast<uint32_t>(
374*3ac0a46fSAndroid Build Coastguard Worker       FT_Get_First_Char(pFont->GetFaceRec(), &dwGlyphIndex));
375*3ac0a46fSAndroid Build Coastguard Worker   // If it doesn't have a single char, just fail
376*3ac0a46fSAndroid Build Coastguard Worker   if (dwGlyphIndex == 0 ||
377*3ac0a46fSAndroid Build Coastguard Worker       dwCurrentChar > pdfium::kMaximumSupplementaryCodePoint) {
378*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
379*3ac0a46fSAndroid Build Coastguard Worker   }
380*3ac0a46fSAndroid Build Coastguard Worker 
381*3ac0a46fSAndroid Build Coastguard Worker   std::multimap<uint32_t, uint32_t> to_unicode;
382*3ac0a46fSAndroid Build Coastguard Worker   std::map<uint32_t, uint32_t> widths;
383*3ac0a46fSAndroid Build Coastguard Worker   while (true) {
384*3ac0a46fSAndroid Build Coastguard Worker     if (dwCurrentChar > pdfium::kMaximumSupplementaryCodePoint) {
385*3ac0a46fSAndroid Build Coastguard Worker       break;
386*3ac0a46fSAndroid Build Coastguard Worker     }
387*3ac0a46fSAndroid Build Coastguard Worker 
388*3ac0a46fSAndroid Build Coastguard Worker     if (!pdfium::Contains(widths, dwGlyphIndex))
389*3ac0a46fSAndroid Build Coastguard Worker       widths[dwGlyphIndex] = pFont->GetGlyphWidth(dwGlyphIndex);
390*3ac0a46fSAndroid Build Coastguard Worker     to_unicode.emplace(dwGlyphIndex, dwCurrentChar);
391*3ac0a46fSAndroid Build Coastguard Worker     dwCurrentChar = static_cast<uint32_t>(
392*3ac0a46fSAndroid Build Coastguard Worker         FT_Get_Next_Char(pFont->GetFaceRec(), dwCurrentChar, &dwGlyphIndex));
393*3ac0a46fSAndroid Build Coastguard Worker     if (dwGlyphIndex == 0)
394*3ac0a46fSAndroid Build Coastguard Worker       break;
395*3ac0a46fSAndroid Build Coastguard Worker   }
396*3ac0a46fSAndroid Build Coastguard Worker   auto widthsArray = pDoc->NewIndirect<CPDF_Array>();
397*3ac0a46fSAndroid Build Coastguard Worker   for (auto it = widths.begin(); it != widths.end(); ++it) {
398*3ac0a46fSAndroid Build Coastguard Worker     int ch = it->first;
399*3ac0a46fSAndroid Build Coastguard Worker     int w = it->second;
400*3ac0a46fSAndroid Build Coastguard Worker     if (std::next(it) == widths.end()) {
401*3ac0a46fSAndroid Build Coastguard Worker       // Only one char left, use format c [w]
402*3ac0a46fSAndroid Build Coastguard Worker       auto oneW = pdfium::MakeRetain<CPDF_Array>();
403*3ac0a46fSAndroid Build Coastguard Worker       oneW->AppendNew<CPDF_Number>(w);
404*3ac0a46fSAndroid Build Coastguard Worker       widthsArray->AppendNew<CPDF_Number>(ch);
405*3ac0a46fSAndroid Build Coastguard Worker       widthsArray->Append(oneW);
406*3ac0a46fSAndroid Build Coastguard Worker       break;
407*3ac0a46fSAndroid Build Coastguard Worker     }
408*3ac0a46fSAndroid Build Coastguard Worker     ++it;
409*3ac0a46fSAndroid Build Coastguard Worker     int next_ch = it->first;
410*3ac0a46fSAndroid Build Coastguard Worker     int next_w = it->second;
411*3ac0a46fSAndroid Build Coastguard Worker     if (next_ch == ch + 1 && next_w == w) {
412*3ac0a46fSAndroid Build Coastguard Worker       // The array can have a group c_first c_last w: all CIDs in the range from
413*3ac0a46fSAndroid Build Coastguard Worker       // c_first to c_last will have width w
414*3ac0a46fSAndroid Build Coastguard Worker       widthsArray->AppendNew<CPDF_Number>(ch);
415*3ac0a46fSAndroid Build Coastguard Worker       ch = next_ch;
416*3ac0a46fSAndroid Build Coastguard Worker       while (true) {
417*3ac0a46fSAndroid Build Coastguard Worker         auto next_it = std::next(it);
418*3ac0a46fSAndroid Build Coastguard Worker         if (next_it == widths.end() || next_it->first != it->first + 1 ||
419*3ac0a46fSAndroid Build Coastguard Worker             next_it->second != it->second) {
420*3ac0a46fSAndroid Build Coastguard Worker           break;
421*3ac0a46fSAndroid Build Coastguard Worker         }
422*3ac0a46fSAndroid Build Coastguard Worker         ++it;
423*3ac0a46fSAndroid Build Coastguard Worker         ch = it->first;
424*3ac0a46fSAndroid Build Coastguard Worker       }
425*3ac0a46fSAndroid Build Coastguard Worker       widthsArray->AppendNew<CPDF_Number>(ch);
426*3ac0a46fSAndroid Build Coastguard Worker       widthsArray->AppendNew<CPDF_Number>(w);
427*3ac0a46fSAndroid Build Coastguard Worker       continue;
428*3ac0a46fSAndroid Build Coastguard Worker     }
429*3ac0a46fSAndroid Build Coastguard Worker     // Otherwise we can have a group of the form c [w1 w2 ...]: c has width
430*3ac0a46fSAndroid Build Coastguard Worker     // w1, c+1 has width w2, etc.
431*3ac0a46fSAndroid Build Coastguard Worker     widthsArray->AppendNew<CPDF_Number>(ch);
432*3ac0a46fSAndroid Build Coastguard Worker     auto curWidthArray = pdfium::MakeRetain<CPDF_Array>();
433*3ac0a46fSAndroid Build Coastguard Worker     curWidthArray->AppendNew<CPDF_Number>(w);
434*3ac0a46fSAndroid Build Coastguard Worker     curWidthArray->AppendNew<CPDF_Number>(next_w);
435*3ac0a46fSAndroid Build Coastguard Worker     while (true) {
436*3ac0a46fSAndroid Build Coastguard Worker       auto next_it = std::next(it);
437*3ac0a46fSAndroid Build Coastguard Worker       if (next_it == widths.end() || next_it->first != it->first + 1)
438*3ac0a46fSAndroid Build Coastguard Worker         break;
439*3ac0a46fSAndroid Build Coastguard Worker       ++it;
440*3ac0a46fSAndroid Build Coastguard Worker       curWidthArray->AppendNew<CPDF_Number>(static_cast<int>(it->second));
441*3ac0a46fSAndroid Build Coastguard Worker     }
442*3ac0a46fSAndroid Build Coastguard Worker     widthsArray->Append(curWidthArray);
443*3ac0a46fSAndroid Build Coastguard Worker   }
444*3ac0a46fSAndroid Build Coastguard Worker   pCIDFont->SetNewFor<CPDF_Reference>("W", pDoc, widthsArray->GetObjNum());
445*3ac0a46fSAndroid Build Coastguard Worker 
446*3ac0a46fSAndroid Build Coastguard Worker   // TODO(npm): Support vertical writing
447*3ac0a46fSAndroid Build Coastguard Worker 
448*3ac0a46fSAndroid Build Coastguard Worker   auto pDescendant = pFontDict->SetNewFor<CPDF_Array>("DescendantFonts");
449*3ac0a46fSAndroid Build Coastguard Worker   pDescendant->AppendNew<CPDF_Reference>(pDoc, pCIDFont->GetObjNum());
450*3ac0a46fSAndroid Build Coastguard Worker 
451*3ac0a46fSAndroid Build Coastguard Worker   RetainPtr<CPDF_Stream> toUnicodeStream = LoadUnicode(pDoc, to_unicode);
452*3ac0a46fSAndroid Build Coastguard Worker   pFontDict->SetNewFor<CPDF_Reference>("ToUnicode", pDoc,
453*3ac0a46fSAndroid Build Coastguard Worker                                        toUnicodeStream->GetObjNum());
454*3ac0a46fSAndroid Build Coastguard Worker   return CPDF_DocPageData::FromDocument(pDoc)->GetFont(pFontDict);
455*3ac0a46fSAndroid Build Coastguard Worker }
456*3ac0a46fSAndroid Build Coastguard Worker 
CPDFTextObjectFromFPDFPageObject(FPDF_PAGEOBJECT page_object)457*3ac0a46fSAndroid Build Coastguard Worker CPDF_TextObject* CPDFTextObjectFromFPDFPageObject(FPDF_PAGEOBJECT page_object) {
458*3ac0a46fSAndroid Build Coastguard Worker   auto* obj = CPDFPageObjectFromFPDFPageObject(page_object);
459*3ac0a46fSAndroid Build Coastguard Worker   return obj ? obj->AsText() : nullptr;
460*3ac0a46fSAndroid Build Coastguard Worker }
461*3ac0a46fSAndroid Build Coastguard Worker 
FPDFGlyphPathFromCFXPath(const CFX_Path * path)462*3ac0a46fSAndroid Build Coastguard Worker FPDF_GLYPHPATH FPDFGlyphPathFromCFXPath(const CFX_Path* path) {
463*3ac0a46fSAndroid Build Coastguard Worker   return reinterpret_cast<FPDF_GLYPHPATH>(path);
464*3ac0a46fSAndroid Build Coastguard Worker }
CFXPathFromFPDFGlyphPath(FPDF_GLYPHPATH path)465*3ac0a46fSAndroid Build Coastguard Worker const CFX_Path* CFXPathFromFPDFGlyphPath(FPDF_GLYPHPATH path) {
466*3ac0a46fSAndroid Build Coastguard Worker   return reinterpret_cast<const CFX_Path*>(path);
467*3ac0a46fSAndroid Build Coastguard Worker }
468*3ac0a46fSAndroid Build Coastguard Worker 
469*3ac0a46fSAndroid Build Coastguard Worker }  // namespace
470*3ac0a46fSAndroid Build Coastguard Worker 
471*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV
FPDFPageObj_NewTextObj(FPDF_DOCUMENT document,FPDF_BYTESTRING font,float font_size)472*3ac0a46fSAndroid Build Coastguard Worker FPDFPageObj_NewTextObj(FPDF_DOCUMENT document,
473*3ac0a46fSAndroid Build Coastguard Worker                        FPDF_BYTESTRING font,
474*3ac0a46fSAndroid Build Coastguard Worker                        float font_size) {
475*3ac0a46fSAndroid Build Coastguard Worker   CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document);
476*3ac0a46fSAndroid Build Coastguard Worker   if (!pDoc)
477*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
478*3ac0a46fSAndroid Build Coastguard Worker 
479*3ac0a46fSAndroid Build Coastguard Worker   RetainPtr<CPDF_Font> pFont =
480*3ac0a46fSAndroid Build Coastguard Worker       CPDF_Font::GetStockFont(pDoc, ByteStringView(font));
481*3ac0a46fSAndroid Build Coastguard Worker   if (!pFont)
482*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
483*3ac0a46fSAndroid Build Coastguard Worker 
484*3ac0a46fSAndroid Build Coastguard Worker   auto pTextObj = std::make_unique<CPDF_TextObject>();
485*3ac0a46fSAndroid Build Coastguard Worker   pTextObj->m_TextState.SetFont(std::move(pFont));
486*3ac0a46fSAndroid Build Coastguard Worker   pTextObj->m_TextState.SetFontSize(font_size);
487*3ac0a46fSAndroid Build Coastguard Worker   pTextObj->DefaultStates();
488*3ac0a46fSAndroid Build Coastguard Worker 
489*3ac0a46fSAndroid Build Coastguard Worker   // Caller takes ownership.
490*3ac0a46fSAndroid Build Coastguard Worker   return FPDFPageObjectFromCPDFPageObject(pTextObj.release());
491*3ac0a46fSAndroid Build Coastguard Worker }
492*3ac0a46fSAndroid Build Coastguard Worker 
493*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
FPDFText_SetText(FPDF_PAGEOBJECT text_object,FPDF_WIDESTRING text)494*3ac0a46fSAndroid Build Coastguard Worker FPDFText_SetText(FPDF_PAGEOBJECT text_object, FPDF_WIDESTRING text) {
495*3ac0a46fSAndroid Build Coastguard Worker   CPDF_TextObject* pTextObj = CPDFTextObjectFromFPDFPageObject(text_object);
496*3ac0a46fSAndroid Build Coastguard Worker   if (!pTextObj)
497*3ac0a46fSAndroid Build Coastguard Worker     return false;
498*3ac0a46fSAndroid Build Coastguard Worker 
499*3ac0a46fSAndroid Build Coastguard Worker   WideString encodedText = WideStringFromFPDFWideString(text);
500*3ac0a46fSAndroid Build Coastguard Worker   ByteString byteText;
501*3ac0a46fSAndroid Build Coastguard Worker   for (wchar_t wc : encodedText) {
502*3ac0a46fSAndroid Build Coastguard Worker     pTextObj->GetFont()->AppendChar(
503*3ac0a46fSAndroid Build Coastguard Worker         &byteText, pTextObj->GetFont()->CharCodeFromUnicode(wc));
504*3ac0a46fSAndroid Build Coastguard Worker   }
505*3ac0a46fSAndroid Build Coastguard Worker   pTextObj->SetText(byteText);
506*3ac0a46fSAndroid Build Coastguard Worker   return true;
507*3ac0a46fSAndroid Build Coastguard Worker }
508*3ac0a46fSAndroid Build Coastguard Worker 
509*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
FPDFText_SetCharcodes(FPDF_PAGEOBJECT text_object,const uint32_t * charcodes,size_t count)510*3ac0a46fSAndroid Build Coastguard Worker FPDFText_SetCharcodes(FPDF_PAGEOBJECT text_object,
511*3ac0a46fSAndroid Build Coastguard Worker                       const uint32_t* charcodes,
512*3ac0a46fSAndroid Build Coastguard Worker                       size_t count) {
513*3ac0a46fSAndroid Build Coastguard Worker   CPDF_TextObject* pTextObj = CPDFTextObjectFromFPDFPageObject(text_object);
514*3ac0a46fSAndroid Build Coastguard Worker   if (!pTextObj)
515*3ac0a46fSAndroid Build Coastguard Worker     return false;
516*3ac0a46fSAndroid Build Coastguard Worker 
517*3ac0a46fSAndroid Build Coastguard Worker   if (!charcodes && count)
518*3ac0a46fSAndroid Build Coastguard Worker     return false;
519*3ac0a46fSAndroid Build Coastguard Worker 
520*3ac0a46fSAndroid Build Coastguard Worker   ByteString byte_text;
521*3ac0a46fSAndroid Build Coastguard Worker   if (charcodes) {
522*3ac0a46fSAndroid Build Coastguard Worker     for (size_t i = 0; i < count; ++i) {
523*3ac0a46fSAndroid Build Coastguard Worker       pTextObj->GetFont()->AppendChar(&byte_text, charcodes[i]);
524*3ac0a46fSAndroid Build Coastguard Worker     }
525*3ac0a46fSAndroid Build Coastguard Worker   }
526*3ac0a46fSAndroid Build Coastguard Worker   pTextObj->SetText(byte_text);
527*3ac0a46fSAndroid Build Coastguard Worker   return true;
528*3ac0a46fSAndroid Build Coastguard Worker }
529*3ac0a46fSAndroid Build Coastguard Worker 
FPDFText_LoadFont(FPDF_DOCUMENT document,const uint8_t * data,uint32_t size,int font_type,FPDF_BOOL cid)530*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_FONT FPDF_CALLCONV FPDFText_LoadFont(FPDF_DOCUMENT document,
531*3ac0a46fSAndroid Build Coastguard Worker                                                       const uint8_t* data,
532*3ac0a46fSAndroid Build Coastguard Worker                                                       uint32_t size,
533*3ac0a46fSAndroid Build Coastguard Worker                                                       int font_type,
534*3ac0a46fSAndroid Build Coastguard Worker                                                       FPDF_BOOL cid) {
535*3ac0a46fSAndroid Build Coastguard Worker   CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document);
536*3ac0a46fSAndroid Build Coastguard Worker   if (!pDoc || !data || size == 0 ||
537*3ac0a46fSAndroid Build Coastguard Worker       (font_type != FPDF_FONT_TYPE1 && font_type != FPDF_FONT_TRUETYPE)) {
538*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
539*3ac0a46fSAndroid Build Coastguard Worker   }
540*3ac0a46fSAndroid Build Coastguard Worker 
541*3ac0a46fSAndroid Build Coastguard Worker   auto span = pdfium::make_span(data, size);
542*3ac0a46fSAndroid Build Coastguard Worker   auto pFont = std::make_unique<CFX_Font>();
543*3ac0a46fSAndroid Build Coastguard Worker 
544*3ac0a46fSAndroid Build Coastguard Worker   // TODO(npm): Maybe use FT_Get_X11_Font_Format to check format? Otherwise, we
545*3ac0a46fSAndroid Build Coastguard Worker   // are allowing giving any font that can be loaded on freetype and setting it
546*3ac0a46fSAndroid Build Coastguard Worker   // as any font type.
547*3ac0a46fSAndroid Build Coastguard Worker   if (!pFont->LoadEmbedded(span, /*force_vertical=*/false, /*object_tag=*/0))
548*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
549*3ac0a46fSAndroid Build Coastguard Worker 
550*3ac0a46fSAndroid Build Coastguard Worker   // Caller takes ownership.
551*3ac0a46fSAndroid Build Coastguard Worker   return FPDFFontFromCPDFFont(
552*3ac0a46fSAndroid Build Coastguard Worker       cid ? LoadCompositeFont(pDoc, std::move(pFont), span, font_type).Leak()
553*3ac0a46fSAndroid Build Coastguard Worker           : LoadSimpleFont(pDoc, std::move(pFont), span, font_type).Leak());
554*3ac0a46fSAndroid Build Coastguard Worker }
555*3ac0a46fSAndroid Build Coastguard Worker 
556*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_FONT FPDF_CALLCONV
FPDFText_LoadStandardFont(FPDF_DOCUMENT document,FPDF_BYTESTRING font)557*3ac0a46fSAndroid Build Coastguard Worker FPDFText_LoadStandardFont(FPDF_DOCUMENT document, FPDF_BYTESTRING font) {
558*3ac0a46fSAndroid Build Coastguard Worker   CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document);
559*3ac0a46fSAndroid Build Coastguard Worker   if (!pDoc)
560*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
561*3ac0a46fSAndroid Build Coastguard Worker 
562*3ac0a46fSAndroid Build Coastguard Worker   // Caller takes ownership.
563*3ac0a46fSAndroid Build Coastguard Worker   return FPDFFontFromCPDFFont(
564*3ac0a46fSAndroid Build Coastguard Worker       CPDF_Font::GetStockFont(pDoc, ByteStringView(font)).Leak());
565*3ac0a46fSAndroid Build Coastguard Worker }
566*3ac0a46fSAndroid Build Coastguard Worker 
567*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
FPDFTextObj_GetFontSize(FPDF_PAGEOBJECT text,float * size)568*3ac0a46fSAndroid Build Coastguard Worker FPDFTextObj_GetFontSize(FPDF_PAGEOBJECT text, float* size) {
569*3ac0a46fSAndroid Build Coastguard Worker   if (!size)
570*3ac0a46fSAndroid Build Coastguard Worker     return false;
571*3ac0a46fSAndroid Build Coastguard Worker 
572*3ac0a46fSAndroid Build Coastguard Worker   CPDF_TextObject* pTextObj = CPDFTextObjectFromFPDFPageObject(text);
573*3ac0a46fSAndroid Build Coastguard Worker   if (!pTextObj)
574*3ac0a46fSAndroid Build Coastguard Worker     return false;
575*3ac0a46fSAndroid Build Coastguard Worker 
576*3ac0a46fSAndroid Build Coastguard Worker   *size = pTextObj->GetFontSize();
577*3ac0a46fSAndroid Build Coastguard Worker   return true;
578*3ac0a46fSAndroid Build Coastguard Worker }
579*3ac0a46fSAndroid Build Coastguard Worker 
580*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT unsigned long FPDF_CALLCONV
FPDFTextObj_GetText(FPDF_PAGEOBJECT text_object,FPDF_TEXTPAGE text_page,FPDF_WCHAR * buffer,unsigned long length)581*3ac0a46fSAndroid Build Coastguard Worker FPDFTextObj_GetText(FPDF_PAGEOBJECT text_object,
582*3ac0a46fSAndroid Build Coastguard Worker                     FPDF_TEXTPAGE text_page,
583*3ac0a46fSAndroid Build Coastguard Worker                     FPDF_WCHAR* buffer,
584*3ac0a46fSAndroid Build Coastguard Worker                     unsigned long length) {
585*3ac0a46fSAndroid Build Coastguard Worker   CPDF_TextObject* pTextObj = CPDFTextObjectFromFPDFPageObject(text_object);
586*3ac0a46fSAndroid Build Coastguard Worker   if (!pTextObj)
587*3ac0a46fSAndroid Build Coastguard Worker     return 0;
588*3ac0a46fSAndroid Build Coastguard Worker 
589*3ac0a46fSAndroid Build Coastguard Worker   CPDF_TextPage* pTextPage = CPDFTextPageFromFPDFTextPage(text_page);
590*3ac0a46fSAndroid Build Coastguard Worker   if (!pTextPage)
591*3ac0a46fSAndroid Build Coastguard Worker     return 0;
592*3ac0a46fSAndroid Build Coastguard Worker 
593*3ac0a46fSAndroid Build Coastguard Worker   WideString text = pTextPage->GetTextByObject(pTextObj);
594*3ac0a46fSAndroid Build Coastguard Worker   return Utf16EncodeMaybeCopyAndReturnLength(text, buffer, length);
595*3ac0a46fSAndroid Build Coastguard Worker }
596*3ac0a46fSAndroid Build Coastguard Worker 
597*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BITMAP FPDF_CALLCONV
FPDFTextObj_GetRenderedBitmap(FPDF_DOCUMENT document,FPDF_PAGE page,FPDF_PAGEOBJECT text_object,float scale)598*3ac0a46fSAndroid Build Coastguard Worker FPDFTextObj_GetRenderedBitmap(FPDF_DOCUMENT document,
599*3ac0a46fSAndroid Build Coastguard Worker                               FPDF_PAGE page,
600*3ac0a46fSAndroid Build Coastguard Worker                               FPDF_PAGEOBJECT text_object,
601*3ac0a46fSAndroid Build Coastguard Worker                               float scale) {
602*3ac0a46fSAndroid Build Coastguard Worker   CPDF_Document* doc = CPDFDocumentFromFPDFDocument(document);
603*3ac0a46fSAndroid Build Coastguard Worker   if (!doc)
604*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
605*3ac0a46fSAndroid Build Coastguard Worker 
606*3ac0a46fSAndroid Build Coastguard Worker   CPDF_Page* optional_page = CPDFPageFromFPDFPage(page);
607*3ac0a46fSAndroid Build Coastguard Worker   if (optional_page && optional_page->GetDocument() != doc)
608*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
609*3ac0a46fSAndroid Build Coastguard Worker 
610*3ac0a46fSAndroid Build Coastguard Worker   CPDF_TextObject* text = CPDFTextObjectFromFPDFPageObject(text_object);
611*3ac0a46fSAndroid Build Coastguard Worker   if (!text)
612*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
613*3ac0a46fSAndroid Build Coastguard Worker 
614*3ac0a46fSAndroid Build Coastguard Worker   if (scale <= 0)
615*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
616*3ac0a46fSAndroid Build Coastguard Worker 
617*3ac0a46fSAndroid Build Coastguard Worker   const CFX_Matrix scale_matrix(scale, 0, 0, scale, 0, 0);
618*3ac0a46fSAndroid Build Coastguard Worker   const CFX_FloatRect& text_rect = text->GetRect();
619*3ac0a46fSAndroid Build Coastguard Worker   const CFX_FloatRect scaled_text_rect = scale_matrix.TransformRect(text_rect);
620*3ac0a46fSAndroid Build Coastguard Worker 
621*3ac0a46fSAndroid Build Coastguard Worker   // `rect` has to use integer values. Round up as needed.
622*3ac0a46fSAndroid Build Coastguard Worker   const FX_RECT rect = scaled_text_rect.GetOuterRect();
623*3ac0a46fSAndroid Build Coastguard Worker   if (rect.IsEmpty())
624*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
625*3ac0a46fSAndroid Build Coastguard Worker 
626*3ac0a46fSAndroid Build Coastguard Worker   auto result_bitmap = pdfium::MakeRetain<CFX_DIBitmap>();
627*3ac0a46fSAndroid Build Coastguard Worker   if (!result_bitmap->Create(rect.Width(), rect.Height(), FXDIB_Format::kArgb))
628*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
629*3ac0a46fSAndroid Build Coastguard Worker 
630*3ac0a46fSAndroid Build Coastguard Worker   auto render_context = std::make_unique<CPDF_PageRenderContext>();
631*3ac0a46fSAndroid Build Coastguard Worker   CPDF_PageRenderContext* render_context_ptr = render_context.get();
632*3ac0a46fSAndroid Build Coastguard Worker   CPDF_Page::RenderContextClearer clearer(optional_page);
633*3ac0a46fSAndroid Build Coastguard Worker   if (optional_page)
634*3ac0a46fSAndroid Build Coastguard Worker     optional_page->SetRenderContext(std::move(render_context));
635*3ac0a46fSAndroid Build Coastguard Worker 
636*3ac0a46fSAndroid Build Coastguard Worker   RetainPtr<CPDF_Dictionary> page_resources =
637*3ac0a46fSAndroid Build Coastguard Worker       optional_page ? optional_page->GetMutablePageResources() : nullptr;
638*3ac0a46fSAndroid Build Coastguard Worker 
639*3ac0a46fSAndroid Build Coastguard Worker   auto device = std::make_unique<CFX_DefaultRenderDevice>();
640*3ac0a46fSAndroid Build Coastguard Worker   CFX_DefaultRenderDevice* device_ptr = device.get();
641*3ac0a46fSAndroid Build Coastguard Worker   render_context_ptr->m_pDevice = std::move(device);
642*3ac0a46fSAndroid Build Coastguard Worker   render_context_ptr->m_pContext = std::make_unique<CPDF_RenderContext>(
643*3ac0a46fSAndroid Build Coastguard Worker       doc, std::move(page_resources), /*pPageCache=*/nullptr);
644*3ac0a46fSAndroid Build Coastguard Worker 
645*3ac0a46fSAndroid Build Coastguard Worker   device_ptr->Attach(result_bitmap);
646*3ac0a46fSAndroid Build Coastguard Worker 
647*3ac0a46fSAndroid Build Coastguard Worker   CFX_Matrix device_matrix(rect.Width(), 0, 0, rect.Height(), 0, 0);
648*3ac0a46fSAndroid Build Coastguard Worker   CPDF_RenderStatus status(render_context_ptr->m_pContext.get(), device_ptr);
649*3ac0a46fSAndroid Build Coastguard Worker   status.SetDeviceMatrix(device_matrix);
650*3ac0a46fSAndroid Build Coastguard Worker   status.Initialize(nullptr, nullptr);
651*3ac0a46fSAndroid Build Coastguard Worker 
652*3ac0a46fSAndroid Build Coastguard Worker   // Need to flip the rendering and also move it to fit within `result_bitmap`.
653*3ac0a46fSAndroid Build Coastguard Worker   CFX_Matrix render_matrix(1, 0, 0, -1, -text_rect.left, text_rect.top);
654*3ac0a46fSAndroid Build Coastguard Worker   render_matrix *= scale_matrix;
655*3ac0a46fSAndroid Build Coastguard Worker   status.RenderSingleObject(text, render_matrix);
656*3ac0a46fSAndroid Build Coastguard Worker 
657*3ac0a46fSAndroid Build Coastguard Worker   // Caller takes ownership.
658*3ac0a46fSAndroid Build Coastguard Worker   return FPDFBitmapFromCFXDIBitmap(result_bitmap.Leak());
659*3ac0a46fSAndroid Build Coastguard Worker }
660*3ac0a46fSAndroid Build Coastguard Worker 
FPDFFont_Close(FPDF_FONT font)661*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT void FPDF_CALLCONV FPDFFont_Close(FPDF_FONT font) {
662*3ac0a46fSAndroid Build Coastguard Worker   // Take back ownership from caller and release.
663*3ac0a46fSAndroid Build Coastguard Worker   RetainPtr<CPDF_Font>().Unleak(CPDFFontFromFPDFFont(font));
664*3ac0a46fSAndroid Build Coastguard Worker }
665*3ac0a46fSAndroid Build Coastguard Worker 
666*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV
FPDFPageObj_CreateTextObj(FPDF_DOCUMENT document,FPDF_FONT font,float font_size)667*3ac0a46fSAndroid Build Coastguard Worker FPDFPageObj_CreateTextObj(FPDF_DOCUMENT document,
668*3ac0a46fSAndroid Build Coastguard Worker                           FPDF_FONT font,
669*3ac0a46fSAndroid Build Coastguard Worker                           float font_size) {
670*3ac0a46fSAndroid Build Coastguard Worker   CPDF_Document* pDoc = CPDFDocumentFromFPDFDocument(document);
671*3ac0a46fSAndroid Build Coastguard Worker   CPDF_Font* pFont = CPDFFontFromFPDFFont(font);
672*3ac0a46fSAndroid Build Coastguard Worker   if (!pDoc || !pFont)
673*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
674*3ac0a46fSAndroid Build Coastguard Worker 
675*3ac0a46fSAndroid Build Coastguard Worker   auto pTextObj = std::make_unique<CPDF_TextObject>();
676*3ac0a46fSAndroid Build Coastguard Worker   pTextObj->m_TextState.SetFont(CPDF_DocPageData::FromDocument(pDoc)->GetFont(
677*3ac0a46fSAndroid Build Coastguard Worker       pFont->GetMutableFontDict()));
678*3ac0a46fSAndroid Build Coastguard Worker   pTextObj->m_TextState.SetFontSize(font_size);
679*3ac0a46fSAndroid Build Coastguard Worker   pTextObj->DefaultStates();
680*3ac0a46fSAndroid Build Coastguard Worker   return FPDFPageObjectFromCPDFPageObject(pTextObj.release());
681*3ac0a46fSAndroid Build Coastguard Worker }
682*3ac0a46fSAndroid Build Coastguard Worker 
683*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_TEXT_RENDERMODE FPDF_CALLCONV
FPDFTextObj_GetTextRenderMode(FPDF_PAGEOBJECT text)684*3ac0a46fSAndroid Build Coastguard Worker FPDFTextObj_GetTextRenderMode(FPDF_PAGEOBJECT text) {
685*3ac0a46fSAndroid Build Coastguard Worker   CPDF_TextObject* pTextObj = CPDFTextObjectFromFPDFPageObject(text);
686*3ac0a46fSAndroid Build Coastguard Worker   if (!pTextObj)
687*3ac0a46fSAndroid Build Coastguard Worker     return FPDF_TEXTRENDERMODE_UNKNOWN;
688*3ac0a46fSAndroid Build Coastguard Worker   return static_cast<FPDF_TEXT_RENDERMODE>(pTextObj->GetTextRenderMode());
689*3ac0a46fSAndroid Build Coastguard Worker }
690*3ac0a46fSAndroid Build Coastguard Worker 
691*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
FPDFTextObj_SetTextRenderMode(FPDF_PAGEOBJECT text,FPDF_TEXT_RENDERMODE render_mode)692*3ac0a46fSAndroid Build Coastguard Worker FPDFTextObj_SetTextRenderMode(FPDF_PAGEOBJECT text,
693*3ac0a46fSAndroid Build Coastguard Worker                               FPDF_TEXT_RENDERMODE render_mode) {
694*3ac0a46fSAndroid Build Coastguard Worker   if (render_mode <= FPDF_TEXTRENDERMODE_UNKNOWN ||
695*3ac0a46fSAndroid Build Coastguard Worker       render_mode > FPDF_TEXTRENDERMODE_LAST) {
696*3ac0a46fSAndroid Build Coastguard Worker     return false;
697*3ac0a46fSAndroid Build Coastguard Worker   }
698*3ac0a46fSAndroid Build Coastguard Worker 
699*3ac0a46fSAndroid Build Coastguard Worker   CPDF_TextObject* pTextObj = CPDFTextObjectFromFPDFPageObject(text);
700*3ac0a46fSAndroid Build Coastguard Worker   if (!pTextObj)
701*3ac0a46fSAndroid Build Coastguard Worker     return false;
702*3ac0a46fSAndroid Build Coastguard Worker 
703*3ac0a46fSAndroid Build Coastguard Worker   pTextObj->SetTextRenderMode(static_cast<TextRenderingMode>(render_mode));
704*3ac0a46fSAndroid Build Coastguard Worker   return true;
705*3ac0a46fSAndroid Build Coastguard Worker }
706*3ac0a46fSAndroid Build Coastguard Worker 
FPDFTextObj_GetFont(FPDF_PAGEOBJECT text)707*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_FONT FPDF_CALLCONV FPDFTextObj_GetFont(FPDF_PAGEOBJECT text) {
708*3ac0a46fSAndroid Build Coastguard Worker   CPDF_TextObject* pTextObj = CPDFTextObjectFromFPDFPageObject(text);
709*3ac0a46fSAndroid Build Coastguard Worker   if (!pTextObj)
710*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
711*3ac0a46fSAndroid Build Coastguard Worker 
712*3ac0a46fSAndroid Build Coastguard Worker   // Unretained reference in public API. NOLINTNEXTLINE
713*3ac0a46fSAndroid Build Coastguard Worker   return FPDFFontFromCPDFFont(pTextObj->GetFont());
714*3ac0a46fSAndroid Build Coastguard Worker }
715*3ac0a46fSAndroid Build Coastguard Worker 
716*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT unsigned long FPDF_CALLCONV
FPDFFont_GetFontName(FPDF_FONT font,char * buffer,unsigned long length)717*3ac0a46fSAndroid Build Coastguard Worker FPDFFont_GetFontName(FPDF_FONT font, char* buffer, unsigned long length) {
718*3ac0a46fSAndroid Build Coastguard Worker   auto* pFont = CPDFFontFromFPDFFont(font);
719*3ac0a46fSAndroid Build Coastguard Worker   if (!pFont)
720*3ac0a46fSAndroid Build Coastguard Worker     return 0;
721*3ac0a46fSAndroid Build Coastguard Worker 
722*3ac0a46fSAndroid Build Coastguard Worker   CFX_Font* pCfxFont = pFont->GetFont();
723*3ac0a46fSAndroid Build Coastguard Worker   ByteString name = pCfxFont->GetFamilyName();
724*3ac0a46fSAndroid Build Coastguard Worker   const unsigned long dwStringLen =
725*3ac0a46fSAndroid Build Coastguard Worker       pdfium::base::checked_cast<unsigned long>(name.GetLength() + 1);
726*3ac0a46fSAndroid Build Coastguard Worker   if (buffer && length >= dwStringLen)
727*3ac0a46fSAndroid Build Coastguard Worker     memcpy(buffer, name.c_str(), dwStringLen);
728*3ac0a46fSAndroid Build Coastguard Worker 
729*3ac0a46fSAndroid Build Coastguard Worker   return dwStringLen;
730*3ac0a46fSAndroid Build Coastguard Worker }
731*3ac0a46fSAndroid Build Coastguard Worker 
FPDFFont_GetFontData(FPDF_FONT font,uint8_t * buffer,size_t buflen,size_t * out_buflen)732*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFFont_GetFontData(FPDF_FONT font,
733*3ac0a46fSAndroid Build Coastguard Worker                                                          uint8_t* buffer,
734*3ac0a46fSAndroid Build Coastguard Worker                                                          size_t buflen,
735*3ac0a46fSAndroid Build Coastguard Worker                                                          size_t* out_buflen) {
736*3ac0a46fSAndroid Build Coastguard Worker   auto* cfont = CPDFFontFromFPDFFont(font);
737*3ac0a46fSAndroid Build Coastguard Worker   if (!cfont || !out_buflen)
738*3ac0a46fSAndroid Build Coastguard Worker     return false;
739*3ac0a46fSAndroid Build Coastguard Worker 
740*3ac0a46fSAndroid Build Coastguard Worker   pdfium::span<uint8_t> data = cfont->GetFont()->GetFontSpan();
741*3ac0a46fSAndroid Build Coastguard Worker   if (buffer && buflen >= data.size())
742*3ac0a46fSAndroid Build Coastguard Worker     fxcrt::spancpy(pdfium::make_span(buffer, buflen), data);
743*3ac0a46fSAndroid Build Coastguard Worker   *out_buflen = data.size();
744*3ac0a46fSAndroid Build Coastguard Worker   return true;
745*3ac0a46fSAndroid Build Coastguard Worker }
746*3ac0a46fSAndroid Build Coastguard Worker 
FPDFFont_GetIsEmbedded(FPDF_FONT font)747*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT int FPDF_CALLCONV FPDFFont_GetIsEmbedded(FPDF_FONT font) {
748*3ac0a46fSAndroid Build Coastguard Worker   auto* cfont = CPDFFontFromFPDFFont(font);
749*3ac0a46fSAndroid Build Coastguard Worker   if (!cfont)
750*3ac0a46fSAndroid Build Coastguard Worker     return -1;
751*3ac0a46fSAndroid Build Coastguard Worker   return cfont->IsEmbedded() ? 1 : 0;
752*3ac0a46fSAndroid Build Coastguard Worker }
753*3ac0a46fSAndroid Build Coastguard Worker 
FPDFFont_GetFlags(FPDF_FONT font)754*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT int FPDF_CALLCONV FPDFFont_GetFlags(FPDF_FONT font) {
755*3ac0a46fSAndroid Build Coastguard Worker   auto* pFont = CPDFFontFromFPDFFont(font);
756*3ac0a46fSAndroid Build Coastguard Worker   if (!pFont)
757*3ac0a46fSAndroid Build Coastguard Worker     return -1;
758*3ac0a46fSAndroid Build Coastguard Worker 
759*3ac0a46fSAndroid Build Coastguard Worker   // Return only flags from ISO 32000-1:2008, table 123.
760*3ac0a46fSAndroid Build Coastguard Worker   return pFont->GetFontFlags() & 0x7ffff;
761*3ac0a46fSAndroid Build Coastguard Worker }
762*3ac0a46fSAndroid Build Coastguard Worker 
FPDFFont_GetWeight(FPDF_FONT font)763*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFFont_GetWeight(FPDF_FONT font) {
764*3ac0a46fSAndroid Build Coastguard Worker   auto* pFont = CPDFFontFromFPDFFont(font);
765*3ac0a46fSAndroid Build Coastguard Worker   return pFont ? pFont->GetFontWeight() : -1;
766*3ac0a46fSAndroid Build Coastguard Worker }
767*3ac0a46fSAndroid Build Coastguard Worker 
FPDFFont_GetItalicAngle(FPDF_FONT font,int * angle)768*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFFont_GetItalicAngle(FPDF_FONT font,
769*3ac0a46fSAndroid Build Coastguard Worker                                                             int* angle) {
770*3ac0a46fSAndroid Build Coastguard Worker   auto* pFont = CPDFFontFromFPDFFont(font);
771*3ac0a46fSAndroid Build Coastguard Worker   if (!pFont || !angle)
772*3ac0a46fSAndroid Build Coastguard Worker     return false;
773*3ac0a46fSAndroid Build Coastguard Worker 
774*3ac0a46fSAndroid Build Coastguard Worker   *angle = pFont->GetItalicAngle();
775*3ac0a46fSAndroid Build Coastguard Worker   return true;
776*3ac0a46fSAndroid Build Coastguard Worker }
777*3ac0a46fSAndroid Build Coastguard Worker 
FPDFFont_GetAscent(FPDF_FONT font,float font_size,float * ascent)778*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFFont_GetAscent(FPDF_FONT font,
779*3ac0a46fSAndroid Build Coastguard Worker                                                        float font_size,
780*3ac0a46fSAndroid Build Coastguard Worker                                                        float* ascent) {
781*3ac0a46fSAndroid Build Coastguard Worker   auto* pFont = CPDFFontFromFPDFFont(font);
782*3ac0a46fSAndroid Build Coastguard Worker   if (!pFont || !ascent)
783*3ac0a46fSAndroid Build Coastguard Worker     return false;
784*3ac0a46fSAndroid Build Coastguard Worker 
785*3ac0a46fSAndroid Build Coastguard Worker   *ascent = pFont->GetTypeAscent() * font_size / 1000.f;
786*3ac0a46fSAndroid Build Coastguard Worker   return true;
787*3ac0a46fSAndroid Build Coastguard Worker }
788*3ac0a46fSAndroid Build Coastguard Worker 
FPDFFont_GetDescent(FPDF_FONT font,float font_size,float * descent)789*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFFont_GetDescent(FPDF_FONT font,
790*3ac0a46fSAndroid Build Coastguard Worker                                                         float font_size,
791*3ac0a46fSAndroid Build Coastguard Worker                                                         float* descent) {
792*3ac0a46fSAndroid Build Coastguard Worker   auto* pFont = CPDFFontFromFPDFFont(font);
793*3ac0a46fSAndroid Build Coastguard Worker   if (!pFont || !descent)
794*3ac0a46fSAndroid Build Coastguard Worker     return false;
795*3ac0a46fSAndroid Build Coastguard Worker 
796*3ac0a46fSAndroid Build Coastguard Worker   *descent = pFont->GetTypeDescent() * font_size / 1000.f;
797*3ac0a46fSAndroid Build Coastguard Worker   return true;
798*3ac0a46fSAndroid Build Coastguard Worker }
799*3ac0a46fSAndroid Build Coastguard Worker 
FPDFFont_GetGlyphWidth(FPDF_FONT font,uint32_t glyph,float font_size,float * width)800*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFFont_GetGlyphWidth(FPDF_FONT font,
801*3ac0a46fSAndroid Build Coastguard Worker                                                            uint32_t glyph,
802*3ac0a46fSAndroid Build Coastguard Worker                                                            float font_size,
803*3ac0a46fSAndroid Build Coastguard Worker                                                            float* width) {
804*3ac0a46fSAndroid Build Coastguard Worker   auto* pFont = CPDFFontFromFPDFFont(font);
805*3ac0a46fSAndroid Build Coastguard Worker   if (!pFont || !width)
806*3ac0a46fSAndroid Build Coastguard Worker     return false;
807*3ac0a46fSAndroid Build Coastguard Worker 
808*3ac0a46fSAndroid Build Coastguard Worker   uint32_t charcode = pFont->CharCodeFromUnicode(static_cast<wchar_t>(glyph));
809*3ac0a46fSAndroid Build Coastguard Worker 
810*3ac0a46fSAndroid Build Coastguard Worker   CPDF_CIDFont* pCIDFont = pFont->AsCIDFont();
811*3ac0a46fSAndroid Build Coastguard Worker   if (pCIDFont && pCIDFont->IsVertWriting()) {
812*3ac0a46fSAndroid Build Coastguard Worker     uint16_t cid = pCIDFont->CIDFromCharCode(charcode);
813*3ac0a46fSAndroid Build Coastguard Worker     *width = pCIDFont->GetVertWidth(cid) * font_size / 1000.f;
814*3ac0a46fSAndroid Build Coastguard Worker   } else {
815*3ac0a46fSAndroid Build Coastguard Worker     *width = pFont->GetCharWidthF(charcode) * font_size / 1000.f;
816*3ac0a46fSAndroid Build Coastguard Worker   }
817*3ac0a46fSAndroid Build Coastguard Worker 
818*3ac0a46fSAndroid Build Coastguard Worker   return true;
819*3ac0a46fSAndroid Build Coastguard Worker }
820*3ac0a46fSAndroid Build Coastguard Worker 
821*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_GLYPHPATH FPDF_CALLCONV
FPDFFont_GetGlyphPath(FPDF_FONT font,uint32_t glyph,float font_size)822*3ac0a46fSAndroid Build Coastguard Worker FPDFFont_GetGlyphPath(FPDF_FONT font, uint32_t glyph, float font_size) {
823*3ac0a46fSAndroid Build Coastguard Worker   auto* pFont = CPDFFontFromFPDFFont(font);
824*3ac0a46fSAndroid Build Coastguard Worker   if (!pFont)
825*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
826*3ac0a46fSAndroid Build Coastguard Worker 
827*3ac0a46fSAndroid Build Coastguard Worker   if (!pdfium::base::IsValueInRangeForNumericType<wchar_t>(glyph))
828*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
829*3ac0a46fSAndroid Build Coastguard Worker 
830*3ac0a46fSAndroid Build Coastguard Worker   uint32_t charcode = pFont->CharCodeFromUnicode(static_cast<wchar_t>(glyph));
831*3ac0a46fSAndroid Build Coastguard Worker   std::vector<TextCharPos> pos =
832*3ac0a46fSAndroid Build Coastguard Worker       GetCharPosList(pdfium::make_span(&charcode, 1),
833*3ac0a46fSAndroid Build Coastguard Worker                      pdfium::span<const float>(), pFont, font_size);
834*3ac0a46fSAndroid Build Coastguard Worker   if (pos.empty())
835*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
836*3ac0a46fSAndroid Build Coastguard Worker 
837*3ac0a46fSAndroid Build Coastguard Worker   CFX_Font* pCfxFont;
838*3ac0a46fSAndroid Build Coastguard Worker   if (pos[0].m_FallbackFontPosition == -1) {
839*3ac0a46fSAndroid Build Coastguard Worker     pCfxFont = pFont->GetFont();
840*3ac0a46fSAndroid Build Coastguard Worker     DCHECK(pCfxFont);  // Never null.
841*3ac0a46fSAndroid Build Coastguard Worker   } else {
842*3ac0a46fSAndroid Build Coastguard Worker     pCfxFont = pFont->GetFontFallback(pos[0].m_FallbackFontPosition);
843*3ac0a46fSAndroid Build Coastguard Worker     if (!pCfxFont)
844*3ac0a46fSAndroid Build Coastguard Worker       return nullptr;
845*3ac0a46fSAndroid Build Coastguard Worker   }
846*3ac0a46fSAndroid Build Coastguard Worker 
847*3ac0a46fSAndroid Build Coastguard Worker   const CFX_Path* pPath =
848*3ac0a46fSAndroid Build Coastguard Worker       pCfxFont->LoadGlyphPath(pos[0].m_GlyphIndex, pos[0].m_FontCharWidth);
849*3ac0a46fSAndroid Build Coastguard Worker 
850*3ac0a46fSAndroid Build Coastguard Worker   return FPDFGlyphPathFromCFXPath(pPath);
851*3ac0a46fSAndroid Build Coastguard Worker }
852*3ac0a46fSAndroid Build Coastguard Worker 
853*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT int FPDF_CALLCONV
FPDFGlyphPath_CountGlyphSegments(FPDF_GLYPHPATH glyphpath)854*3ac0a46fSAndroid Build Coastguard Worker FPDFGlyphPath_CountGlyphSegments(FPDF_GLYPHPATH glyphpath) {
855*3ac0a46fSAndroid Build Coastguard Worker   auto* pPath = CFXPathFromFPDFGlyphPath(glyphpath);
856*3ac0a46fSAndroid Build Coastguard Worker   if (!pPath)
857*3ac0a46fSAndroid Build Coastguard Worker     return -1;
858*3ac0a46fSAndroid Build Coastguard Worker 
859*3ac0a46fSAndroid Build Coastguard Worker   return fxcrt::CollectionSize<int>(pPath->GetPoints());
860*3ac0a46fSAndroid Build Coastguard Worker }
861*3ac0a46fSAndroid Build Coastguard Worker 
862*3ac0a46fSAndroid Build Coastguard Worker FPDF_EXPORT FPDF_PATHSEGMENT FPDF_CALLCONV
FPDFGlyphPath_GetGlyphPathSegment(FPDF_GLYPHPATH glyphpath,int index)863*3ac0a46fSAndroid Build Coastguard Worker FPDFGlyphPath_GetGlyphPathSegment(FPDF_GLYPHPATH glyphpath, int index) {
864*3ac0a46fSAndroid Build Coastguard Worker   auto* pPath = CFXPathFromFPDFGlyphPath(glyphpath);
865*3ac0a46fSAndroid Build Coastguard Worker   if (!pPath)
866*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
867*3ac0a46fSAndroid Build Coastguard Worker 
868*3ac0a46fSAndroid Build Coastguard Worker   pdfium::span<const CFX_Path::Point> points = pPath->GetPoints();
869*3ac0a46fSAndroid Build Coastguard Worker   if (!fxcrt::IndexInBounds(points, index))
870*3ac0a46fSAndroid Build Coastguard Worker     return nullptr;
871*3ac0a46fSAndroid Build Coastguard Worker 
872*3ac0a46fSAndroid Build Coastguard Worker   return FPDFPathSegmentFromFXPathPoint(&points[index]);
873*3ac0a46fSAndroid Build Coastguard Worker }
874