1 // Copyright 2016 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 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6
7 #include "core/fxge/android/cfpf_skiafont.h"
8
9 #include <algorithm>
10
11 #include "core/fxcrt/fx_system.h"
12 #include "core/fxge/android/cfpf_skiabufferfont.h"
13 #include "core/fxge/android/cfpf_skiafilefont.h"
14 #include "core/fxge/android/cfpf_skiafontdescriptor.h"
15 #include "core/fxge/android/cfpf_skiafontmgr.h"
16 #include "core/fxge/android/cfpf_skiapathfont.h"
17 #include "core/fxge/fx_freetype.h"
18
19 #define FPF_EM_ADJUST(em, a) (em == 0 ? (a) : (a)*1000 / em)
20
CFPF_SkiaFont()21 CFPF_SkiaFont::CFPF_SkiaFont()
22 : m_pFontMgr(nullptr),
23 m_pFontDes(nullptr),
24 m_Face(nullptr),
25 m_dwStyle(0),
26 m_uCharset(0),
27 m_dwRefCount(0) {}
28
~CFPF_SkiaFont()29 CFPF_SkiaFont::~CFPF_SkiaFont() {
30 if (m_Face)
31 FXFT_Done_Face(m_Face);
32 }
33
Release()34 void CFPF_SkiaFont::Release() {
35 if (--m_dwRefCount == 0)
36 delete this;
37 }
38
Retain()39 CFPF_SkiaFont* CFPF_SkiaFont::Retain() {
40 m_dwRefCount++;
41 return this;
42 }
43
GetFamilyName()44 ByteString CFPF_SkiaFont::GetFamilyName() {
45 if (!m_Face)
46 return ByteString();
47 return ByteString(FXFT_Get_Face_Family_Name(m_Face));
48 }
49
GetPsName()50 ByteString CFPF_SkiaFont::GetPsName() {
51 if (!m_Face)
52 return ByteString();
53 return FXFT_Get_Postscript_Name(m_Face);
54 }
55
GetGlyphIndex(wchar_t wUnicode)56 int32_t CFPF_SkiaFont::GetGlyphIndex(wchar_t wUnicode) {
57 if (!m_Face)
58 return wUnicode;
59 if (FXFT_Select_Charmap(m_Face, FXFT_ENCODING_UNICODE))
60 return 0;
61 return FXFT_Get_Char_Index(m_Face, wUnicode);
62 }
63
GetGlyphWidth(int32_t iGlyphIndex)64 int32_t CFPF_SkiaFont::GetGlyphWidth(int32_t iGlyphIndex) {
65 if (!m_Face)
66 return 0;
67 if (FXFT_Load_Glyph(
68 m_Face, iGlyphIndex,
69 FXFT_LOAD_NO_SCALE | FXFT_LOAD_IGNORE_GLOBAL_ADVANCE_WIDTH)) {
70 return 0;
71 }
72 return FPF_EM_ADJUST(FXFT_Get_Face_UnitsPerEM(m_Face),
73 FXFT_Get_Glyph_HoriAdvance(m_Face));
74 }
75
GetAscent() const76 int32_t CFPF_SkiaFont::GetAscent() const {
77 if (!m_Face)
78 return 0;
79 return FPF_EM_ADJUST(FXFT_Get_Face_UnitsPerEM(m_Face),
80 FXFT_Get_Face_Ascender(m_Face));
81 }
82
GetDescent() const83 int32_t CFPF_SkiaFont::GetDescent() const {
84 if (!m_Face)
85 return 0;
86 return FPF_EM_ADJUST(FXFT_Get_Face_UnitsPerEM(m_Face),
87 FXFT_Get_Face_Descender(m_Face));
88 }
89
GetGlyphBBox(int32_t iGlyphIndex,FX_RECT & rtBBox)90 bool CFPF_SkiaFont::GetGlyphBBox(int32_t iGlyphIndex, FX_RECT& rtBBox) {
91 if (!m_Face)
92 return false;
93 if (FXFT_Is_Face_Tricky(m_Face)) {
94 if (FXFT_Set_Char_Size(m_Face, 0, 1000 * 64, 72, 72))
95 return false;
96 if (FXFT_Load_Glyph(m_Face, iGlyphIndex,
97 FXFT_LOAD_IGNORE_GLOBAL_ADVANCE_WIDTH)) {
98 FXFT_Set_Pixel_Sizes(m_Face, 0, 64);
99 return false;
100 }
101 FXFT_Glyph glyph;
102 if (FXFT_Get_Glyph(m_Face->glyph, &glyph)) {
103 FXFT_Set_Pixel_Sizes(m_Face, 0, 64);
104 return false;
105 }
106 FXFT_BBox cbox;
107 FXFT_Glyph_Get_CBox(glyph, FXFT_GLYPH_BBOX_PIXELS, &cbox);
108 int32_t x_ppem = m_Face->size->metrics.x_ppem;
109 int32_t y_ppem = m_Face->size->metrics.y_ppem;
110 rtBBox.left = FPF_EM_ADJUST(x_ppem, cbox.xMin);
111 rtBBox.right = FPF_EM_ADJUST(x_ppem, cbox.xMax);
112 rtBBox.top = FPF_EM_ADJUST(y_ppem, cbox.yMax);
113 rtBBox.bottom = FPF_EM_ADJUST(y_ppem, cbox.yMin);
114 rtBBox.top = std::min(rtBBox.top, GetAscent());
115 rtBBox.bottom = std::max(rtBBox.bottom, GetDescent());
116 FXFT_Done_Glyph(glyph);
117 return FXFT_Set_Pixel_Sizes(m_Face, 0, 64) == 0;
118 }
119 if (FXFT_Load_Glyph(
120 m_Face, iGlyphIndex,
121 FXFT_LOAD_NO_SCALE | FXFT_LOAD_IGNORE_GLOBAL_ADVANCE_WIDTH)) {
122 return false;
123 }
124 rtBBox.left = FPF_EM_ADJUST(FXFT_Get_Face_UnitsPerEM(m_Face),
125 FXFT_Get_Glyph_HoriBearingX(m_Face));
126 rtBBox.bottom = FPF_EM_ADJUST(FXFT_Get_Face_UnitsPerEM(m_Face),
127 FXFT_Get_Glyph_HoriBearingY(m_Face));
128 rtBBox.right = FPF_EM_ADJUST(
129 FXFT_Get_Face_UnitsPerEM(m_Face),
130 FXFT_Get_Glyph_HoriBearingX(m_Face) + FXFT_Get_Glyph_Width(m_Face));
131 rtBBox.top = FPF_EM_ADJUST(
132 FXFT_Get_Face_UnitsPerEM(m_Face),
133 FXFT_Get_Glyph_HoriBearingY(m_Face) - FXFT_Get_Glyph_Height(m_Face));
134 return true;
135 }
136
GetBBox(FX_RECT & rtBBox)137 bool CFPF_SkiaFont::GetBBox(FX_RECT& rtBBox) {
138 if (!m_Face) {
139 return false;
140 }
141 rtBBox.left = FPF_EM_ADJUST(FXFT_Get_Face_UnitsPerEM(m_Face),
142 FXFT_Get_Face_xMin(m_Face));
143 rtBBox.top = FPF_EM_ADJUST(FXFT_Get_Face_UnitsPerEM(m_Face),
144 FXFT_Get_Face_yMin(m_Face));
145 rtBBox.right = FPF_EM_ADJUST(FXFT_Get_Face_UnitsPerEM(m_Face),
146 FXFT_Get_Face_xMax(m_Face));
147 rtBBox.bottom = FPF_EM_ADJUST(FXFT_Get_Face_UnitsPerEM(m_Face),
148 FXFT_Get_Face_yMax(m_Face));
149 return true;
150 }
151
GetHeight() const152 int32_t CFPF_SkiaFont::GetHeight() const {
153 if (!m_Face)
154 return 0;
155 return FPF_EM_ADJUST(FXFT_Get_Face_UnitsPerEM(m_Face),
156 FXFT_Get_Face_Height(m_Face));
157 }
158
GetItalicAngle() const159 int32_t CFPF_SkiaFont::GetItalicAngle() const {
160 if (!m_Face)
161 return 0;
162
163 TT_Postscript* ttInfo =
164 (TT_Postscript*)FT_Get_Sfnt_Table(m_Face, ft_sfnt_post);
165 if (ttInfo)
166 return ttInfo->italicAngle;
167 return 0;
168 }
169
GetFontData(uint32_t dwTable,uint8_t * pBuffer,uint32_t dwSize)170 uint32_t CFPF_SkiaFont::GetFontData(uint32_t dwTable,
171 uint8_t* pBuffer,
172 uint32_t dwSize) {
173 if (!m_Face)
174 return 0;
175
176 FT_ULong ulSize = pdfium::base::checked_cast<FT_ULong>(dwSize);
177 if (FXFT_Load_Sfnt_Table(m_Face, dwTable, 0, pBuffer, &ulSize))
178 return 0;
179 return pdfium::base::checked_cast<uint32_t>(ulSize);
180 }
181
InitFont(CFPF_SkiaFontMgr * pFontMgr,CFPF_SkiaFontDescriptor * pFontDes,const ByteStringView & bsFamily,uint32_t dwStyle,uint8_t uCharset)182 bool CFPF_SkiaFont::InitFont(CFPF_SkiaFontMgr* pFontMgr,
183 CFPF_SkiaFontDescriptor* pFontDes,
184 const ByteStringView& bsFamily,
185 uint32_t dwStyle,
186 uint8_t uCharset) {
187 if (!pFontMgr || !pFontDes)
188 return false;
189
190 switch (pFontDes->GetType()) {
191 case FPF_SKIAFONTTYPE_Path: {
192 CFPF_SkiaPathFont* pFont = (CFPF_SkiaPathFont*)pFontDes;
193 m_Face = pFontMgr->GetFontFace(pFont->m_pPath, pFont->m_iFaceIndex);
194 break;
195 }
196 case FPF_SKIAFONTTYPE_File: {
197 CFPF_SkiaFileFont* pFont = (CFPF_SkiaFileFont*)pFontDes;
198 m_Face = pFontMgr->GetFontFace(pFont->m_pFile, pFont->m_iFaceIndex);
199 break;
200 }
201 case FPF_SKIAFONTTYPE_Buffer: {
202 CFPF_SkiaBufferFont* pFont = (CFPF_SkiaBufferFont*)pFontDes;
203 m_Face = pFontMgr->GetFontFace((const uint8_t*)pFont->m_pBuffer,
204 pFont->m_szBuffer, pFont->m_iFaceIndex);
205 break;
206 }
207 default:
208 return false;
209 }
210 if (!m_Face)
211 return false;
212
213 m_dwStyle = dwStyle;
214 m_uCharset = uCharset;
215 m_pFontMgr = pFontMgr;
216 m_pFontDes = pFontDes;
217 m_dwRefCount = 1;
218 return true;
219 }
220