| 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);
|
| }
|
|
|