Remove a few unused variables, functions, and member variables.
[pdfium.git] / core / src / fpdfapi / fpdf_render / fpdf_render_text.cpp
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/fxge/fx_ge.h"
8 #include "../../../include/fpdfapi/fpdf_render.h"
9 #include "../../../include/fpdfapi/fpdf_pageobj.h"
10 #include "../fpdf_page/pageint.h"
11 #include "render_int.h"
12 extern FX_BOOL IsAvailableMatrix(const CFX_AffineMatrix& matrix);
13 CPDF_Type3Cache::~CPDF_Type3Cache()
14 {
15     FX_POSITION pos = m_SizeMap.GetStartPosition();
16     CFX_ByteString Key;
17     CPDF_Type3Glyphs* pSizeCache = NULL;
18     while(pos) {
19         pSizeCache = (CPDF_Type3Glyphs*)m_SizeMap.GetNextValue(pos);
20         delete pSizeCache;
21     }
22     m_SizeMap.RemoveAll();
23 }
24 CFX_GlyphBitmap* CPDF_Type3Cache::LoadGlyph(FX_DWORD charcode, const CFX_AffineMatrix* pMatrix, FX_FLOAT retinaScaleX, FX_FLOAT retinaScaleY)
25 {
26     _CPDF_UniqueKeyGen keygen;
27     keygen.Generate(4, FXSYS_round(pMatrix->a * 10000), FXSYS_round(pMatrix->b * 10000),
28                     FXSYS_round(pMatrix->c * 10000), FXSYS_round(pMatrix->d * 10000));
29     CFX_ByteStringC FaceGlyphsKey(keygen.m_Key, keygen.m_KeyLen);
30     CPDF_Type3Glyphs* pSizeCache = NULL;
31     if(!m_SizeMap.Lookup(FaceGlyphsKey, (void*&)pSizeCache)) {
32         pSizeCache = FX_NEW CPDF_Type3Glyphs;
33         m_SizeMap.SetAt(FaceGlyphsKey, pSizeCache);
34     }
35     CFX_GlyphBitmap* pGlyphBitmap;
36     if(pSizeCache->m_GlyphMap.Lookup((FX_LPVOID)(FX_UINTPTR)charcode, (void*&)pGlyphBitmap)) {
37         return pGlyphBitmap;
38     }
39     pGlyphBitmap = RenderGlyph(pSizeCache, charcode, pMatrix, retinaScaleX, retinaScaleY);
40     pSizeCache->m_GlyphMap.SetAt((FX_LPVOID)(FX_UINTPTR)charcode, pGlyphBitmap);
41     return pGlyphBitmap;
42 }
43 CPDF_Type3Glyphs::~CPDF_Type3Glyphs()
44 {
45     FX_POSITION pos = m_GlyphMap.GetStartPosition();
46     FX_LPVOID Key;
47     CFX_GlyphBitmap* pGlyphBitmap;
48     while(pos) {
49         m_GlyphMap.GetNextAssoc(pos, Key, (void*&)pGlyphBitmap);
50         delete pGlyphBitmap;
51     }
52 }
53 static int _AdjustBlue(FX_FLOAT pos, int& count, int blues[])
54 {
55     FX_FLOAT min_distance = 1000000.0f * 1.0f;
56     int closest_pos = -1;
57     for (int i = 0; i < count; i ++) {
58         FX_FLOAT distance = (FX_FLOAT)FXSYS_fabs(pos - (FX_FLOAT)blues[i]);
59         if (distance < 1.0f * 80.0f / 100.0f && distance < min_distance) {
60             min_distance = distance;
61             closest_pos = i;
62         }
63     }
64     if (closest_pos >= 0) {
65         return blues[closest_pos];
66     }
67     int new_pos = FXSYS_round(pos);
68     if (count == TYPE3_MAX_BLUES) {
69         return new_pos;
70     }
71     blues[count++] = new_pos;
72     return new_pos;
73 }
74 void CPDF_Type3Glyphs::AdjustBlue(FX_FLOAT top, FX_FLOAT bottom, int& top_line, int& bottom_line)
75 {
76     top_line = _AdjustBlue(top, m_TopBlueCount, m_TopBlue);
77     bottom_line = _AdjustBlue(bottom, m_BottomBlueCount, m_BottomBlue);
78 }
79 static FX_BOOL _IsScanLine1bpp(FX_LPBYTE pBuf, int width)
80 {
81     int size = width / 8;
82     for (int i = 0; i < size; i ++)
83         if (pBuf[i]) {
84             return TRUE;
85         }
86     if (width % 8)
87         if (pBuf[width / 8] & (0xff << (8 - width % 8))) {
88             return TRUE;
89         }
90     return FALSE;
91 }
92 static FX_BOOL _IsScanLine8bpp(FX_LPBYTE pBuf, int width)
93 {
94     for (int i = 0; i < width; i ++)
95         if (pBuf[i] > 0x40) {
96             return TRUE;
97         }
98     return FALSE;
99 }
100 static int _DetectFirstLastScan(const CFX_DIBitmap* pBitmap, FX_BOOL bFirst)
101 {
102     int height = pBitmap->GetHeight(), pitch = pBitmap->GetPitch(), width = pBitmap->GetWidth();
103     int bpp = pBitmap->GetBPP();
104     if (bpp > 8) {
105         width *= bpp / 8;
106     }
107     FX_LPBYTE pBuf = pBitmap->GetBuffer();
108     int line = bFirst ? 0 : height - 1;
109     int line_step = bFirst ? 1 : -1;
110     int line_end = bFirst ? height : -1;
111     while (line != line_end) {
112         if (bpp == 1) {
113             if (_IsScanLine1bpp(pBuf + line * pitch, width)) {
114                 return line;
115             }
116         } else {
117             if (_IsScanLine8bpp(pBuf + line * pitch, width)) {
118                 return line;
119             }
120         }
121         line += line_step;
122     }
123     return -1;
124 }
125 CFX_GlyphBitmap* CPDF_Type3Cache::RenderGlyph(CPDF_Type3Glyphs* pSize, FX_DWORD charcode, const CFX_AffineMatrix* pMatrix, FX_FLOAT retinaScaleX, FX_FLOAT retinaScaleY)
126 {
127     CPDF_Type3Char* pChar = m_pFont->LoadChar(charcode);
128     if (pChar == NULL || pChar->m_pBitmap == NULL) {
129         return NULL;
130     }
131     CFX_DIBitmap* pBitmap = pChar->m_pBitmap;
132     CFX_AffineMatrix image_matrix, text_matrix;
133     image_matrix = pChar->m_ImageMatrix;
134     text_matrix.Set(pMatrix->a, pMatrix->b, pMatrix->c, pMatrix->d, 0, 0);
135     image_matrix.Concat(text_matrix);
136     CFX_DIBitmap* pResBitmap = NULL;
137     int left, top;
138     if (FXSYS_fabs(image_matrix.b) < FXSYS_fabs(image_matrix.a) / 100 && FXSYS_fabs(image_matrix.c) < FXSYS_fabs(image_matrix.d) / 100) {
139         int top_line, bottom_line;
140         top_line = _DetectFirstLastScan(pBitmap, TRUE);
141         bottom_line = _DetectFirstLastScan(pBitmap, FALSE);
142         if (top_line == 0 && bottom_line == pBitmap->GetHeight() - 1) {
143             FX_FLOAT top_y = image_matrix.d + image_matrix.f;
144             FX_FLOAT bottom_y = image_matrix.f;
145             FX_BOOL bFlipped = top_y > bottom_y;
146             if (bFlipped) {
147                 FX_FLOAT temp = top_y;
148                 top_y = bottom_y;
149                 bottom_y = temp;
150             }
151             pSize->AdjustBlue(top_y, bottom_y, top_line, bottom_line);
152             pResBitmap = pBitmap->StretchTo((int)(FXSYS_round(image_matrix.a) * retinaScaleX), (int)((bFlipped ? top_line - bottom_line : bottom_line - top_line) * retinaScaleY));
153             top = top_line;
154             if (image_matrix.a < 0) {
155                 image_matrix.Scale(retinaScaleX, retinaScaleY);
156                 left = FXSYS_round(image_matrix.e + image_matrix.a);
157             } else {
158                 left = FXSYS_round(image_matrix.e);
159             }
160         } else {
161         }
162     }
163     if (pResBitmap == NULL) {
164         image_matrix.Scale(retinaScaleX, retinaScaleY);
165         pResBitmap = pBitmap->TransformTo(&image_matrix, left, top);
166     }
167     if (pResBitmap == NULL) {
168         return NULL;
169     }
170     CFX_GlyphBitmap* pGlyph = FX_NEW CFX_GlyphBitmap;
171     pGlyph->m_Left = left;
172     pGlyph->m_Top = -top;
173     pGlyph->m_Bitmap.TakeOver(pResBitmap);
174     delete pResBitmap;
175     return pGlyph;
176 }
177 void _CPDF_UniqueKeyGen::Generate(int count, ...)
178 {
179     va_list argList;
180     va_start(argList, count);
181     for (int i = 0; i < count; i ++) {
182         int p = va_arg(argList, int);
183         ((FX_DWORD*)m_Key)[i] = p;
184     }
185     va_end(argList);
186     m_KeyLen = count * sizeof(FX_DWORD);
187 }
188 FX_BOOL CPDF_RenderStatus::ProcessText(const CPDF_TextObject* textobj, const CFX_AffineMatrix* pObj2Device, CFX_PathData* pClippingPath)
189 {
190     if(textobj->m_nChars == 0) {
191         return TRUE;
192     }
193     int text_render_mode = textobj->m_TextState.GetObject()->m_TextMode;
194     if (text_render_mode == 3) {
195         return TRUE;
196     }
197     CPDF_Font* pFont = textobj->m_TextState.GetFont();
198     if (pFont->GetFontType() == PDFFONT_TYPE3) {
199         return ProcessType3Text(textobj, pObj2Device);
200     }
201     FX_BOOL bFill = FALSE, bStroke = FALSE, bClip = FALSE;
202     if (pClippingPath) {
203         bClip = TRUE;
204     } else {
205         switch (text_render_mode) {
206             case 0:
207             case 4:
208                 bFill = TRUE;
209                 break;
210             case 1:
211             case 5:
212                 if (pFont->GetFace() == NULL && !(pFont->GetSubstFont()->m_SubstFlags & FXFONT_SUBST_GLYPHPATH)) {
213                     bFill = TRUE;
214                 } else {
215                     bStroke = TRUE;
216                 }
217                 break;
218             case 2:
219             case 6:
220                 if (pFont->GetFace() == NULL && !(pFont->GetSubstFont()->m_SubstFlags & FXFONT_SUBST_GLYPHPATH)) {
221                     bFill = TRUE;
222                 } else {
223                     bFill = bStroke = TRUE;
224                 }
225                 break;
226             case 3:
227             case 7:
228                 return TRUE;
229             default:
230                 bFill = TRUE;
231         }
232     }
233     FX_ARGB stroke_argb = 0, fill_argb = 0;
234     FX_BOOL bPattern = FALSE;
235     if (bStroke) {
236         if (textobj->m_ColorState.GetStrokeColor()->IsPattern()) {
237             bPattern = TRUE;
238         } else {
239             stroke_argb = GetStrokeArgb(textobj);
240         }
241     }
242     if (bFill) {
243         if (textobj->m_ColorState.GetFillColor()->IsPattern()) {
244             bPattern = TRUE;
245         } else {
246             fill_argb = GetFillArgb(textobj);
247         }
248     }
249     CFX_AffineMatrix text_matrix;
250     textobj->GetTextMatrix(&text_matrix);
251     if(IsAvailableMatrix(text_matrix) == FALSE) {
252         return TRUE;
253     }
254     FX_FLOAT font_size = textobj->m_TextState.GetFontSize();
255     if (bPattern) {
256         DrawTextPathWithPattern(textobj, pObj2Device, pFont, font_size, &text_matrix, bFill, bStroke);
257         return TRUE;
258     }
259 #if defined(_FPDFAPI_MINI_)
260     if (bFill) {
261         bStroke = FALSE;
262     }
263     if (bStroke) {
264         if (font_size * text_matrix.GetXUnit() * pObj2Device->GetXUnit() < 6) {
265             bStroke = FALSE;
266         }
267     }
268 #endif
269     if (bClip || bStroke) {
270         const CFX_AffineMatrix* pDeviceMatrix = pObj2Device;
271         CFX_AffineMatrix device_matrix;
272         if (bStroke) {
273             const FX_FLOAT* pCTM = textobj->m_TextState.GetObject()->m_CTM;
274             if (pCTM[0] != 1.0f || pCTM[3] != 1.0f) {
275                 CFX_AffineMatrix ctm(pCTM[0], pCTM[1], pCTM[2], pCTM[3], 0, 0);
276                 text_matrix.ConcatInverse(ctm);
277                 device_matrix.Copy(ctm);
278                 device_matrix.Concat(*pObj2Device);
279                 pDeviceMatrix = &device_matrix;
280             }
281         }
282         int flag = 0;
283         if (bStroke && bFill) {
284             flag |= FX_FILL_STROKE;
285             flag |= FX_STROKE_TEXT_MODE;
286         }
287 #if !defined(_FPDFAPI_MINI_) || defined(_FXCORE_FEATURE_ALL_)
288         const CPDF_GeneralStateData* pGeneralData = ((CPDF_PageObject*)textobj)->m_GeneralState;
289         if (pGeneralData && pGeneralData->m_StrokeAdjust) {
290             flag |= FX_STROKE_ADJUST;
291         }
292 #endif
293         if (m_Options.m_Flags & RENDER_NOTEXTSMOOTH) {
294             flag |= FXFILL_NOPATHSMOOTH;
295         }
296         return CPDF_TextRenderer::DrawTextPath(m_pDevice, textobj->m_nChars, textobj->m_pCharCodes, textobj->m_pCharPos, pFont, font_size,
297                                                &text_matrix, pDeviceMatrix, textobj->m_GraphState, fill_argb, stroke_argb, pClippingPath, flag);
298     }
299     text_matrix.Concat(*pObj2Device);
300     return CPDF_TextRenderer::DrawNormalText(m_pDevice, textobj->m_nChars, textobj->m_pCharCodes, textobj->m_pCharPos, pFont, font_size,
301             &text_matrix, fill_argb, &m_Options);
302 }
303 CPDF_Type3Cache* CPDF_RenderStatus::GetCachedType3(CPDF_Type3Font* pFont)
304 {
305     if (pFont->m_pDocument == NULL) {
306         return NULL;
307     }
308     pFont->m_pDocument->GetPageData()->GetFont(pFont->GetFontDict(), FALSE);
309     return pFont->m_pDocument->GetRenderData()->GetCachedType3(pFont);
310 }
311 static void ReleaseCachedType3(CPDF_Type3Font* pFont)
312 {
313     if (pFont->m_pDocument == NULL) {
314         return;
315     }
316     pFont->m_pDocument->GetRenderData()->ReleaseCachedType3(pFont);
317     pFont->m_pDocument->GetPageData()->ReleaseFont(pFont->GetFontDict());
318 }
319 FX_BOOL CPDF_Type3Char::LoadBitmap(CPDF_RenderContext* pContext)
320 {
321     if (m_pBitmap != NULL || m_pForm == NULL) {
322         return TRUE;
323     }
324     if (m_pForm->CountObjects() == 1 && !m_bColored) {
325         CPDF_PageObject *pPageObj = m_pForm->GetObjectAt(m_pForm->GetFirstObjectPosition());
326         if (pPageObj->m_Type == PDFPAGE_IMAGE) {
327             CPDF_ImageObject* pImage = (CPDF_ImageObject*)pPageObj;
328             m_ImageMatrix = pImage->m_Matrix;
329             const CFX_DIBSource* pSource = pImage->m_pImage->LoadDIBSource();
330             if (pSource) {
331                 m_pBitmap = pSource->Clone();
332                 delete pSource;
333             }
334             delete m_pForm;
335             m_pForm = NULL;
336             return TRUE;
337         }
338         if (pPageObj->m_Type == PDFPAGE_INLINES) {
339             CPDF_InlineImages *pInlines = (CPDF_InlineImages *)pPageObj;
340             if (pInlines->m_pStream) {
341                 m_ImageMatrix = pInlines->m_Matrices[0];
342                 CPDF_DIBSource dibsrc;
343                 if (!dibsrc.Load(pContext->m_pDocument, pInlines->m_pStream, NULL, NULL, NULL, NULL)) {
344                     return FALSE;
345                 }
346                 m_pBitmap = dibsrc.Clone();
347                 delete m_pForm;
348                 m_pForm = NULL;
349                 return TRUE;
350             }
351         }
352     }
353     return FALSE;
354 }
355 class CPDF_RefType3Cache
356 {
357 public:
358     CPDF_RefType3Cache(CPDF_Type3Font* pType3Font)
359     {
360         m_dwCount = 0;
361         m_pType3Font = pType3Font;
362     }
363     ~CPDF_RefType3Cache()
364     {
365         while(m_dwCount--) {
366             ReleaseCachedType3(m_pType3Font);
367         }
368     }
369     FX_DWORD m_dwCount;
370     CPDF_Type3Font* m_pType3Font;
371 };
372 FX_BOOL CPDF_RenderStatus::ProcessType3Text(const CPDF_TextObject* textobj, const CFX_AffineMatrix* pObj2Device)
373 {
374     CPDF_Type3Font* pType3Font = textobj->m_TextState.GetFont()->GetType3Font();
375     for (int j = 0; j < m_Type3FontCache.GetSize(); j++)
376         if ((CPDF_Type3Font*)m_Type3FontCache.GetAt(j) == pType3Font) {
377             return TRUE;
378         }
379     CFX_Matrix dCTM = m_pDevice->GetCTM();
380     FX_FLOAT sa = FXSYS_fabs(dCTM.a);
381     FX_FLOAT sd = FXSYS_fabs(dCTM.d);
382     CFX_AffineMatrix text_matrix;
383     textobj->GetTextMatrix(&text_matrix);
384     CFX_AffineMatrix char_matrix = pType3Font->GetFontMatrix();
385     FX_FLOAT font_size = textobj->m_TextState.GetFontSize();
386     char_matrix.Scale(font_size, font_size);
387     FX_ARGB fill_argb = GetFillArgb(textobj, TRUE);
388     int fill_alpha = FXARGB_A(fill_argb);
389     int device_class = m_pDevice->GetDeviceClass();
390     FXTEXT_GLYPHPOS* pGlyphAndPos = NULL;
391     if (device_class == FXDC_DISPLAY) {
392         pGlyphAndPos = FX_Alloc(FXTEXT_GLYPHPOS, textobj->m_nChars);
393     } else if (fill_alpha < 255) {
394         return FALSE;
395     }
396     CPDF_RefType3Cache refTypeCache(pType3Font);
397     FX_DWORD *pChars = textobj->m_pCharCodes;
398     if (textobj->m_nChars == 1) {
399         pChars = (FX_DWORD*)(&textobj->m_pCharCodes);
400     }
401     for (int iChar = 0; iChar < textobj->m_nChars; iChar ++) {
402         FX_DWORD charcode = pChars[iChar];
403         if (charcode == (FX_DWORD) - 1) {
404             continue;
405         }
406         CPDF_Type3Char* pType3Char = pType3Font->LoadChar(charcode);
407         if (pType3Char == NULL) {
408             continue;
409         }
410         CFX_AffineMatrix matrix = char_matrix;
411         matrix.e += iChar ? textobj->m_pCharPos[iChar - 1] : 0;
412         matrix.Concat(text_matrix);
413         matrix.Concat(*pObj2Device);
414         if (!pType3Char->LoadBitmap(m_pContext)) {
415             if (pGlyphAndPos) {
416                 for (int i = 0; i < iChar; i ++) {
417                     FXTEXT_GLYPHPOS& glyph = pGlyphAndPos[i];
418                     if (glyph.m_pGlyph == NULL) {
419                         continue;
420                     }
421                     m_pDevice->SetBitMask(&glyph.m_pGlyph->m_Bitmap,
422                                           glyph.m_OriginX + glyph.m_pGlyph->m_Left,
423                                           glyph.m_OriginY - glyph.m_pGlyph->m_Top, fill_argb);
424                 }
425                 FX_Free(pGlyphAndPos);
426                 pGlyphAndPos = NULL;
427             }
428             CPDF_GraphicStates* pStates = CloneObjStates(textobj, FALSE);
429             CPDF_RenderOptions Options = m_Options;
430             Options.m_Flags |= RENDER_FORCE_HALFTONE | RENDER_RECT_AA;
431             Options.m_Flags &= ~RENDER_FORCE_DOWNSAMPLE;
432             CPDF_Dictionary* pFormResource = NULL;
433             if (pType3Char->m_pForm && pType3Char->m_pForm->m_pFormDict) {
434                 pFormResource = pType3Char->m_pForm->m_pFormDict->GetDict(FX_BSTRC("Resources"));
435             }
436             if (fill_alpha == 255) {
437                 CPDF_RenderStatus status;
438                 status.Initialize(m_Level + 1, m_pContext, m_pDevice, NULL, NULL, this, pStates, &Options,
439                                   pType3Char->m_pForm->m_Transparency, m_bDropObjects, pFormResource, FALSE, pType3Char, fill_argb);
440                 status.m_Type3FontCache.Append(m_Type3FontCache);
441                 status.m_Type3FontCache.Add(pType3Font);
442                 m_pDevice->SaveState();
443                 status.RenderObjectList(pType3Char->m_pForm, &matrix);
444                 m_pDevice->RestoreState();
445             } else {
446                 CFX_FloatRect rect_f = pType3Char->m_pForm->CalcBoundingBox();
447                 rect_f.Transform(&matrix);
448                 FX_RECT rect = rect_f.GetOutterRect();
449                 CFX_FxgeDevice bitmap_device;
450                 if (!bitmap_device.Create((int)(rect.Width() * sa), (int)(rect.Height() * sd), FXDIB_Argb)) {
451                     return TRUE;
452                 }
453                 bitmap_device.GetBitmap()->Clear(0);
454                 CPDF_RenderStatus status;
455                 status.Initialize(m_Level + 1, m_pContext, &bitmap_device, NULL, NULL, this, pStates, &Options,
456                                   pType3Char->m_pForm->m_Transparency, m_bDropObjects, pFormResource, FALSE, pType3Char, fill_argb);
457                 status.m_Type3FontCache.Append(m_Type3FontCache);
458                 status.m_Type3FontCache.Add(pType3Font);
459                 matrix.TranslateI(-rect.left, -rect.top);
460                 matrix.Scale(sa, sd);
461                 status.RenderObjectList(pType3Char->m_pForm, &matrix);
462                 m_pDevice->SetDIBits(bitmap_device.GetBitmap(), rect.left, rect.top);
463             }
464             delete pStates;
465         } else if (pType3Char->m_pBitmap) {
466             if (device_class == FXDC_DISPLAY) {
467                 CPDF_Type3Cache* pCache = GetCachedType3(pType3Font);
468                 refTypeCache.m_dwCount++;
469                 CFX_GlyphBitmap* pBitmap = pCache->LoadGlyph(charcode, &matrix, sa, sd);
470                 if (pBitmap == NULL) {
471                     continue;
472                 }
473                 int origin_x = FXSYS_round(matrix.e);
474                 int origin_y = FXSYS_round(matrix.f);
475                 if (pGlyphAndPos) {
476                     pGlyphAndPos[iChar].m_pGlyph = pBitmap;
477                     pGlyphAndPos[iChar].m_OriginX = origin_x;
478                     pGlyphAndPos[iChar].m_OriginY = origin_y;
479                 } else {
480                     m_pDevice->SetBitMask(&pBitmap->m_Bitmap, origin_x + pBitmap->m_Left, origin_y - pBitmap->m_Top, fill_argb);
481                 }
482             } else {
483                 CFX_AffineMatrix image_matrix = pType3Char->m_ImageMatrix;
484                 image_matrix.Concat(matrix);
485                 CPDF_ImageRenderer renderer;
486                 if (renderer.Start(this, pType3Char->m_pBitmap, fill_argb, 255, &image_matrix, 0, FALSE)) {
487                     renderer.Continue(NULL);
488                 }
489                 if (!renderer.m_Result) {
490                     return FALSE;
491                 }
492             }
493         }
494     }
495     if (pGlyphAndPos) {
496         FX_RECT rect = FXGE_GetGlyphsBBox(pGlyphAndPos, textobj->m_nChars, 0, sa, sd);
497         CFX_DIBitmap bitmap;
498         if (!bitmap.Create((int)(rect.Width() * sa), (int)(rect.Height() * sd), FXDIB_8bppMask)) {
499             FX_Free(pGlyphAndPos);
500             return TRUE;
501         }
502         bitmap.Clear(0);
503         for (int iChar = 0; iChar < textobj->m_nChars; iChar ++) {
504             FXTEXT_GLYPHPOS& glyph = pGlyphAndPos[iChar];
505             if (glyph.m_pGlyph == NULL) {
506                 continue;
507             }
508             bitmap.TransferBitmap((int)((glyph.m_OriginX + glyph.m_pGlyph->m_Left - rect.left) * sa),
509                                   (int)((glyph.m_OriginY - glyph.m_pGlyph->m_Top - rect.top) * sd),
510                                   glyph.m_pGlyph->m_Bitmap.GetWidth(), glyph.m_pGlyph->m_Bitmap.GetHeight(),
511                                   &glyph.m_pGlyph->m_Bitmap, 0, 0);
512         }
513         m_pDevice->SetBitMask(&bitmap, rect.left, rect.top, fill_argb);
514         FX_Free(pGlyphAndPos);
515     }
516     return TRUE;
517 }
518 class CPDF_CharPosList
519 {
520 public:
521     CPDF_CharPosList();
522     ~CPDF_CharPosList();
523     void                                Load(int nChars, FX_DWORD* pCharCodes, FX_FLOAT* pCharPos, CPDF_Font* pFont, FX_FLOAT font_size);
524     FXTEXT_CHARPOS*             m_pCharPos;
525     FX_DWORD                    m_nChars;
526 };
527 FX_FLOAT _CIDTransformToFloat(FX_BYTE ch);
528 CPDF_CharPosList::CPDF_CharPosList()
529 {
530     m_pCharPos = NULL;
531 }
532 CPDF_CharPosList::~CPDF_CharPosList()
533 {
534     if (m_pCharPos) {
535         FX_Free(m_pCharPos);
536     }
537 }
538 void CPDF_CharPosList::Load(int nChars, FX_DWORD* pCharCodes, FX_FLOAT* pCharPos, CPDF_Font* pFont,
539                             FX_FLOAT FontSize)
540 {
541     m_pCharPos = FX_Alloc(FXTEXT_CHARPOS, nChars);
542     m_nChars = 0;
543     CPDF_CIDFont* pCIDFont = pFont->GetCIDFont();
544     FX_BOOL bVertWriting = pCIDFont && pCIDFont->IsVertWriting();
545     for (int iChar = 0; iChar < nChars; iChar ++) {
546         FX_DWORD CharCode = nChars == 1 ? (FX_DWORD)(FX_UINTPTR)pCharCodes : pCharCodes[iChar];
547         if (CharCode == (FX_DWORD) - 1) {
548             continue;
549         }
550         FX_BOOL bVert = FALSE;
551         FXTEXT_CHARPOS& charpos = m_pCharPos[m_nChars++];
552         if (pCIDFont) {
553             charpos.m_bFontStyle = pCIDFont->IsFontStyleFromCharCode(CharCode);
554         }
555         charpos.m_GlyphIndex = pFont->GlyphFromCharCode(CharCode, &bVert);
556 #if _FXM_PLATFORM_  == _FXM_PLATFORM_APPLE_
557         charpos.m_ExtGID = pFont->GlyphFromCharCodeExt(CharCode);
558 #endif
559         if (!pFont->IsEmbedded() && pFont->GetFontType() != PDFFONT_CIDFONT) {
560             charpos.m_FontCharWidth = pFont->GetCharWidthF(CharCode);
561         } else {
562             charpos.m_FontCharWidth = 0;
563         }
564         charpos.m_OriginX = iChar ? pCharPos[iChar - 1] : 0;
565         charpos.m_OriginY = 0;
566         charpos.m_bGlyphAdjust = FALSE;
567         if (pCIDFont == NULL) {
568             continue;
569         }
570         FX_WORD CID = pCIDFont->CIDFromCharCode(CharCode);
571         if (bVertWriting) {
572             charpos.m_OriginY = charpos.m_OriginX;
573             charpos.m_OriginX = 0;
574             short vx, vy;
575             pCIDFont->GetVertOrigin(CID, vx, vy);
576             charpos.m_OriginX -= FontSize * vx / 1000;
577             charpos.m_OriginY -= FontSize * vy / 1000;
578         }
579         FX_LPCBYTE pTransform = pCIDFont->GetCIDTransform(CID);
580         if (pTransform && !bVert) {
581             charpos.m_AdjustMatrix[0] = _CIDTransformToFloat(pTransform[0]);
582             charpos.m_AdjustMatrix[2] = _CIDTransformToFloat(pTransform[2]);
583             charpos.m_AdjustMatrix[1] = _CIDTransformToFloat(pTransform[1]);
584             charpos.m_AdjustMatrix[3] = _CIDTransformToFloat(pTransform[3]);
585             charpos.m_OriginX += _CIDTransformToFloat(pTransform[4]) * FontSize;
586             charpos.m_OriginY += _CIDTransformToFloat(pTransform[5]) * FontSize;
587             charpos.m_bGlyphAdjust = TRUE;
588         }
589     }
590 }
591 FX_BOOL CPDF_TextRenderer::DrawTextPath(CFX_RenderDevice* pDevice, int nChars, FX_DWORD* pCharCodes, FX_FLOAT* pCharPos,
592                                         CPDF_Font* pFont, FX_FLOAT font_size,
593                                         const CFX_AffineMatrix* pText2User, const CFX_AffineMatrix* pUser2Device,
594                                         const CFX_GraphStateData* pGraphState,
595                                         FX_ARGB fill_argb, FX_ARGB stroke_argb, CFX_PathData* pClippingPath, int nFlag)
596 {
597     CFX_FontCache* pCache = pFont->m_pDocument ? pFont->m_pDocument->GetRenderData()->GetFontCache() : NULL;
598     CPDF_CharPosList CharPosList;
599     CharPosList.Load(nChars, pCharCodes, pCharPos, pFont, font_size);
600     return pDevice->DrawTextPath(CharPosList.m_nChars, CharPosList.m_pCharPos,
601                                  &pFont->m_Font, pCache, font_size, pText2User, pUser2Device,
602                                  pGraphState, fill_argb, stroke_argb, pClippingPath, nFlag);
603 }
604 void CPDF_TextRenderer::DrawTextString(CFX_RenderDevice* pDevice, int left, int top, CPDF_Font* pFont, int height,
605                                        const CFX_ByteString& str, FX_ARGB argb)
606 {
607     FX_RECT font_bbox;
608     pFont->GetFontBBox(font_bbox);
609     FX_FLOAT font_size = (FX_FLOAT)height * 1000.0f / (FX_FLOAT)(font_bbox.top - font_bbox.bottom);
610     FX_FLOAT origin_x = (FX_FLOAT)left;
611     FX_FLOAT origin_y = (FX_FLOAT)top + font_size * (FX_FLOAT)font_bbox.top / 1000.0f;
612     CFX_AffineMatrix matrix(1.0f, 0, 0, -1.0f, 0, 0);
613     DrawTextString(pDevice, origin_x, origin_y, pFont, font_size, &matrix, str, argb);
614 }
615 void CPDF_TextRenderer::DrawTextString(CFX_RenderDevice* pDevice, FX_FLOAT origin_x, FX_FLOAT origin_y, CPDF_Font* pFont, FX_FLOAT font_size,
616                                        const CFX_AffineMatrix* pMatrix, const CFX_ByteString& str, FX_ARGB fill_argb,
617                                        FX_ARGB stroke_argb, const CFX_GraphStateData* pGraphState, const CPDF_RenderOptions* pOptions)
618 {
619     int nChars = pFont->CountChar(str, str.GetLength());
620     if (nChars == 0) {
621         return;
622     }
623     FX_DWORD charcode;
624     int offset = 0;
625     FX_DWORD* pCharCodes;
626     FX_FLOAT* pCharPos;
627     if (nChars == 1) {
628         charcode = pFont->GetNextChar(str, offset);
629         pCharCodes = (FX_DWORD*)(FX_UINTPTR)charcode;
630         pCharPos = NULL;
631     } else {
632         pCharCodes = FX_Alloc(FX_DWORD, nChars);
633         pCharPos = FX_Alloc(FX_FLOAT, nChars - 1);
634         FX_FLOAT cur_pos = 0;
635         for (int i = 0; i < nChars; i ++) {
636             pCharCodes[i] = pFont->GetNextChar(str, offset);
637             if (i) {
638                 pCharPos[i - 1] = cur_pos;
639             }
640             cur_pos += pFont->GetCharWidthF(pCharCodes[i]) * font_size / 1000;
641         }
642     }
643     CFX_AffineMatrix matrix;
644     if (pMatrix) {
645         matrix = *pMatrix;
646     }
647     matrix.e = origin_x;
648     matrix.f = origin_y;
649     if (pFont->GetFontType() == PDFFONT_TYPE3)
650         ;
651     else if (stroke_argb == 0) {
652         DrawNormalText(pDevice, nChars, pCharCodes, pCharPos, pFont, font_size, &matrix, fill_argb, pOptions);
653     } else
654         DrawTextPath(pDevice, nChars, pCharCodes, pCharPos, pFont, font_size, &matrix, NULL, pGraphState,
655                      fill_argb, stroke_argb, NULL);
656     if (nChars > 1) {
657         FX_Free(pCharCodes);
658         FX_Free(pCharPos);
659     }
660 }
661 FX_BOOL CPDF_TextRenderer::DrawNormalText(CFX_RenderDevice* pDevice, int nChars, FX_DWORD* pCharCodes, FX_FLOAT* pCharPos,
662         CPDF_Font* pFont, FX_FLOAT font_size,
663         const CFX_AffineMatrix* pText2Device,
664         FX_ARGB fill_argb, const CPDF_RenderOptions* pOptions)
665 {
666     CFX_FontCache* pCache = pFont->m_pDocument ? pFont->m_pDocument->GetRenderData()->GetFontCache() : NULL;
667     CPDF_CharPosList CharPosList;
668     CharPosList.Load(nChars, pCharCodes, pCharPos, pFont, font_size);
669     int FXGE_flags = 0;
670     if (pOptions) {
671         FX_DWORD dwFlags = pOptions->m_Flags;
672         if (dwFlags & RENDER_CLEARTYPE) {
673             FXGE_flags |= FXTEXT_CLEARTYPE;
674             if (dwFlags & RENDER_BGR_STRIPE) {
675                 FXGE_flags |= FXTEXT_BGR_STRIPE;
676             }
677         }
678         if (dwFlags & RENDER_NOTEXTSMOOTH) {
679             FXGE_flags |= FXTEXT_NOSMOOTH;
680         }
681         if (dwFlags & RENDER_PRINTGRAPHICTEXT) {
682             FXGE_flags |= FXTEXT_PRINTGRAPHICTEXT;
683         }
684         if (dwFlags & RENDER_NO_NATIVETEXT) {
685             FXGE_flags |= FXTEXT_NO_NATIVETEXT;
686         }
687         if (dwFlags & RENDER_PRINTIMAGETEXT) {
688             FXGE_flags |= FXTEXT_PRINTIMAGETEXT;
689         }
690     } else {
691         FXGE_flags = FXTEXT_CLEARTYPE;
692     }
693     if (pFont->GetFontType() & PDFFONT_CIDFONT) {
694         FXGE_flags |= FXFONT_CIDFONT;
695     }
696     return pDevice->DrawNormalText(CharPosList.m_nChars, CharPosList.m_pCharPos, &pFont->m_Font, pCache, font_size, pText2Device, fill_argb, FXGE_flags);
697 }
698 void CPDF_RenderStatus::DrawTextPathWithPattern(const CPDF_TextObject* textobj, const CFX_AffineMatrix* pObj2Device,
699         CPDF_Font* pFont, FX_FLOAT font_size,
700         const CFX_AffineMatrix* pTextMatrix, FX_BOOL bFill, FX_BOOL bStroke)
701 {
702     if (!bStroke) {
703         CPDF_PathObject path;
704         CPDF_TextObject* pCopy = FX_NEW CPDF_TextObject;
705         pCopy->Copy(textobj);
706         path.m_bStroke = FALSE;
707         path.m_FillType = FXFILL_WINDING;
708         path.m_ClipPath.AppendTexts(&pCopy, 1);
709         path.m_ColorState = textobj->m_ColorState;
710         path.m_Path.New()->AppendRect(textobj->m_Left, textobj->m_Bottom, textobj->m_Right, textobj->m_Top);
711         path.m_Left = textobj->m_Left;
712         path.m_Bottom = textobj->m_Bottom;
713         path.m_Right = textobj->m_Right;
714         path.m_Top = textobj->m_Top;
715         RenderSingleObject(&path, pObj2Device);
716         return;
717     }
718     CFX_FontCache* pCache;
719     if (pFont->m_pDocument) {
720         pCache = pFont->m_pDocument->GetRenderData()->GetFontCache();
721     } else {
722         pCache = CFX_GEModule::Get()->GetFontCache();
723     }
724     CFX_FaceCache* pFaceCache = pCache->GetCachedFace(&pFont->m_Font);
725     FX_FONTCACHE_DEFINE(pCache, &pFont->m_Font);
726     CPDF_CharPosList CharPosList;
727     CharPosList.Load(textobj->m_nChars, textobj->m_pCharCodes, textobj->m_pCharPos, pFont, font_size);
728     for (FX_DWORD i = 0; i < CharPosList.m_nChars; i ++) {
729         FXTEXT_CHARPOS& charpos = CharPosList.m_pCharPos[i];
730         const CFX_PathData* pPath = pFaceCache->LoadGlyphPath(&pFont->m_Font, charpos.m_GlyphIndex,
731                                     charpos.m_FontCharWidth);
732         if (pPath == NULL) {
733             continue;
734         }
735         CPDF_PathObject path;
736         path.m_GraphState = textobj->m_GraphState;
737         path.m_ColorState = textobj->m_ColorState;
738         CFX_AffineMatrix matrix;
739         if (charpos.m_bGlyphAdjust)
740             matrix.Set(charpos.m_AdjustMatrix[0], charpos.m_AdjustMatrix[1],
741                        charpos.m_AdjustMatrix[2], charpos.m_AdjustMatrix[3], 0, 0);
742         matrix.Concat(font_size, 0, 0, font_size, charpos.m_OriginX, charpos.m_OriginY);
743         path.m_Path.New()->Append(pPath, &matrix);
744         path.m_Matrix = *pTextMatrix;
745         path.m_bStroke = bStroke;
746         path.m_FillType = bFill ? FXFILL_WINDING : 0;
747         path.CalcBoundingBox();
748         ProcessPath(&path, pObj2Device);
749     }
750 }
751 CFX_PathData* CPDF_Font::LoadGlyphPath(FX_DWORD charcode, int dest_width)
752 {
753     int glyph_index = GlyphFromCharCode(charcode);
754     if (m_Font.m_Face == NULL) {
755         return NULL;
756     }
757     return m_Font.LoadGlyphPath(glyph_index, dest_width);
758 }