Index: core/src/fpdfapi/fpdf_render/fpdf_render_text.cpp |
diff --git a/core/src/fpdfapi/fpdf_render/fpdf_render_text.cpp b/core/src/fpdfapi/fpdf_render/fpdf_render_text.cpp |
index ad5dc58a1c0735b1c8c90b9c28b1862989b06474..7d6921f544cd27146a2b637df5c64075460220a8 100644 |
--- a/core/src/fpdfapi/fpdf_render/fpdf_render_text.cpp |
+++ b/core/src/fpdfapi/fpdf_render/fpdf_render_text.cpp |
@@ -10,723 +10,791 @@ |
#include "../fpdf_page/pageint.h" |
#include "render_int.h" |
extern FX_BOOL IsAvailableMatrix(const CFX_AffineMatrix& matrix); |
-CPDF_Type3Cache::~CPDF_Type3Cache() |
-{ |
- FX_POSITION pos = m_SizeMap.GetStartPosition(); |
- CFX_ByteString Key; |
- CPDF_Type3Glyphs* pSizeCache = NULL; |
- while(pos) { |
- pSizeCache = (CPDF_Type3Glyphs*)m_SizeMap.GetNextValue(pos); |
- delete pSizeCache; |
- } |
- m_SizeMap.RemoveAll(); |
+CPDF_Type3Cache::~CPDF_Type3Cache() { |
+ FX_POSITION pos = m_SizeMap.GetStartPosition(); |
+ CFX_ByteString Key; |
+ CPDF_Type3Glyphs* pSizeCache = NULL; |
+ while (pos) { |
+ pSizeCache = (CPDF_Type3Glyphs*)m_SizeMap.GetNextValue(pos); |
+ delete pSizeCache; |
+ } |
+ m_SizeMap.RemoveAll(); |
} |
-CFX_GlyphBitmap* CPDF_Type3Cache::LoadGlyph(FX_DWORD charcode, const CFX_AffineMatrix* pMatrix, FX_FLOAT retinaScaleX, FX_FLOAT retinaScaleY) |
-{ |
- _CPDF_UniqueKeyGen keygen; |
- keygen.Generate(4, FXSYS_round(pMatrix->a * 10000), FXSYS_round(pMatrix->b * 10000), |
- FXSYS_round(pMatrix->c * 10000), FXSYS_round(pMatrix->d * 10000)); |
- CFX_ByteStringC FaceGlyphsKey(keygen.m_Key, keygen.m_KeyLen); |
- CPDF_Type3Glyphs* pSizeCache = NULL; |
- if(!m_SizeMap.Lookup(FaceGlyphsKey, (void*&)pSizeCache)) { |
- pSizeCache = new CPDF_Type3Glyphs; |
- m_SizeMap.SetAt(FaceGlyphsKey, pSizeCache); |
- } |
- CFX_GlyphBitmap* pGlyphBitmap; |
- if(pSizeCache->m_GlyphMap.Lookup((void*)(uintptr_t)charcode, (void*&)pGlyphBitmap)) { |
- return pGlyphBitmap; |
- } |
- pGlyphBitmap = RenderGlyph(pSizeCache, charcode, pMatrix, retinaScaleX, retinaScaleY); |
- pSizeCache->m_GlyphMap.SetAt((void*)(uintptr_t)charcode, pGlyphBitmap); |
+CFX_GlyphBitmap* CPDF_Type3Cache::LoadGlyph(FX_DWORD charcode, |
+ const CFX_AffineMatrix* pMatrix, |
+ FX_FLOAT retinaScaleX, |
+ FX_FLOAT retinaScaleY) { |
+ _CPDF_UniqueKeyGen keygen; |
+ keygen.Generate( |
+ 4, FXSYS_round(pMatrix->a * 10000), FXSYS_round(pMatrix->b * 10000), |
+ FXSYS_round(pMatrix->c * 10000), FXSYS_round(pMatrix->d * 10000)); |
+ CFX_ByteStringC FaceGlyphsKey(keygen.m_Key, keygen.m_KeyLen); |
+ CPDF_Type3Glyphs* pSizeCache = NULL; |
+ if (!m_SizeMap.Lookup(FaceGlyphsKey, (void*&)pSizeCache)) { |
+ pSizeCache = new CPDF_Type3Glyphs; |
+ m_SizeMap.SetAt(FaceGlyphsKey, pSizeCache); |
+ } |
+ CFX_GlyphBitmap* pGlyphBitmap; |
+ if (pSizeCache->m_GlyphMap.Lookup((void*)(uintptr_t)charcode, |
+ (void*&)pGlyphBitmap)) { |
return pGlyphBitmap; |
+ } |
+ pGlyphBitmap = |
+ RenderGlyph(pSizeCache, charcode, pMatrix, retinaScaleX, retinaScaleY); |
+ pSizeCache->m_GlyphMap.SetAt((void*)(uintptr_t)charcode, pGlyphBitmap); |
+ return pGlyphBitmap; |
} |
-CPDF_Type3Glyphs::~CPDF_Type3Glyphs() |
-{ |
- FX_POSITION pos = m_GlyphMap.GetStartPosition(); |
- void* Key; |
- CFX_GlyphBitmap* pGlyphBitmap; |
- while(pos) { |
- m_GlyphMap.GetNextAssoc(pos, Key, (void*&)pGlyphBitmap); |
- delete pGlyphBitmap; |
- } |
+CPDF_Type3Glyphs::~CPDF_Type3Glyphs() { |
+ FX_POSITION pos = m_GlyphMap.GetStartPosition(); |
+ void* Key; |
+ CFX_GlyphBitmap* pGlyphBitmap; |
+ while (pos) { |
+ m_GlyphMap.GetNextAssoc(pos, Key, (void*&)pGlyphBitmap); |
+ delete pGlyphBitmap; |
+ } |
} |
-static int _AdjustBlue(FX_FLOAT pos, int& count, int blues[]) |
-{ |
- FX_FLOAT min_distance = 1000000.0f * 1.0f; |
- int closest_pos = -1; |
- for (int i = 0; i < count; i ++) { |
- FX_FLOAT distance = (FX_FLOAT)FXSYS_fabs(pos - (FX_FLOAT)blues[i]); |
- if (distance < 1.0f * 80.0f / 100.0f && distance < min_distance) { |
- min_distance = distance; |
- closest_pos = i; |
- } |
- } |
- if (closest_pos >= 0) { |
- return blues[closest_pos]; |
- } |
- int new_pos = FXSYS_round(pos); |
- if (count == TYPE3_MAX_BLUES) { |
- return new_pos; |
- } |
- blues[count++] = new_pos; |
+static int _AdjustBlue(FX_FLOAT pos, int& count, int blues[]) { |
+ FX_FLOAT min_distance = 1000000.0f * 1.0f; |
+ int closest_pos = -1; |
+ for (int i = 0; i < count; i++) { |
+ FX_FLOAT distance = (FX_FLOAT)FXSYS_fabs(pos - (FX_FLOAT)blues[i]); |
+ if (distance < 1.0f * 80.0f / 100.0f && distance < min_distance) { |
+ min_distance = distance; |
+ closest_pos = i; |
+ } |
+ } |
+ if (closest_pos >= 0) { |
+ return blues[closest_pos]; |
+ } |
+ int new_pos = FXSYS_round(pos); |
+ if (count == TYPE3_MAX_BLUES) { |
return new_pos; |
+ } |
+ blues[count++] = new_pos; |
+ return new_pos; |
} |
-void CPDF_Type3Glyphs::AdjustBlue(FX_FLOAT top, FX_FLOAT bottom, int& top_line, int& bottom_line) |
-{ |
- top_line = _AdjustBlue(top, m_TopBlueCount, m_TopBlue); |
- bottom_line = _AdjustBlue(bottom, m_BottomBlueCount, m_BottomBlue); |
+void CPDF_Type3Glyphs::AdjustBlue(FX_FLOAT top, |
+ FX_FLOAT bottom, |
+ int& top_line, |
+ int& bottom_line) { |
+ top_line = _AdjustBlue(top, m_TopBlueCount, m_TopBlue); |
+ bottom_line = _AdjustBlue(bottom, m_BottomBlueCount, m_BottomBlue); |
} |
-static FX_BOOL _IsScanLine1bpp(uint8_t* pBuf, int width) |
-{ |
- int size = width / 8; |
- for (int i = 0; i < size; i ++) |
- if (pBuf[i]) { |
- return TRUE; |
- } |
- if (width % 8) |
- if (pBuf[width / 8] & (0xff << (8 - width % 8))) { |
- return TRUE; |
- } |
- return FALSE; |
-} |
-static FX_BOOL _IsScanLine8bpp(uint8_t* pBuf, int width) |
-{ |
- for (int i = 0; i < width; i ++) |
- if (pBuf[i] > 0x40) { |
- return TRUE; |
- } |
- return FALSE; |
+static FX_BOOL _IsScanLine1bpp(uint8_t* pBuf, int width) { |
+ int size = width / 8; |
+ for (int i = 0; i < size; i++) |
+ if (pBuf[i]) { |
+ return TRUE; |
+ } |
+ if (width % 8) |
+ if (pBuf[width / 8] & (0xff << (8 - width % 8))) { |
+ return TRUE; |
+ } |
+ return FALSE; |
} |
-static int _DetectFirstLastScan(const CFX_DIBitmap* pBitmap, FX_BOOL bFirst) |
-{ |
- int height = pBitmap->GetHeight(), pitch = pBitmap->GetPitch(), width = pBitmap->GetWidth(); |
- int bpp = pBitmap->GetBPP(); |
- if (bpp > 8) { |
- width *= bpp / 8; |
- } |
- uint8_t* pBuf = pBitmap->GetBuffer(); |
- int line = bFirst ? 0 : height - 1; |
- int line_step = bFirst ? 1 : -1; |
- int line_end = bFirst ? height : -1; |
- while (line != line_end) { |
- if (bpp == 1) { |
- if (_IsScanLine1bpp(pBuf + line * pitch, width)) { |
- return line; |
- } |
- } else { |
- if (_IsScanLine8bpp(pBuf + line * pitch, width)) { |
- return line; |
- } |
- } |
- line += line_step; |
+static FX_BOOL _IsScanLine8bpp(uint8_t* pBuf, int width) { |
+ for (int i = 0; i < width; i++) |
+ if (pBuf[i] > 0x40) { |
+ return TRUE; |
} |
- return -1; |
+ return FALSE; |
} |
-CFX_GlyphBitmap* CPDF_Type3Cache::RenderGlyph(CPDF_Type3Glyphs* pSize, FX_DWORD charcode, const CFX_AffineMatrix* pMatrix, FX_FLOAT retinaScaleX, FX_FLOAT retinaScaleY) |
-{ |
- CPDF_Type3Char* pChar = m_pFont->LoadChar(charcode); |
- if (pChar == NULL || pChar->m_pBitmap == NULL) { |
- return NULL; |
- } |
- CFX_DIBitmap* pBitmap = pChar->m_pBitmap; |
- CFX_AffineMatrix image_matrix, text_matrix; |
- image_matrix = pChar->m_ImageMatrix; |
- text_matrix.Set(pMatrix->a, pMatrix->b, pMatrix->c, pMatrix->d, 0, 0); |
- image_matrix.Concat(text_matrix); |
- CFX_DIBitmap* pResBitmap = NULL; |
- int left, top; |
- if (FXSYS_fabs(image_matrix.b) < FXSYS_fabs(image_matrix.a) / 100 && FXSYS_fabs(image_matrix.c) < FXSYS_fabs(image_matrix.d) / 100) { |
- int top_line, bottom_line; |
- top_line = _DetectFirstLastScan(pBitmap, TRUE); |
- bottom_line = _DetectFirstLastScan(pBitmap, FALSE); |
- if (top_line == 0 && bottom_line == pBitmap->GetHeight() - 1) { |
- FX_FLOAT top_y = image_matrix.d + image_matrix.f; |
- FX_FLOAT bottom_y = image_matrix.f; |
- FX_BOOL bFlipped = top_y > bottom_y; |
- if (bFlipped) { |
- FX_FLOAT temp = top_y; |
- top_y = bottom_y; |
- bottom_y = temp; |
- } |
- pSize->AdjustBlue(top_y, bottom_y, top_line, bottom_line); |
- pResBitmap = pBitmap->StretchTo((int)(FXSYS_round(image_matrix.a) * retinaScaleX), (int)((bFlipped ? top_line - bottom_line : bottom_line - top_line) * retinaScaleY)); |
- top = top_line; |
- if (image_matrix.a < 0) { |
- image_matrix.Scale(retinaScaleX, retinaScaleY); |
- left = FXSYS_round(image_matrix.e + image_matrix.a); |
- } else { |
- left = FXSYS_round(image_matrix.e); |
- } |
- } else { |
- } |
+static int _DetectFirstLastScan(const CFX_DIBitmap* pBitmap, FX_BOOL bFirst) { |
+ int height = pBitmap->GetHeight(), pitch = pBitmap->GetPitch(), |
+ width = pBitmap->GetWidth(); |
+ int bpp = pBitmap->GetBPP(); |
+ if (bpp > 8) { |
+ width *= bpp / 8; |
+ } |
+ uint8_t* pBuf = pBitmap->GetBuffer(); |
+ int line = bFirst ? 0 : height - 1; |
+ int line_step = bFirst ? 1 : -1; |
+ int line_end = bFirst ? height : -1; |
+ while (line != line_end) { |
+ if (bpp == 1) { |
+ if (_IsScanLine1bpp(pBuf + line * pitch, width)) { |
+ return line; |
+ } |
+ } else { |
+ if (_IsScanLine8bpp(pBuf + line * pitch, width)) { |
+ return line; |
+ } |
} |
- if (pResBitmap == NULL) { |
+ line += line_step; |
+ } |
+ return -1; |
+} |
+CFX_GlyphBitmap* CPDF_Type3Cache::RenderGlyph(CPDF_Type3Glyphs* pSize, |
+ FX_DWORD charcode, |
+ const CFX_AffineMatrix* pMatrix, |
+ FX_FLOAT retinaScaleX, |
+ FX_FLOAT retinaScaleY) { |
+ CPDF_Type3Char* pChar = m_pFont->LoadChar(charcode); |
+ if (pChar == NULL || pChar->m_pBitmap == NULL) { |
+ return NULL; |
+ } |
+ CFX_DIBitmap* pBitmap = pChar->m_pBitmap; |
+ CFX_AffineMatrix image_matrix, text_matrix; |
+ image_matrix = pChar->m_ImageMatrix; |
+ text_matrix.Set(pMatrix->a, pMatrix->b, pMatrix->c, pMatrix->d, 0, 0); |
+ image_matrix.Concat(text_matrix); |
+ CFX_DIBitmap* pResBitmap = NULL; |
+ int left, top; |
+ if (FXSYS_fabs(image_matrix.b) < FXSYS_fabs(image_matrix.a) / 100 && |
+ FXSYS_fabs(image_matrix.c) < FXSYS_fabs(image_matrix.d) / 100) { |
+ int top_line, bottom_line; |
+ top_line = _DetectFirstLastScan(pBitmap, TRUE); |
+ bottom_line = _DetectFirstLastScan(pBitmap, FALSE); |
+ if (top_line == 0 && bottom_line == pBitmap->GetHeight() - 1) { |
+ FX_FLOAT top_y = image_matrix.d + image_matrix.f; |
+ FX_FLOAT bottom_y = image_matrix.f; |
+ FX_BOOL bFlipped = top_y > bottom_y; |
+ if (bFlipped) { |
+ FX_FLOAT temp = top_y; |
+ top_y = bottom_y; |
+ bottom_y = temp; |
+ } |
+ pSize->AdjustBlue(top_y, bottom_y, top_line, bottom_line); |
+ pResBitmap = pBitmap->StretchTo( |
+ (int)(FXSYS_round(image_matrix.a) * retinaScaleX), |
+ (int)((bFlipped ? top_line - bottom_line : bottom_line - top_line) * |
+ retinaScaleY)); |
+ top = top_line; |
+ if (image_matrix.a < 0) { |
image_matrix.Scale(retinaScaleX, retinaScaleY); |
- pResBitmap = pBitmap->TransformTo(&image_matrix, left, top); |
- } |
- if (pResBitmap == NULL) { |
- return NULL; |
+ left = FXSYS_round(image_matrix.e + image_matrix.a); |
+ } else { |
+ left = FXSYS_round(image_matrix.e); |
+ } |
+ } else { |
} |
- CFX_GlyphBitmap* pGlyph = new CFX_GlyphBitmap; |
- pGlyph->m_Left = left; |
- pGlyph->m_Top = -top; |
- pGlyph->m_Bitmap.TakeOver(pResBitmap); |
- delete pResBitmap; |
- return pGlyph; |
+ } |
+ if (pResBitmap == NULL) { |
+ image_matrix.Scale(retinaScaleX, retinaScaleY); |
+ pResBitmap = pBitmap->TransformTo(&image_matrix, left, top); |
+ } |
+ if (pResBitmap == NULL) { |
+ return NULL; |
+ } |
+ CFX_GlyphBitmap* pGlyph = new CFX_GlyphBitmap; |
+ pGlyph->m_Left = left; |
+ pGlyph->m_Top = -top; |
+ pGlyph->m_Bitmap.TakeOver(pResBitmap); |
+ delete pResBitmap; |
+ return pGlyph; |
} |
-void _CPDF_UniqueKeyGen::Generate(int count, ...) |
-{ |
- va_list argList; |
- va_start(argList, count); |
- for (int i = 0; i < count; i ++) { |
- int p = va_arg(argList, int); |
- ((FX_DWORD*)m_Key)[i] = p; |
- } |
- va_end(argList); |
- m_KeyLen = count * sizeof(FX_DWORD); |
+void _CPDF_UniqueKeyGen::Generate(int count, ...) { |
+ va_list argList; |
+ va_start(argList, count); |
+ for (int i = 0; i < count; i++) { |
+ int p = va_arg(argList, int); |
+ ((FX_DWORD*)m_Key)[i] = p; |
+ } |
+ va_end(argList); |
+ m_KeyLen = count * sizeof(FX_DWORD); |
} |
-FX_BOOL CPDF_RenderStatus::ProcessText(const CPDF_TextObject* textobj, const CFX_AffineMatrix* pObj2Device, CFX_PathData* pClippingPath) |
-{ |
- if(textobj->m_nChars == 0) { |
- return TRUE; |
- } |
- int text_render_mode = textobj->m_TextState.GetObject()->m_TextMode; |
- if (text_render_mode == 3) { |
- return TRUE; |
- } |
- CPDF_Font* pFont = textobj->m_TextState.GetFont(); |
- if (pFont->GetFontType() == PDFFONT_TYPE3) { |
- return ProcessType3Text(textobj, pObj2Device); |
- } |
- FX_BOOL bFill = FALSE, bStroke = FALSE, bClip = FALSE; |
- if (pClippingPath) { |
- bClip = TRUE; |
- } else { |
- switch (text_render_mode) { |
- case 0: |
- case 4: |
- bFill = TRUE; |
- break; |
- case 1: |
- case 5: |
- if (pFont->GetFace() == NULL && !(pFont->GetSubstFont()->m_SubstFlags & FXFONT_SUBST_GLYPHPATH)) { |
- bFill = TRUE; |
- } else { |
- bStroke = TRUE; |
- } |
- break; |
- case 2: |
- case 6: |
- if (pFont->GetFace() == NULL && !(pFont->GetSubstFont()->m_SubstFlags & FXFONT_SUBST_GLYPHPATH)) { |
- bFill = TRUE; |
- } else { |
- bFill = bStroke = TRUE; |
- } |
- break; |
- case 3: |
- case 7: |
- return TRUE; |
- default: |
- bFill = TRUE; |
- } |
- } |
- FX_ARGB stroke_argb = 0, fill_argb = 0; |
- FX_BOOL bPattern = FALSE; |
- if (bStroke) { |
- if (textobj->m_ColorState.GetStrokeColor()->IsPattern()) { |
- bPattern = TRUE; |
+FX_BOOL CPDF_RenderStatus::ProcessText(const CPDF_TextObject* textobj, |
+ const CFX_AffineMatrix* pObj2Device, |
+ CFX_PathData* pClippingPath) { |
+ if (textobj->m_nChars == 0) { |
+ return TRUE; |
+ } |
+ int text_render_mode = textobj->m_TextState.GetObject()->m_TextMode; |
+ if (text_render_mode == 3) { |
+ return TRUE; |
+ } |
+ CPDF_Font* pFont = textobj->m_TextState.GetFont(); |
+ if (pFont->GetFontType() == PDFFONT_TYPE3) { |
+ return ProcessType3Text(textobj, pObj2Device); |
+ } |
+ FX_BOOL bFill = FALSE, bStroke = FALSE, bClip = FALSE; |
+ if (pClippingPath) { |
+ bClip = TRUE; |
+ } else { |
+ switch (text_render_mode) { |
+ case 0: |
+ case 4: |
+ bFill = TRUE; |
+ break; |
+ case 1: |
+ case 5: |
+ if (pFont->GetFace() == NULL && |
+ !(pFont->GetSubstFont()->m_SubstFlags & FXFONT_SUBST_GLYPHPATH)) { |
+ bFill = TRUE; |
} else { |
- stroke_argb = GetStrokeArgb(textobj); |
- } |
- } |
- if (bFill) { |
- if (textobj->m_ColorState.GetFillColor()->IsPattern()) { |
- bPattern = TRUE; |
+ bStroke = TRUE; |
+ } |
+ break; |
+ case 2: |
+ case 6: |
+ if (pFont->GetFace() == NULL && |
+ !(pFont->GetSubstFont()->m_SubstFlags & FXFONT_SUBST_GLYPHPATH)) { |
+ bFill = TRUE; |
} else { |
- fill_argb = GetFillArgb(textobj); |
+ bFill = bStroke = TRUE; |
} |
- } |
- CFX_AffineMatrix text_matrix; |
- textobj->GetTextMatrix(&text_matrix); |
- if(IsAvailableMatrix(text_matrix) == FALSE) { |
- return TRUE; |
- } |
- FX_FLOAT font_size = textobj->m_TextState.GetFontSize(); |
- if (bPattern) { |
- DrawTextPathWithPattern(textobj, pObj2Device, pFont, font_size, &text_matrix, bFill, bStroke); |
+ break; |
+ case 3: |
+ case 7: |
return TRUE; |
+ default: |
+ bFill = TRUE; |
+ } |
+ } |
+ FX_ARGB stroke_argb = 0, fill_argb = 0; |
+ FX_BOOL bPattern = FALSE; |
+ if (bStroke) { |
+ if (textobj->m_ColorState.GetStrokeColor()->IsPattern()) { |
+ bPattern = TRUE; |
+ } else { |
+ stroke_argb = GetStrokeArgb(textobj); |
} |
- if (bClip || bStroke) { |
- const CFX_AffineMatrix* pDeviceMatrix = pObj2Device; |
- CFX_AffineMatrix device_matrix; |
- if (bStroke) { |
- const FX_FLOAT* pCTM = textobj->m_TextState.GetObject()->m_CTM; |
- if (pCTM[0] != 1.0f || pCTM[3] != 1.0f) { |
- CFX_AffineMatrix ctm(pCTM[0], pCTM[1], pCTM[2], pCTM[3], 0, 0); |
- text_matrix.ConcatInverse(ctm); |
- device_matrix.Copy(ctm); |
- device_matrix.Concat(*pObj2Device); |
- pDeviceMatrix = &device_matrix; |
- } |
- } |
- int flag = 0; |
- if (bStroke && bFill) { |
- flag |= FX_FILL_STROKE; |
- flag |= FX_STROKE_TEXT_MODE; |
- } |
- const CPDF_GeneralStateData* pGeneralData = ((CPDF_PageObject*)textobj)->m_GeneralState; |
- if (pGeneralData && pGeneralData->m_StrokeAdjust) { |
- flag |= FX_STROKE_ADJUST; |
- } |
- if (m_Options.m_Flags & RENDER_NOTEXTSMOOTH) { |
- flag |= FXFILL_NOPATHSMOOTH; |
- } |
- return CPDF_TextRenderer::DrawTextPath(m_pDevice, textobj->m_nChars, textobj->m_pCharCodes, textobj->m_pCharPos, pFont, font_size, |
- &text_matrix, pDeviceMatrix, textobj->m_GraphState, fill_argb, stroke_argb, pClippingPath, flag); |
+ } |
+ if (bFill) { |
+ if (textobj->m_ColorState.GetFillColor()->IsPattern()) { |
+ bPattern = TRUE; |
+ } else { |
+ fill_argb = GetFillArgb(textobj); |
} |
- text_matrix.Concat(*pObj2Device); |
- return CPDF_TextRenderer::DrawNormalText(m_pDevice, textobj->m_nChars, textobj->m_pCharCodes, textobj->m_pCharPos, pFont, font_size, |
- &text_matrix, fill_argb, &m_Options); |
+ } |
+ CFX_AffineMatrix text_matrix; |
+ textobj->GetTextMatrix(&text_matrix); |
+ if (IsAvailableMatrix(text_matrix) == FALSE) { |
+ return TRUE; |
+ } |
+ FX_FLOAT font_size = textobj->m_TextState.GetFontSize(); |
+ if (bPattern) { |
+ DrawTextPathWithPattern(textobj, pObj2Device, pFont, font_size, |
+ &text_matrix, bFill, bStroke); |
+ return TRUE; |
+ } |
+ if (bClip || bStroke) { |
+ const CFX_AffineMatrix* pDeviceMatrix = pObj2Device; |
+ CFX_AffineMatrix device_matrix; |
+ if (bStroke) { |
+ const FX_FLOAT* pCTM = textobj->m_TextState.GetObject()->m_CTM; |
+ if (pCTM[0] != 1.0f || pCTM[3] != 1.0f) { |
+ CFX_AffineMatrix ctm(pCTM[0], pCTM[1], pCTM[2], pCTM[3], 0, 0); |
+ text_matrix.ConcatInverse(ctm); |
+ device_matrix.Copy(ctm); |
+ device_matrix.Concat(*pObj2Device); |
+ pDeviceMatrix = &device_matrix; |
+ } |
+ } |
+ int flag = 0; |
+ if (bStroke && bFill) { |
+ flag |= FX_FILL_STROKE; |
+ flag |= FX_STROKE_TEXT_MODE; |
+ } |
+ const CPDF_GeneralStateData* pGeneralData = |
+ ((CPDF_PageObject*)textobj)->m_GeneralState; |
+ if (pGeneralData && pGeneralData->m_StrokeAdjust) { |
+ flag |= FX_STROKE_ADJUST; |
+ } |
+ if (m_Options.m_Flags & RENDER_NOTEXTSMOOTH) { |
+ flag |= FXFILL_NOPATHSMOOTH; |
+ } |
+ return CPDF_TextRenderer::DrawTextPath( |
+ m_pDevice, textobj->m_nChars, textobj->m_pCharCodes, |
+ textobj->m_pCharPos, pFont, font_size, &text_matrix, pDeviceMatrix, |
+ textobj->m_GraphState, fill_argb, stroke_argb, pClippingPath, flag); |
+ } |
+ text_matrix.Concat(*pObj2Device); |
+ return CPDF_TextRenderer::DrawNormalText( |
+ m_pDevice, textobj->m_nChars, textobj->m_pCharCodes, textobj->m_pCharPos, |
+ pFont, font_size, &text_matrix, fill_argb, &m_Options); |
} |
-CPDF_Type3Cache* CPDF_RenderStatus::GetCachedType3(CPDF_Type3Font* pFont) |
-{ |
- if (pFont->m_pDocument == NULL) { |
- return NULL; |
- } |
- pFont->m_pDocument->GetPageData()->GetFont(pFont->GetFontDict(), FALSE); |
- return pFont->m_pDocument->GetRenderData()->GetCachedType3(pFont); |
+CPDF_Type3Cache* CPDF_RenderStatus::GetCachedType3(CPDF_Type3Font* pFont) { |
+ if (pFont->m_pDocument == NULL) { |
+ return NULL; |
+ } |
+ pFont->m_pDocument->GetPageData()->GetFont(pFont->GetFontDict(), FALSE); |
+ return pFont->m_pDocument->GetRenderData()->GetCachedType3(pFont); |
} |
-static void ReleaseCachedType3(CPDF_Type3Font* pFont) |
-{ |
- if (pFont->m_pDocument == NULL) { |
- return; |
- } |
- pFont->m_pDocument->GetRenderData()->ReleaseCachedType3(pFont); |
- pFont->m_pDocument->GetPageData()->ReleaseFont(pFont->GetFontDict()); |
+static void ReleaseCachedType3(CPDF_Type3Font* pFont) { |
+ if (pFont->m_pDocument == NULL) { |
+ return; |
+ } |
+ pFont->m_pDocument->GetRenderData()->ReleaseCachedType3(pFont); |
+ pFont->m_pDocument->GetPageData()->ReleaseFont(pFont->GetFontDict()); |
} |
-FX_BOOL CPDF_Type3Char::LoadBitmap(CPDF_RenderContext* pContext) |
-{ |
- if (m_pBitmap != NULL || m_pForm == NULL) { |
- return TRUE; |
- } |
- if (m_pForm->CountObjects() == 1 && !m_bColored) { |
- CPDF_PageObject *pPageObj = m_pForm->GetObjectAt(m_pForm->GetFirstObjectPosition()); |
- if (pPageObj->m_Type == PDFPAGE_IMAGE) { |
- CPDF_ImageObject* pImage = (CPDF_ImageObject*)pPageObj; |
- m_ImageMatrix = pImage->m_Matrix; |
- const CFX_DIBSource* pSource = pImage->m_pImage->LoadDIBSource(); |
- if (pSource) { |
- m_pBitmap = pSource->Clone(); |
- delete pSource; |
- } |
- delete m_pForm; |
- m_pForm = NULL; |
- return TRUE; |
- } |
- } |
- return FALSE; |
+FX_BOOL CPDF_Type3Char::LoadBitmap(CPDF_RenderContext* pContext) { |
+ if (m_pBitmap != NULL || m_pForm == NULL) { |
+ return TRUE; |
+ } |
+ if (m_pForm->CountObjects() == 1 && !m_bColored) { |
+ CPDF_PageObject* pPageObj = |
+ m_pForm->GetObjectAt(m_pForm->GetFirstObjectPosition()); |
+ if (pPageObj->m_Type == PDFPAGE_IMAGE) { |
+ CPDF_ImageObject* pImage = (CPDF_ImageObject*)pPageObj; |
+ m_ImageMatrix = pImage->m_Matrix; |
+ const CFX_DIBSource* pSource = pImage->m_pImage->LoadDIBSource(); |
+ if (pSource) { |
+ m_pBitmap = pSource->Clone(); |
+ delete pSource; |
+ } |
+ delete m_pForm; |
+ m_pForm = NULL; |
+ return TRUE; |
+ } |
+ } |
+ return FALSE; |
} |
-class CPDF_RefType3Cache |
-{ |
-public: |
- CPDF_RefType3Cache(CPDF_Type3Font* pType3Font) |
- { |
- m_dwCount = 0; |
- m_pType3Font = pType3Font; |
- } |
- ~CPDF_RefType3Cache() |
- { |
- while(m_dwCount--) { |
- ReleaseCachedType3(m_pType3Font); |
- } |
- } |
- FX_DWORD m_dwCount; |
- CPDF_Type3Font* m_pType3Font; |
+class CPDF_RefType3Cache { |
+ public: |
+ CPDF_RefType3Cache(CPDF_Type3Font* pType3Font) { |
+ m_dwCount = 0; |
+ m_pType3Font = pType3Font; |
+ } |
+ ~CPDF_RefType3Cache() { |
+ while (m_dwCount--) { |
+ ReleaseCachedType3(m_pType3Font); |
+ } |
+ } |
+ FX_DWORD m_dwCount; |
+ CPDF_Type3Font* m_pType3Font; |
}; |
-FX_BOOL CPDF_RenderStatus::ProcessType3Text(const CPDF_TextObject* textobj, const CFX_AffineMatrix* pObj2Device) |
-{ |
- CPDF_Type3Font* pType3Font = textobj->m_TextState.GetFont()->GetType3Font(); |
- for (int j = 0; j < m_Type3FontCache.GetSize(); j++) |
- if ((CPDF_Type3Font*)m_Type3FontCache.GetAt(j) == pType3Font) { |
- return TRUE; |
- } |
- CFX_Matrix dCTM = m_pDevice->GetCTM(); |
- FX_FLOAT sa = FXSYS_fabs(dCTM.a); |
- FX_FLOAT sd = FXSYS_fabs(dCTM.d); |
- CFX_AffineMatrix text_matrix; |
- textobj->GetTextMatrix(&text_matrix); |
- CFX_AffineMatrix char_matrix = pType3Font->GetFontMatrix(); |
- FX_FLOAT font_size = textobj->m_TextState.GetFontSize(); |
- char_matrix.Scale(font_size, font_size); |
- FX_ARGB fill_argb = GetFillArgb(textobj, TRUE); |
- int fill_alpha = FXARGB_A(fill_argb); |
- int device_class = m_pDevice->GetDeviceClass(); |
- FXTEXT_GLYPHPOS* pGlyphAndPos = NULL; |
- if (device_class == FXDC_DISPLAY) { |
- pGlyphAndPos = FX_Alloc(FXTEXT_GLYPHPOS, textobj->m_nChars); |
- } else if (fill_alpha < 255) { |
- return FALSE; |
- } |
- CPDF_RefType3Cache refTypeCache(pType3Font); |
- FX_DWORD *pChars = textobj->m_pCharCodes; |
- if (textobj->m_nChars == 1) { |
- pChars = (FX_DWORD*)(&textobj->m_pCharCodes); |
- } |
- for (int iChar = 0; iChar < textobj->m_nChars; iChar ++) { |
- FX_DWORD charcode = pChars[iChar]; |
- if (charcode == (FX_DWORD) - 1) { |
- continue; |
- } |
- CPDF_Type3Char* pType3Char = pType3Font->LoadChar(charcode); |
- if (pType3Char == NULL) { |
+FX_BOOL CPDF_RenderStatus::ProcessType3Text( |
+ const CPDF_TextObject* textobj, |
+ const CFX_AffineMatrix* pObj2Device) { |
+ CPDF_Type3Font* pType3Font = textobj->m_TextState.GetFont()->GetType3Font(); |
+ for (int j = 0; j < m_Type3FontCache.GetSize(); j++) |
+ if ((CPDF_Type3Font*)m_Type3FontCache.GetAt(j) == pType3Font) { |
+ return TRUE; |
+ } |
+ CFX_Matrix dCTM = m_pDevice->GetCTM(); |
+ FX_FLOAT sa = FXSYS_fabs(dCTM.a); |
+ FX_FLOAT sd = FXSYS_fabs(dCTM.d); |
+ CFX_AffineMatrix text_matrix; |
+ textobj->GetTextMatrix(&text_matrix); |
+ CFX_AffineMatrix char_matrix = pType3Font->GetFontMatrix(); |
+ FX_FLOAT font_size = textobj->m_TextState.GetFontSize(); |
+ char_matrix.Scale(font_size, font_size); |
+ FX_ARGB fill_argb = GetFillArgb(textobj, TRUE); |
+ int fill_alpha = FXARGB_A(fill_argb); |
+ int device_class = m_pDevice->GetDeviceClass(); |
+ FXTEXT_GLYPHPOS* pGlyphAndPos = NULL; |
+ if (device_class == FXDC_DISPLAY) { |
+ pGlyphAndPos = FX_Alloc(FXTEXT_GLYPHPOS, textobj->m_nChars); |
+ } else if (fill_alpha < 255) { |
+ return FALSE; |
+ } |
+ CPDF_RefType3Cache refTypeCache(pType3Font); |
+ FX_DWORD* pChars = textobj->m_pCharCodes; |
+ if (textobj->m_nChars == 1) { |
+ pChars = (FX_DWORD*)(&textobj->m_pCharCodes); |
+ } |
+ for (int iChar = 0; iChar < textobj->m_nChars; iChar++) { |
+ FX_DWORD charcode = pChars[iChar]; |
+ if (charcode == (FX_DWORD)-1) { |
+ continue; |
+ } |
+ CPDF_Type3Char* pType3Char = pType3Font->LoadChar(charcode); |
+ if (pType3Char == NULL) { |
+ continue; |
+ } |
+ CFX_AffineMatrix matrix = char_matrix; |
+ matrix.e += iChar ? textobj->m_pCharPos[iChar - 1] : 0; |
+ matrix.Concat(text_matrix); |
+ matrix.Concat(*pObj2Device); |
+ if (!pType3Char->LoadBitmap(m_pContext)) { |
+ if (pGlyphAndPos) { |
+ for (int i = 0; i < iChar; i++) { |
+ FXTEXT_GLYPHPOS& glyph = pGlyphAndPos[i]; |
+ if (glyph.m_pGlyph == NULL) { |
continue; |
+ } |
+ m_pDevice->SetBitMask(&glyph.m_pGlyph->m_Bitmap, |
+ glyph.m_OriginX + glyph.m_pGlyph->m_Left, |
+ glyph.m_OriginY - glyph.m_pGlyph->m_Top, |
+ fill_argb); |
} |
- CFX_AffineMatrix matrix = char_matrix; |
- matrix.e += iChar ? textobj->m_pCharPos[iChar - 1] : 0; |
- matrix.Concat(text_matrix); |
- matrix.Concat(*pObj2Device); |
- if (!pType3Char->LoadBitmap(m_pContext)) { |
- if (pGlyphAndPos) { |
- for (int i = 0; i < iChar; i ++) { |
- FXTEXT_GLYPHPOS& glyph = pGlyphAndPos[i]; |
- if (glyph.m_pGlyph == NULL) { |
- continue; |
- } |
- m_pDevice->SetBitMask(&glyph.m_pGlyph->m_Bitmap, |
- glyph.m_OriginX + glyph.m_pGlyph->m_Left, |
- glyph.m_OriginY - glyph.m_pGlyph->m_Top, fill_argb); |
- } |
- FX_Free(pGlyphAndPos); |
- pGlyphAndPos = NULL; |
- } |
- CPDF_GraphicStates* pStates = CloneObjStates(textobj, FALSE); |
- CPDF_RenderOptions Options = m_Options; |
- Options.m_Flags |= RENDER_FORCE_HALFTONE | RENDER_RECT_AA; |
- Options.m_Flags &= ~RENDER_FORCE_DOWNSAMPLE; |
- CPDF_Dictionary* pFormResource = NULL; |
- if (pType3Char->m_pForm && pType3Char->m_pForm->m_pFormDict) { |
- pFormResource = pType3Char->m_pForm->m_pFormDict->GetDict(FX_BSTRC("Resources")); |
- } |
- if (fill_alpha == 255) { |
- CPDF_RenderStatus status; |
- status.Initialize(m_pContext, m_pDevice, NULL, NULL, this, pStates, &Options, |
- pType3Char->m_pForm->m_Transparency, m_bDropObjects, pFormResource, FALSE, pType3Char, fill_argb); |
- status.m_Type3FontCache.Append(m_Type3FontCache); |
- status.m_Type3FontCache.Add(pType3Font); |
- m_pDevice->SaveState(); |
- status.RenderObjectList(pType3Char->m_pForm, &matrix); |
- m_pDevice->RestoreState(); |
- } else { |
- CFX_FloatRect rect_f = pType3Char->m_pForm->CalcBoundingBox(); |
- rect_f.Transform(&matrix); |
- FX_RECT rect = rect_f.GetOutterRect(); |
- CFX_FxgeDevice bitmap_device; |
- if (!bitmap_device.Create((int)(rect.Width() * sa), (int)(rect.Height() * sd), FXDIB_Argb)) { |
- return TRUE; |
- } |
- bitmap_device.GetBitmap()->Clear(0); |
- CPDF_RenderStatus status; |
- status.Initialize(m_pContext, &bitmap_device, NULL, NULL, this, pStates, &Options, |
- pType3Char->m_pForm->m_Transparency, m_bDropObjects, pFormResource, FALSE, pType3Char, fill_argb); |
- status.m_Type3FontCache.Append(m_Type3FontCache); |
- status.m_Type3FontCache.Add(pType3Font); |
- matrix.TranslateI(-rect.left, -rect.top); |
- matrix.Scale(sa, sd); |
- status.RenderObjectList(pType3Char->m_pForm, &matrix); |
- m_pDevice->SetDIBits(bitmap_device.GetBitmap(), rect.left, rect.top); |
- } |
- delete pStates; |
- } else if (pType3Char->m_pBitmap) { |
- if (device_class == FXDC_DISPLAY) { |
- CPDF_Type3Cache* pCache = GetCachedType3(pType3Font); |
- refTypeCache.m_dwCount++; |
- CFX_GlyphBitmap* pBitmap = pCache->LoadGlyph(charcode, &matrix, sa, sd); |
- if (pBitmap == NULL) { |
- continue; |
- } |
- int origin_x = FXSYS_round(matrix.e); |
- int origin_y = FXSYS_round(matrix.f); |
- if (pGlyphAndPos) { |
- pGlyphAndPos[iChar].m_pGlyph = pBitmap; |
- pGlyphAndPos[iChar].m_OriginX = origin_x; |
- pGlyphAndPos[iChar].m_OriginY = origin_y; |
- } else { |
- m_pDevice->SetBitMask(&pBitmap->m_Bitmap, origin_x + pBitmap->m_Left, origin_y - pBitmap->m_Top, fill_argb); |
- } |
- } else { |
- CFX_AffineMatrix image_matrix = pType3Char->m_ImageMatrix; |
- image_matrix.Concat(matrix); |
- CPDF_ImageRenderer renderer; |
- if (renderer.Start(this, pType3Char->m_pBitmap, fill_argb, 255, &image_matrix, 0, FALSE)) { |
- renderer.Continue(NULL); |
- } |
- if (!renderer.m_Result) { |
- return FALSE; |
- } |
- } |
- } |
- } |
- if (pGlyphAndPos) { |
- FX_RECT rect = FXGE_GetGlyphsBBox(pGlyphAndPos, textobj->m_nChars, 0, sa, sd); |
- CFX_DIBitmap bitmap; |
- if (!bitmap.Create((int)(rect.Width() * sa), (int)(rect.Height() * sd), FXDIB_8bppMask)) { |
- FX_Free(pGlyphAndPos); |
- return TRUE; |
- } |
- bitmap.Clear(0); |
- for (int iChar = 0; iChar < textobj->m_nChars; iChar ++) { |
- FXTEXT_GLYPHPOS& glyph = pGlyphAndPos[iChar]; |
- if (glyph.m_pGlyph == NULL) { |
- continue; |
- } |
- bitmap.TransferBitmap((int)((glyph.m_OriginX + glyph.m_pGlyph->m_Left - rect.left) * sa), |
- (int)((glyph.m_OriginY - glyph.m_pGlyph->m_Top - rect.top) * sd), |
- glyph.m_pGlyph->m_Bitmap.GetWidth(), glyph.m_pGlyph->m_Bitmap.GetHeight(), |
- &glyph.m_pGlyph->m_Bitmap, 0, 0); |
- } |
- m_pDevice->SetBitMask(&bitmap, rect.left, rect.top, fill_argb); |
FX_Free(pGlyphAndPos); |
- } |
- return TRUE; |
+ pGlyphAndPos = NULL; |
+ } |
+ CPDF_GraphicStates* pStates = CloneObjStates(textobj, FALSE); |
+ CPDF_RenderOptions Options = m_Options; |
+ Options.m_Flags |= RENDER_FORCE_HALFTONE | RENDER_RECT_AA; |
+ Options.m_Flags &= ~RENDER_FORCE_DOWNSAMPLE; |
+ CPDF_Dictionary* pFormResource = NULL; |
+ if (pType3Char->m_pForm && pType3Char->m_pForm->m_pFormDict) { |
+ pFormResource = |
+ pType3Char->m_pForm->m_pFormDict->GetDict(FX_BSTRC("Resources")); |
+ } |
+ if (fill_alpha == 255) { |
+ CPDF_RenderStatus status; |
+ status.Initialize(m_pContext, m_pDevice, NULL, NULL, this, pStates, |
+ &Options, pType3Char->m_pForm->m_Transparency, |
+ m_bDropObjects, pFormResource, FALSE, pType3Char, |
+ fill_argb); |
+ status.m_Type3FontCache.Append(m_Type3FontCache); |
+ status.m_Type3FontCache.Add(pType3Font); |
+ m_pDevice->SaveState(); |
+ status.RenderObjectList(pType3Char->m_pForm, &matrix); |
+ m_pDevice->RestoreState(); |
+ } else { |
+ CFX_FloatRect rect_f = pType3Char->m_pForm->CalcBoundingBox(); |
+ rect_f.Transform(&matrix); |
+ FX_RECT rect = rect_f.GetOutterRect(); |
+ CFX_FxgeDevice bitmap_device; |
+ if (!bitmap_device.Create((int)(rect.Width() * sa), |
+ (int)(rect.Height() * sd), FXDIB_Argb)) { |
+ return TRUE; |
+ } |
+ bitmap_device.GetBitmap()->Clear(0); |
+ CPDF_RenderStatus status; |
+ status.Initialize(m_pContext, &bitmap_device, NULL, NULL, this, pStates, |
+ &Options, pType3Char->m_pForm->m_Transparency, |
+ m_bDropObjects, pFormResource, FALSE, pType3Char, |
+ fill_argb); |
+ status.m_Type3FontCache.Append(m_Type3FontCache); |
+ status.m_Type3FontCache.Add(pType3Font); |
+ matrix.TranslateI(-rect.left, -rect.top); |
+ matrix.Scale(sa, sd); |
+ status.RenderObjectList(pType3Char->m_pForm, &matrix); |
+ m_pDevice->SetDIBits(bitmap_device.GetBitmap(), rect.left, rect.top); |
+ } |
+ delete pStates; |
+ } else if (pType3Char->m_pBitmap) { |
+ if (device_class == FXDC_DISPLAY) { |
+ CPDF_Type3Cache* pCache = GetCachedType3(pType3Font); |
+ refTypeCache.m_dwCount++; |
+ CFX_GlyphBitmap* pBitmap = pCache->LoadGlyph(charcode, &matrix, sa, sd); |
+ if (pBitmap == NULL) { |
+ continue; |
+ } |
+ int origin_x = FXSYS_round(matrix.e); |
+ int origin_y = FXSYS_round(matrix.f); |
+ if (pGlyphAndPos) { |
+ pGlyphAndPos[iChar].m_pGlyph = pBitmap; |
+ pGlyphAndPos[iChar].m_OriginX = origin_x; |
+ pGlyphAndPos[iChar].m_OriginY = origin_y; |
+ } else { |
+ m_pDevice->SetBitMask(&pBitmap->m_Bitmap, origin_x + pBitmap->m_Left, |
+ origin_y - pBitmap->m_Top, fill_argb); |
+ } |
+ } else { |
+ CFX_AffineMatrix image_matrix = pType3Char->m_ImageMatrix; |
+ image_matrix.Concat(matrix); |
+ CPDF_ImageRenderer renderer; |
+ if (renderer.Start(this, pType3Char->m_pBitmap, fill_argb, 255, |
+ &image_matrix, 0, FALSE)) { |
+ renderer.Continue(NULL); |
+ } |
+ if (!renderer.m_Result) { |
+ return FALSE; |
+ } |
+ } |
+ } |
+ } |
+ if (pGlyphAndPos) { |
+ FX_RECT rect = |
+ FXGE_GetGlyphsBBox(pGlyphAndPos, textobj->m_nChars, 0, sa, sd); |
+ CFX_DIBitmap bitmap; |
+ if (!bitmap.Create((int)(rect.Width() * sa), (int)(rect.Height() * sd), |
+ FXDIB_8bppMask)) { |
+ FX_Free(pGlyphAndPos); |
+ return TRUE; |
+ } |
+ bitmap.Clear(0); |
+ for (int iChar = 0; iChar < textobj->m_nChars; iChar++) { |
+ FXTEXT_GLYPHPOS& glyph = pGlyphAndPos[iChar]; |
+ if (glyph.m_pGlyph == NULL) { |
+ continue; |
+ } |
+ bitmap.TransferBitmap( |
+ (int)((glyph.m_OriginX + glyph.m_pGlyph->m_Left - rect.left) * sa), |
+ (int)((glyph.m_OriginY - glyph.m_pGlyph->m_Top - rect.top) * sd), |
+ glyph.m_pGlyph->m_Bitmap.GetWidth(), |
+ glyph.m_pGlyph->m_Bitmap.GetHeight(), &glyph.m_pGlyph->m_Bitmap, 0, |
+ 0); |
+ } |
+ m_pDevice->SetBitMask(&bitmap, rect.left, rect.top, fill_argb); |
+ FX_Free(pGlyphAndPos); |
+ } |
+ return TRUE; |
} |
-class CPDF_CharPosList |
-{ |
-public: |
- CPDF_CharPosList(); |
- ~CPDF_CharPosList(); |
- void Load(int nChars, FX_DWORD* pCharCodes, FX_FLOAT* pCharPos, CPDF_Font* pFont, FX_FLOAT font_size); |
- FXTEXT_CHARPOS* m_pCharPos; |
- FX_DWORD m_nChars; |
+class CPDF_CharPosList { |
+ public: |
+ CPDF_CharPosList(); |
+ ~CPDF_CharPosList(); |
+ void Load(int nChars, |
+ FX_DWORD* pCharCodes, |
+ FX_FLOAT* pCharPos, |
+ CPDF_Font* pFont, |
+ FX_FLOAT font_size); |
+ FXTEXT_CHARPOS* m_pCharPos; |
+ FX_DWORD m_nChars; |
}; |
FX_FLOAT _CIDTransformToFloat(uint8_t ch); |
-CPDF_CharPosList::CPDF_CharPosList() |
-{ |
- m_pCharPos = NULL; |
+CPDF_CharPosList::CPDF_CharPosList() { |
+ m_pCharPos = NULL; |
} |
-CPDF_CharPosList::~CPDF_CharPosList() |
-{ |
- if (m_pCharPos) { |
- FX_Free(m_pCharPos); |
- } |
+CPDF_CharPosList::~CPDF_CharPosList() { |
+ if (m_pCharPos) { |
+ FX_Free(m_pCharPos); |
+ } |
} |
-void CPDF_CharPosList::Load(int nChars, FX_DWORD* pCharCodes, FX_FLOAT* pCharPos, CPDF_Font* pFont, |
- FX_FLOAT FontSize) |
-{ |
- m_pCharPos = FX_Alloc(FXTEXT_CHARPOS, nChars); |
- m_nChars = 0; |
- CPDF_CIDFont* pCIDFont = pFont->GetCIDFont(); |
- FX_BOOL bVertWriting = pCIDFont && pCIDFont->IsVertWriting(); |
- for (int iChar = 0; iChar < nChars; iChar ++) { |
- FX_DWORD CharCode = nChars == 1 ? (FX_DWORD)(uintptr_t)pCharCodes : pCharCodes[iChar]; |
- if (CharCode == (FX_DWORD) - 1) { |
- continue; |
- } |
- FX_BOOL bVert = FALSE; |
- FXTEXT_CHARPOS& charpos = m_pCharPos[m_nChars++]; |
- if (pCIDFont) { |
- charpos.m_bFontStyle = pCIDFont->IsFontStyleFromCharCode(CharCode); |
- } |
- charpos.m_GlyphIndex = pFont->GlyphFromCharCode(CharCode, &bVert); |
-#if _FXM_PLATFORM_ == _FXM_PLATFORM_APPLE_ |
- charpos.m_ExtGID = pFont->GlyphFromCharCodeExt(CharCode); |
+void CPDF_CharPosList::Load(int nChars, |
+ FX_DWORD* pCharCodes, |
+ FX_FLOAT* pCharPos, |
+ CPDF_Font* pFont, |
+ FX_FLOAT FontSize) { |
+ m_pCharPos = FX_Alloc(FXTEXT_CHARPOS, nChars); |
+ m_nChars = 0; |
+ CPDF_CIDFont* pCIDFont = pFont->GetCIDFont(); |
+ FX_BOOL bVertWriting = pCIDFont && pCIDFont->IsVertWriting(); |
+ for (int iChar = 0; iChar < nChars; iChar++) { |
+ FX_DWORD CharCode = |
+ nChars == 1 ? (FX_DWORD)(uintptr_t)pCharCodes : pCharCodes[iChar]; |
+ if (CharCode == (FX_DWORD)-1) { |
+ continue; |
+ } |
+ FX_BOOL bVert = FALSE; |
+ FXTEXT_CHARPOS& charpos = m_pCharPos[m_nChars++]; |
+ if (pCIDFont) { |
+ charpos.m_bFontStyle = pCIDFont->IsFontStyleFromCharCode(CharCode); |
+ } |
+ charpos.m_GlyphIndex = pFont->GlyphFromCharCode(CharCode, &bVert); |
+#if _FXM_PLATFORM_ == _FXM_PLATFORM_APPLE_ |
+ charpos.m_ExtGID = pFont->GlyphFromCharCodeExt(CharCode); |
#endif |
- if (!pFont->IsEmbedded() && pFont->GetFontType() != PDFFONT_CIDFONT) { |
- charpos.m_FontCharWidth = pFont->GetCharWidthF(CharCode); |
- } else { |
- charpos.m_FontCharWidth = 0; |
- } |
- charpos.m_OriginX = iChar ? pCharPos[iChar - 1] : 0; |
- charpos.m_OriginY = 0; |
- charpos.m_bGlyphAdjust = FALSE; |
- if (pCIDFont == NULL) { |
- continue; |
- } |
- FX_WORD CID = pCIDFont->CIDFromCharCode(CharCode); |
- if (bVertWriting) { |
- charpos.m_OriginY = charpos.m_OriginX; |
- charpos.m_OriginX = 0; |
- short vx, vy; |
- pCIDFont->GetVertOrigin(CID, vx, vy); |
- charpos.m_OriginX -= FontSize * vx / 1000; |
- charpos.m_OriginY -= FontSize * vy / 1000; |
- } |
- const uint8_t* pTransform = pCIDFont->GetCIDTransform(CID); |
- if (pTransform && !bVert) { |
- charpos.m_AdjustMatrix[0] = _CIDTransformToFloat(pTransform[0]); |
- charpos.m_AdjustMatrix[2] = _CIDTransformToFloat(pTransform[2]); |
- charpos.m_AdjustMatrix[1] = _CIDTransformToFloat(pTransform[1]); |
- charpos.m_AdjustMatrix[3] = _CIDTransformToFloat(pTransform[3]); |
- charpos.m_OriginX += _CIDTransformToFloat(pTransform[4]) * FontSize; |
- charpos.m_OriginY += _CIDTransformToFloat(pTransform[5]) * FontSize; |
- charpos.m_bGlyphAdjust = TRUE; |
- } |
- } |
+ if (!pFont->IsEmbedded() && pFont->GetFontType() != PDFFONT_CIDFONT) { |
+ charpos.m_FontCharWidth = pFont->GetCharWidthF(CharCode); |
+ } else { |
+ charpos.m_FontCharWidth = 0; |
+ } |
+ charpos.m_OriginX = iChar ? pCharPos[iChar - 1] : 0; |
+ charpos.m_OriginY = 0; |
+ charpos.m_bGlyphAdjust = FALSE; |
+ if (pCIDFont == NULL) { |
+ continue; |
+ } |
+ FX_WORD CID = pCIDFont->CIDFromCharCode(CharCode); |
+ if (bVertWriting) { |
+ charpos.m_OriginY = charpos.m_OriginX; |
+ charpos.m_OriginX = 0; |
+ short vx, vy; |
+ pCIDFont->GetVertOrigin(CID, vx, vy); |
+ charpos.m_OriginX -= FontSize * vx / 1000; |
+ charpos.m_OriginY -= FontSize * vy / 1000; |
+ } |
+ const uint8_t* pTransform = pCIDFont->GetCIDTransform(CID); |
+ if (pTransform && !bVert) { |
+ charpos.m_AdjustMatrix[0] = _CIDTransformToFloat(pTransform[0]); |
+ charpos.m_AdjustMatrix[2] = _CIDTransformToFloat(pTransform[2]); |
+ charpos.m_AdjustMatrix[1] = _CIDTransformToFloat(pTransform[1]); |
+ charpos.m_AdjustMatrix[3] = _CIDTransformToFloat(pTransform[3]); |
+ charpos.m_OriginX += _CIDTransformToFloat(pTransform[4]) * FontSize; |
+ charpos.m_OriginY += _CIDTransformToFloat(pTransform[5]) * FontSize; |
+ charpos.m_bGlyphAdjust = TRUE; |
+ } |
+ } |
} |
-FX_BOOL CPDF_TextRenderer::DrawTextPath(CFX_RenderDevice* pDevice, int nChars, FX_DWORD* pCharCodes, FX_FLOAT* pCharPos, |
- CPDF_Font* pFont, FX_FLOAT font_size, |
- const CFX_AffineMatrix* pText2User, const CFX_AffineMatrix* pUser2Device, |
+FX_BOOL CPDF_TextRenderer::DrawTextPath(CFX_RenderDevice* pDevice, |
+ int nChars, |
+ FX_DWORD* pCharCodes, |
+ FX_FLOAT* pCharPos, |
+ CPDF_Font* pFont, |
+ FX_FLOAT font_size, |
+ const CFX_AffineMatrix* pText2User, |
+ const CFX_AffineMatrix* pUser2Device, |
const CFX_GraphStateData* pGraphState, |
- FX_ARGB fill_argb, FX_ARGB stroke_argb, CFX_PathData* pClippingPath, int nFlag) |
-{ |
- CFX_FontCache* pCache = pFont->m_pDocument ? pFont->m_pDocument->GetRenderData()->GetFontCache() : NULL; |
- CPDF_CharPosList CharPosList; |
- CharPosList.Load(nChars, pCharCodes, pCharPos, pFont, font_size); |
- return pDevice->DrawTextPath(CharPosList.m_nChars, CharPosList.m_pCharPos, |
- &pFont->m_Font, pCache, font_size, pText2User, pUser2Device, |
- pGraphState, fill_argb, stroke_argb, pClippingPath, nFlag); |
+ FX_ARGB fill_argb, |
+ FX_ARGB stroke_argb, |
+ CFX_PathData* pClippingPath, |
+ int nFlag) { |
+ CFX_FontCache* pCache = |
+ pFont->m_pDocument ? pFont->m_pDocument->GetRenderData()->GetFontCache() |
+ : NULL; |
+ CPDF_CharPosList CharPosList; |
+ CharPosList.Load(nChars, pCharCodes, pCharPos, pFont, font_size); |
+ return pDevice->DrawTextPath(CharPosList.m_nChars, CharPosList.m_pCharPos, |
+ &pFont->m_Font, pCache, font_size, pText2User, |
+ pUser2Device, pGraphState, fill_argb, |
+ stroke_argb, pClippingPath, nFlag); |
} |
-void CPDF_TextRenderer::DrawTextString(CFX_RenderDevice* pDevice, int left, int top, CPDF_Font* pFont, int height, |
- const CFX_ByteString& str, FX_ARGB argb) |
-{ |
- FX_RECT font_bbox; |
- pFont->GetFontBBox(font_bbox); |
- FX_FLOAT font_size = (FX_FLOAT)height * 1000.0f / (FX_FLOAT)(font_bbox.top - font_bbox.bottom); |
- FX_FLOAT origin_x = (FX_FLOAT)left; |
- FX_FLOAT origin_y = (FX_FLOAT)top + font_size * (FX_FLOAT)font_bbox.top / 1000.0f; |
- CFX_AffineMatrix matrix(1.0f, 0, 0, -1.0f, 0, 0); |
- DrawTextString(pDevice, origin_x, origin_y, pFont, font_size, &matrix, str, argb); |
+void CPDF_TextRenderer::DrawTextString(CFX_RenderDevice* pDevice, |
+ int left, |
+ int top, |
+ CPDF_Font* pFont, |
+ int height, |
+ const CFX_ByteString& str, |
+ FX_ARGB argb) { |
+ FX_RECT font_bbox; |
+ pFont->GetFontBBox(font_bbox); |
+ FX_FLOAT font_size = |
+ (FX_FLOAT)height * 1000.0f / (FX_FLOAT)(font_bbox.top - font_bbox.bottom); |
+ FX_FLOAT origin_x = (FX_FLOAT)left; |
+ FX_FLOAT origin_y = |
+ (FX_FLOAT)top + font_size * (FX_FLOAT)font_bbox.top / 1000.0f; |
+ CFX_AffineMatrix matrix(1.0f, 0, 0, -1.0f, 0, 0); |
+ DrawTextString(pDevice, origin_x, origin_y, pFont, font_size, &matrix, str, |
+ argb); |
} |
-void CPDF_TextRenderer::DrawTextString(CFX_RenderDevice* pDevice, FX_FLOAT origin_x, FX_FLOAT origin_y, CPDF_Font* pFont, FX_FLOAT font_size, |
- const CFX_AffineMatrix* pMatrix, const CFX_ByteString& str, FX_ARGB fill_argb, |
- FX_ARGB stroke_argb, const CFX_GraphStateData* pGraphState, const CPDF_RenderOptions* pOptions) |
-{ |
- int nChars = pFont->CountChar(str, str.GetLength()); |
- if (nChars == 0) { |
- return; |
- } |
- FX_DWORD charcode; |
- int offset = 0; |
- FX_DWORD* pCharCodes; |
- FX_FLOAT* pCharPos; |
- if (nChars == 1) { |
- charcode = pFont->GetNextChar(str, str.GetLength(), offset); |
- pCharCodes = (FX_DWORD*)(uintptr_t)charcode; |
- pCharPos = NULL; |
- } else { |
- pCharCodes = FX_Alloc(FX_DWORD, nChars); |
- pCharPos = FX_Alloc(FX_FLOAT, nChars - 1); |
- FX_FLOAT cur_pos = 0; |
- for (int i = 0; i < nChars; i ++) { |
- pCharCodes[i] = pFont->GetNextChar(str, str.GetLength(), offset); |
- if (i) { |
- pCharPos[i - 1] = cur_pos; |
- } |
- cur_pos += pFont->GetCharWidthF(pCharCodes[i]) * font_size / 1000; |
- } |
- } |
- CFX_AffineMatrix matrix; |
- if (pMatrix) { |
- matrix = *pMatrix; |
- } |
- matrix.e = origin_x; |
- matrix.f = origin_y; |
- if (pFont->GetFontType() == PDFFONT_TYPE3) |
- ; |
- else if (stroke_argb == 0) { |
- DrawNormalText(pDevice, nChars, pCharCodes, pCharPos, pFont, font_size, &matrix, fill_argb, pOptions); |
- } else |
- DrawTextPath(pDevice, nChars, pCharCodes, pCharPos, pFont, font_size, &matrix, NULL, pGraphState, |
- fill_argb, stroke_argb, NULL); |
- if (nChars > 1) { |
- FX_Free(pCharCodes); |
- FX_Free(pCharPos); |
- } |
+void CPDF_TextRenderer::DrawTextString(CFX_RenderDevice* pDevice, |
+ FX_FLOAT origin_x, |
+ FX_FLOAT origin_y, |
+ CPDF_Font* pFont, |
+ FX_FLOAT font_size, |
+ const CFX_AffineMatrix* pMatrix, |
+ const CFX_ByteString& str, |
+ FX_ARGB fill_argb, |
+ FX_ARGB stroke_argb, |
+ const CFX_GraphStateData* pGraphState, |
+ const CPDF_RenderOptions* pOptions) { |
+ int nChars = pFont->CountChar(str, str.GetLength()); |
+ if (nChars == 0) { |
+ return; |
+ } |
+ FX_DWORD charcode; |
+ int offset = 0; |
+ FX_DWORD* pCharCodes; |
+ FX_FLOAT* pCharPos; |
+ if (nChars == 1) { |
+ charcode = pFont->GetNextChar(str, str.GetLength(), offset); |
+ pCharCodes = (FX_DWORD*)(uintptr_t)charcode; |
+ pCharPos = NULL; |
+ } else { |
+ pCharCodes = FX_Alloc(FX_DWORD, nChars); |
+ pCharPos = FX_Alloc(FX_FLOAT, nChars - 1); |
+ FX_FLOAT cur_pos = 0; |
+ for (int i = 0; i < nChars; i++) { |
+ pCharCodes[i] = pFont->GetNextChar(str, str.GetLength(), offset); |
+ if (i) { |
+ pCharPos[i - 1] = cur_pos; |
+ } |
+ cur_pos += pFont->GetCharWidthF(pCharCodes[i]) * font_size / 1000; |
+ } |
+ } |
+ CFX_AffineMatrix matrix; |
+ if (pMatrix) { |
+ matrix = *pMatrix; |
+ } |
+ matrix.e = origin_x; |
+ matrix.f = origin_y; |
+ if (pFont->GetFontType() == PDFFONT_TYPE3) |
+ ; |
+ else if (stroke_argb == 0) { |
+ DrawNormalText(pDevice, nChars, pCharCodes, pCharPos, pFont, font_size, |
+ &matrix, fill_argb, pOptions); |
+ } else |
+ DrawTextPath(pDevice, nChars, pCharCodes, pCharPos, pFont, font_size, |
+ &matrix, NULL, pGraphState, fill_argb, stroke_argb, NULL); |
+ if (nChars > 1) { |
+ FX_Free(pCharCodes); |
+ FX_Free(pCharPos); |
+ } |
} |
-FX_BOOL CPDF_TextRenderer::DrawNormalText(CFX_RenderDevice* pDevice, int nChars, FX_DWORD* pCharCodes, FX_FLOAT* pCharPos, |
- CPDF_Font* pFont, FX_FLOAT font_size, |
- const CFX_AffineMatrix* pText2Device, |
- FX_ARGB fill_argb, const CPDF_RenderOptions* pOptions) |
-{ |
- CFX_FontCache* pCache = pFont->m_pDocument ? pFont->m_pDocument->GetRenderData()->GetFontCache() : NULL; |
- CPDF_CharPosList CharPosList; |
- CharPosList.Load(nChars, pCharCodes, pCharPos, pFont, font_size); |
- int FXGE_flags = 0; |
- if (pOptions) { |
- FX_DWORD dwFlags = pOptions->m_Flags; |
- if (dwFlags & RENDER_CLEARTYPE) { |
- FXGE_flags |= FXTEXT_CLEARTYPE; |
- if (dwFlags & RENDER_BGR_STRIPE) { |
- FXGE_flags |= FXTEXT_BGR_STRIPE; |
- } |
- } |
- if (dwFlags & RENDER_NOTEXTSMOOTH) { |
- FXGE_flags |= FXTEXT_NOSMOOTH; |
- } |
- if (dwFlags & RENDER_PRINTGRAPHICTEXT) { |
- FXGE_flags |= FXTEXT_PRINTGRAPHICTEXT; |
- } |
- if (dwFlags & RENDER_NO_NATIVETEXT) { |
- FXGE_flags |= FXTEXT_NO_NATIVETEXT; |
- } |
- if (dwFlags & RENDER_PRINTIMAGETEXT) { |
- FXGE_flags |= FXTEXT_PRINTIMAGETEXT; |
- } |
- } else { |
- FXGE_flags = FXTEXT_CLEARTYPE; |
- } |
- if (pFont->GetFontType() & PDFFONT_CIDFONT) { |
- FXGE_flags |= FXFONT_CIDFONT; |
- } |
- return pDevice->DrawNormalText(CharPosList.m_nChars, CharPosList.m_pCharPos, &pFont->m_Font, pCache, font_size, pText2Device, fill_argb, FXGE_flags); |
+FX_BOOL CPDF_TextRenderer::DrawNormalText(CFX_RenderDevice* pDevice, |
+ int nChars, |
+ FX_DWORD* pCharCodes, |
+ FX_FLOAT* pCharPos, |
+ CPDF_Font* pFont, |
+ FX_FLOAT font_size, |
+ const CFX_AffineMatrix* pText2Device, |
+ FX_ARGB fill_argb, |
+ const CPDF_RenderOptions* pOptions) { |
+ CFX_FontCache* pCache = |
+ pFont->m_pDocument ? pFont->m_pDocument->GetRenderData()->GetFontCache() |
+ : NULL; |
+ CPDF_CharPosList CharPosList; |
+ CharPosList.Load(nChars, pCharCodes, pCharPos, pFont, font_size); |
+ int FXGE_flags = 0; |
+ if (pOptions) { |
+ FX_DWORD dwFlags = pOptions->m_Flags; |
+ if (dwFlags & RENDER_CLEARTYPE) { |
+ FXGE_flags |= FXTEXT_CLEARTYPE; |
+ if (dwFlags & RENDER_BGR_STRIPE) { |
+ FXGE_flags |= FXTEXT_BGR_STRIPE; |
+ } |
+ } |
+ if (dwFlags & RENDER_NOTEXTSMOOTH) { |
+ FXGE_flags |= FXTEXT_NOSMOOTH; |
+ } |
+ if (dwFlags & RENDER_PRINTGRAPHICTEXT) { |
+ FXGE_flags |= FXTEXT_PRINTGRAPHICTEXT; |
+ } |
+ if (dwFlags & RENDER_NO_NATIVETEXT) { |
+ FXGE_flags |= FXTEXT_NO_NATIVETEXT; |
+ } |
+ if (dwFlags & RENDER_PRINTIMAGETEXT) { |
+ FXGE_flags |= FXTEXT_PRINTIMAGETEXT; |
+ } |
+ } else { |
+ FXGE_flags = FXTEXT_CLEARTYPE; |
+ } |
+ if (pFont->GetFontType() & PDFFONT_CIDFONT) { |
+ FXGE_flags |= FXFONT_CIDFONT; |
+ } |
+ return pDevice->DrawNormalText(CharPosList.m_nChars, CharPosList.m_pCharPos, |
+ &pFont->m_Font, pCache, font_size, |
+ pText2Device, fill_argb, FXGE_flags); |
} |
-void CPDF_RenderStatus::DrawTextPathWithPattern(const CPDF_TextObject* textobj, const CFX_AffineMatrix* pObj2Device, |
- CPDF_Font* pFont, FX_FLOAT font_size, |
- const CFX_AffineMatrix* pTextMatrix, FX_BOOL bFill, FX_BOOL bStroke) |
-{ |
- if (!bStroke) { |
- CPDF_PathObject path; |
- CPDF_TextObject* pCopy = new CPDF_TextObject; |
- pCopy->Copy(textobj); |
- path.m_bStroke = FALSE; |
- path.m_FillType = FXFILL_WINDING; |
- path.m_ClipPath.AppendTexts(&pCopy, 1); |
- path.m_ColorState = textobj->m_ColorState; |
- path.m_Path.New()->AppendRect(textobj->m_Left, textobj->m_Bottom, textobj->m_Right, textobj->m_Top); |
- path.m_Left = textobj->m_Left; |
- path.m_Bottom = textobj->m_Bottom; |
- path.m_Right = textobj->m_Right; |
- path.m_Top = textobj->m_Top; |
- RenderSingleObject(&path, pObj2Device); |
- return; |
- } |
- CFX_FontCache* pCache; |
- if (pFont->m_pDocument) { |
- pCache = pFont->m_pDocument->GetRenderData()->GetFontCache(); |
- } else { |
- pCache = CFX_GEModule::Get()->GetFontCache(); |
- } |
- CFX_FaceCache* pFaceCache = pCache->GetCachedFace(&pFont->m_Font); |
- FX_FONTCACHE_DEFINE(pCache, &pFont->m_Font); |
- CPDF_CharPosList CharPosList; |
- CharPosList.Load(textobj->m_nChars, textobj->m_pCharCodes, textobj->m_pCharPos, pFont, font_size); |
- for (FX_DWORD i = 0; i < CharPosList.m_nChars; i ++) { |
- FXTEXT_CHARPOS& charpos = CharPosList.m_pCharPos[i]; |
- const CFX_PathData* pPath = pFaceCache->LoadGlyphPath(&pFont->m_Font, charpos.m_GlyphIndex, |
- charpos.m_FontCharWidth); |
- if (pPath == NULL) { |
- continue; |
- } |
- CPDF_PathObject path; |
- path.m_GraphState = textobj->m_GraphState; |
- path.m_ColorState = textobj->m_ColorState; |
- CFX_AffineMatrix matrix; |
- if (charpos.m_bGlyphAdjust) |
- matrix.Set(charpos.m_AdjustMatrix[0], charpos.m_AdjustMatrix[1], |
- charpos.m_AdjustMatrix[2], charpos.m_AdjustMatrix[3], 0, 0); |
- matrix.Concat(font_size, 0, 0, font_size, charpos.m_OriginX, charpos.m_OriginY); |
- path.m_Path.New()->Append(pPath, &matrix); |
- path.m_Matrix = *pTextMatrix; |
- path.m_bStroke = bStroke; |
- path.m_FillType = bFill ? FXFILL_WINDING : 0; |
- path.CalcBoundingBox(); |
- ProcessPath(&path, pObj2Device); |
- } |
+void CPDF_RenderStatus::DrawTextPathWithPattern( |
+ const CPDF_TextObject* textobj, |
+ const CFX_AffineMatrix* pObj2Device, |
+ CPDF_Font* pFont, |
+ FX_FLOAT font_size, |
+ const CFX_AffineMatrix* pTextMatrix, |
+ FX_BOOL bFill, |
+ FX_BOOL bStroke) { |
+ if (!bStroke) { |
+ CPDF_PathObject path; |
+ CPDF_TextObject* pCopy = new CPDF_TextObject; |
+ pCopy->Copy(textobj); |
+ path.m_bStroke = FALSE; |
+ path.m_FillType = FXFILL_WINDING; |
+ path.m_ClipPath.AppendTexts(&pCopy, 1); |
+ path.m_ColorState = textobj->m_ColorState; |
+ path.m_Path.New()->AppendRect(textobj->m_Left, textobj->m_Bottom, |
+ textobj->m_Right, textobj->m_Top); |
+ path.m_Left = textobj->m_Left; |
+ path.m_Bottom = textobj->m_Bottom; |
+ path.m_Right = textobj->m_Right; |
+ path.m_Top = textobj->m_Top; |
+ RenderSingleObject(&path, pObj2Device); |
+ return; |
+ } |
+ CFX_FontCache* pCache; |
+ if (pFont->m_pDocument) { |
+ pCache = pFont->m_pDocument->GetRenderData()->GetFontCache(); |
+ } else { |
+ pCache = CFX_GEModule::Get()->GetFontCache(); |
+ } |
+ CFX_FaceCache* pFaceCache = pCache->GetCachedFace(&pFont->m_Font); |
+ FX_FONTCACHE_DEFINE(pCache, &pFont->m_Font); |
+ CPDF_CharPosList CharPosList; |
+ CharPosList.Load(textobj->m_nChars, textobj->m_pCharCodes, |
+ textobj->m_pCharPos, pFont, font_size); |
+ for (FX_DWORD i = 0; i < CharPosList.m_nChars; i++) { |
+ FXTEXT_CHARPOS& charpos = CharPosList.m_pCharPos[i]; |
+ const CFX_PathData* pPath = pFaceCache->LoadGlyphPath( |
+ &pFont->m_Font, charpos.m_GlyphIndex, charpos.m_FontCharWidth); |
+ if (pPath == NULL) { |
+ continue; |
+ } |
+ CPDF_PathObject path; |
+ path.m_GraphState = textobj->m_GraphState; |
+ path.m_ColorState = textobj->m_ColorState; |
+ CFX_AffineMatrix matrix; |
+ if (charpos.m_bGlyphAdjust) |
+ matrix.Set(charpos.m_AdjustMatrix[0], charpos.m_AdjustMatrix[1], |
+ charpos.m_AdjustMatrix[2], charpos.m_AdjustMatrix[3], 0, 0); |
+ matrix.Concat(font_size, 0, 0, font_size, charpos.m_OriginX, |
+ charpos.m_OriginY); |
+ path.m_Path.New()->Append(pPath, &matrix); |
+ path.m_Matrix = *pTextMatrix; |
+ path.m_bStroke = bStroke; |
+ path.m_FillType = bFill ? FXFILL_WINDING : 0; |
+ path.CalcBoundingBox(); |
+ ProcessPath(&path, pObj2Device); |
+ } |
} |
-CFX_PathData* CPDF_Font::LoadGlyphPath(FX_DWORD charcode, int dest_width) |
-{ |
- int glyph_index = GlyphFromCharCode(charcode); |
- if (m_Font.m_Face == NULL) { |
- return NULL; |
- } |
- return m_Font.LoadGlyphPath(glyph_index, dest_width); |
+CFX_PathData* CPDF_Font::LoadGlyphPath(FX_DWORD charcode, int dest_width) { |
+ int glyph_index = GlyphFromCharCode(charcode); |
+ if (m_Font.m_Face == NULL) { |
+ return NULL; |
+ } |
+ return m_Font.LoadGlyphPath(glyph_index, dest_width); |
} |