• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2014 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 "../../include/formfiller/FormFiller.h"
8 #include "../../include/formfiller/FFL_CBA_Fontmap.h"
9 
CBA_FontMap(CPDFSDK_Annot * pAnnot,IFX_SystemHandler * pSystemHandler)10 CBA_FontMap::CBA_FontMap(CPDFSDK_Annot* pAnnot, IFX_SystemHandler* pSystemHandler) :
11 	CPWL_FontMap(pSystemHandler),
12 	m_pDocument(NULL),
13 	m_pAnnotDict(NULL),
14 	m_pDefaultFont(NULL),
15 	m_sAPType("N")
16 {
17 	ASSERT(pAnnot != NULL);
18 
19 	CPDF_Page* pPage = pAnnot->GetPDFPage();
20 
21 	m_pDocument = pPage->m_pDocument;
22 	m_pAnnotDict = pAnnot->GetPDFAnnot()->m_pAnnotDict;
23 }
24 
CBA_FontMap(CPDF_Document * pDocument,CPDF_Dictionary * pAnnotDict,IFX_SystemHandler * pSystemHandler)25 CBA_FontMap::CBA_FontMap(CPDF_Document* pDocument, CPDF_Dictionary* pAnnotDict,
26 						 IFX_SystemHandler* pSystemHandler) :
27 	CPWL_FontMap(pSystemHandler),
28 	m_pDocument(pDocument),
29 	m_pAnnotDict(pAnnotDict),
30 	m_pDefaultFont(NULL),
31 	m_sAPType("N")
32 {
33 }
34 
~CBA_FontMap()35 CBA_FontMap::~CBA_FontMap()
36 {
37 }
38 
Reset()39 void CBA_FontMap::Reset()
40 {
41 	Empty();
42 	m_pDefaultFont = NULL;
43 	m_sDefaultFontName = "";
44 }
45 
Initial(FX_LPCSTR fontname)46 void CBA_FontMap::Initial(FX_LPCSTR fontname)
47 {
48 	FX_INT32 nCharset = DEFAULT_CHARSET;
49 
50 	if (!m_pDefaultFont)
51 	{
52 		m_pDefaultFont = GetAnnotDefaultFont(m_sDefaultFontName);
53 		if (m_pDefaultFont)
54 		{
55 			if (const CFX_SubstFont* pSubstFont = m_pDefaultFont->GetSubstFont())
56 				nCharset = pSubstFont->m_Charset;
57 			else
58 			{
59 				if (m_sDefaultFontName == "Wingdings" || m_sDefaultFontName == "Wingdings2" ||
60 					m_sDefaultFontName == "Wingdings3" || m_sDefaultFontName == "Webdings")
61 						nCharset = SYMBOL_CHARSET;
62 				else
63 					nCharset = ANSI_CHARSET;
64 			}
65 			AddFontData(m_pDefaultFont, m_sDefaultFontName, nCharset);
66 			AddFontToAnnotDict(m_pDefaultFont, m_sDefaultFontName);
67 		}
68 	}
69 
70 	if (nCharset != ANSI_CHARSET)
71 		CPWL_FontMap::Initial(fontname);
72 }
73 
SetDefaultFont(CPDF_Font * pFont,const CFX_ByteString & sFontName)74 void CBA_FontMap::SetDefaultFont(CPDF_Font * pFont, const CFX_ByteString & sFontName)
75 {
76 	ASSERT(pFont != NULL);
77 
78 	if (m_pDefaultFont) return;
79 
80 	m_pDefaultFont = pFont;
81 	m_sDefaultFontName = sFontName;
82 
83 //	if (m_sDefaultFontName.IsEmpty())
84 //		m_sDefaultFontName = pFont->GetFontTypeName();
85 
86 	FX_INT32 nCharset = DEFAULT_CHARSET;
87 	if (const CFX_SubstFont* pSubstFont = m_pDefaultFont->GetSubstFont())
88 		nCharset = pSubstFont->m_Charset;
89 	AddFontData(m_pDefaultFont, m_sDefaultFontName, nCharset);
90 }
91 
FindFontSameCharset(CFX_ByteString & sFontAlias,FX_INT32 nCharset)92 CPDF_Font* CBA_FontMap::FindFontSameCharset(CFX_ByteString& sFontAlias, FX_INT32 nCharset)
93 {
94 	ASSERT(m_pAnnotDict != NULL);
95 
96 	if (m_pAnnotDict->GetString("Subtype") == "Widget")
97 	{
98 		CPDF_Document* pDocument = GetDocument();
99 		ASSERT(pDocument != NULL);
100 
101 		CPDF_Dictionary * pRootDict = pDocument->GetRoot();
102 		if (!pRootDict) return NULL;
103 
104 		CPDF_Dictionary* pAcroFormDict = pRootDict->GetDict("AcroForm");
105 		if (!pAcroFormDict) return NULL;
106 
107 		CPDF_Dictionary * pDRDict = pAcroFormDict->GetDict("DR");
108 		if (!pDRDict) return NULL;
109 
110 		return FindResFontSameCharset(pDRDict, sFontAlias, nCharset);
111 	}
112 
113 	return NULL;
114 }
115 
GetDocument()116 CPDF_Document* CBA_FontMap::GetDocument()
117 {
118 	return m_pDocument;
119 }
120 
FindResFontSameCharset(CPDF_Dictionary * pResDict,CFX_ByteString & sFontAlias,FX_INT32 nCharset)121 CPDF_Font* CBA_FontMap::FindResFontSameCharset(CPDF_Dictionary* pResDict, CFX_ByteString& sFontAlias,
122 													FX_INT32 nCharset)
123 {
124 	if (!pResDict) return NULL;
125 
126 	CPDF_Document* pDocument = GetDocument();
127 	ASSERT(pDocument != NULL);
128 
129 	CPDF_Dictionary* pFonts = pResDict->GetDict("Font");
130 	if (pFonts == NULL) return NULL;
131 
132 	CPDF_Font* pFind = NULL;
133 
134 	FX_POSITION pos = pFonts->GetStartPos();
135 	while (pos)
136 	{
137 		CPDF_Object* pObj = NULL;
138 		CFX_ByteString csKey;
139 		pObj = pFonts->GetNextElement(pos, csKey);
140 		if (pObj == NULL) continue;
141 
142 		CPDF_Object* pDirect = pObj->GetDirect();
143 		if (pDirect == NULL || pDirect->GetType() != PDFOBJ_DICTIONARY) continue;
144 
145 		CPDF_Dictionary* pElement = (CPDF_Dictionary*)pDirect;
146 		if (pElement->GetString("Type") != "Font") continue;
147 
148 		CPDF_Font* pFont = pDocument->LoadFont(pElement);
149 		if (pFont == NULL) continue;
150 		const CFX_SubstFont* pSubst = pFont->GetSubstFont();
151 		if (pSubst == NULL) continue;
152 		if (pSubst->m_Charset == nCharset)
153 		{
154 			sFontAlias = csKey;
155 			pFind = pFont;
156 		}
157 	}
158 	return pFind;
159 }
160 
AddedFont(CPDF_Font * pFont,const CFX_ByteString & sFontAlias)161 void CBA_FontMap::AddedFont(CPDF_Font* pFont, const CFX_ByteString& sFontAlias)
162 {
163 	AddFontToAnnotDict(pFont, sFontAlias);
164 }
165 
AddFontToAnnotDict(CPDF_Font * pFont,const CFX_ByteString & sAlias)166 void CBA_FontMap::AddFontToAnnotDict(CPDF_Font* pFont, const CFX_ByteString& sAlias)
167 {
168 	if (!pFont)	return;
169 
170 	ASSERT(m_pAnnotDict != NULL);
171 	ASSERT(m_pDocument != NULL);
172 
173 	CPDF_Dictionary* pAPDict = m_pAnnotDict->GetDict("AP");
174 
175 	if (pAPDict == NULL)
176 	{
177 		pAPDict = FX_NEW CPDF_Dictionary;
178 		m_pAnnotDict->SetAt("AP", pAPDict);
179 	}
180 
181 	//to avoid checkbox and radiobutton
182 	CPDF_Object* pObject = pAPDict->GetElement(m_sAPType);
183 	if (pObject && pObject->GetType() == PDFOBJ_DICTIONARY)
184 		return;
185 
186 	CPDF_Stream* pStream = pAPDict->GetStream(m_sAPType);
187 	if (pStream == NULL)
188 	{
189 		pStream = FX_NEW CPDF_Stream(NULL, 0, NULL);
190 		FX_INT32 objnum = m_pDocument->AddIndirectObject(pStream);
191 		pAPDict->SetAtReference(m_sAPType, m_pDocument, objnum);
192 	}
193 
194 	CPDF_Dictionary * pStreamDict = pStream->GetDict();
195 
196 	if (!pStreamDict)
197 	{
198 		pStreamDict = FX_NEW CPDF_Dictionary;
199 		pStream->InitStream(NULL, 0, pStreamDict);
200 	}
201 
202 	if (pStreamDict)
203 	{
204 		CPDF_Dictionary* pStreamResList = pStreamDict->GetDict("Resources");
205 		if (!pStreamResList)
206 		{
207 			pStreamResList = FX_NEW CPDF_Dictionary();
208 			pStreamDict->SetAt("Resources", pStreamResList);
209 		}
210 
211 		if (pStreamResList)
212 		{
213 			CPDF_Dictionary* pStreamResFontList = pStreamResList->GetDict("Font");
214 			if (!pStreamResFontList)
215 			{
216 				pStreamResFontList = FX_NEW CPDF_Dictionary;
217 				FX_INT32 objnum = m_pDocument->AddIndirectObject(pStreamResFontList);
218 				pStreamResList->SetAtReference("Font", m_pDocument, objnum);
219 			}
220 			if (!pStreamResFontList->KeyExist(sAlias))
221 				pStreamResFontList->SetAtReference(sAlias, m_pDocument, pFont->GetFontDict());
222 		}
223 	}
224 }
225 
GetAnnotDefaultFont(CFX_ByteString & sAlias)226 CPDF_Font* CBA_FontMap::GetAnnotDefaultFont(CFX_ByteString &sAlias)
227 {
228 	ASSERT(m_pAnnotDict != NULL);
229 	ASSERT(m_pDocument != NULL);
230 
231 	CPDF_Dictionary* pAcroFormDict = NULL;
232 
233 	FX_BOOL bWidget = (m_pAnnotDict->GetString("Subtype") == "Widget");
234 
235 	if (bWidget)
236 	{
237 		if (CPDF_Dictionary * pRootDict = m_pDocument->GetRoot())
238 			pAcroFormDict = pRootDict->GetDict("AcroForm");
239 	}
240 
241 	CFX_ByteString sDA;
242 
243 	sDA = FPDF_GetFieldAttr(m_pAnnotDict, "DA")->GetString();
244 
245 	if (bWidget)
246 	{
247 		if (sDA.IsEmpty())
248 		{
249 			sDA = FPDF_GetFieldAttr(pAcroFormDict, "DA")->GetString();
250 		}
251 	}
252 
253 	CPDF_Dictionary * pFontDict = NULL;
254 
255 	if (!sDA.IsEmpty())
256 	{
257 		CPDF_SimpleParser syntax(sDA);
258 		syntax.FindTagParam("Tf", 2);
259 		CFX_ByteString sFontName = syntax.GetWord();
260 		sAlias = PDF_NameDecode(sFontName).Mid(1);
261 
262 		if (CPDF_Dictionary * pDRDict = m_pAnnotDict->GetDict("DR"))
263 			if (CPDF_Dictionary* pDRFontDict = pDRDict->GetDict("Font"))
264 				pFontDict = pDRFontDict->GetDict(sAlias);
265 
266 		if (!pFontDict)
267 			if (CPDF_Dictionary* pAPDict = m_pAnnotDict->GetDict("AP"))
268 				if (CPDF_Dictionary* pNormalDict = pAPDict->GetDict("N"))
269 					if (CPDF_Dictionary* pNormalResDict = pNormalDict->GetDict("Resources"))
270 						if (CPDF_Dictionary* pResFontDict = pNormalResDict->GetDict("Font"))
271 							pFontDict = pResFontDict->GetDict(sAlias);
272 
273 		if (bWidget)
274 		{
275 			if (!pFontDict)
276 			{
277 				if (pAcroFormDict)
278 				{
279 					if (CPDF_Dictionary * pDRDict = pAcroFormDict->GetDict("DR"))
280 						if (CPDF_Dictionary* pDRFontDict = pDRDict->GetDict("Font"))
281 							pFontDict = pDRFontDict->GetDict(sAlias);
282 				}
283 			}
284 		}
285 	}
286 
287 	if (pFontDict)
288 		return m_pDocument->LoadFont(pFontDict);
289 	else
290 		return NULL;
291 }
292 
SetAPType(const CFX_ByteString & sAPType)293 void CBA_FontMap::SetAPType(const CFX_ByteString& sAPType)
294 {
295 	m_sAPType = sAPType;
296 
297 	Reset();
298 	Initial();
299 }
300 
301