| // Copyright 2014 PDFium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| |
| #include "core/fpdfapi/fpdf_render/render_int.h" |
| |
| #include <vector> |
| |
| #include "core/fpdfapi/fpdf_font/cpdf_cidfont.h" |
| #include "core/fpdfapi/fpdf_font/cpdf_type3char.h" |
| #include "core/fpdfapi/fpdf_font/cpdf_type3font.h" |
| #include "core/fpdfapi/fpdf_font/include/cpdf_font.h" |
| #include "core/fpdfapi/fpdf_page/include/cpdf_form.h" |
| #include "core/fpdfapi/fpdf_page/include/cpdf_imageobject.h" |
| #include "core/fpdfapi/fpdf_page/include/cpdf_pageobject.h" |
| #include "core/fpdfapi/fpdf_page/include/cpdf_pathobject.h" |
| #include "core/fpdfapi/fpdf_page/include/cpdf_textobject.h" |
| #include "core/fpdfapi/fpdf_page/pageint.h" |
| #include "core/fpdfapi/fpdf_parser/include/cpdf_dictionary.h" |
| #include "core/fpdfapi/fpdf_parser/include/cpdf_document.h" |
| #include "core/fpdfapi/fpdf_render/include/cpdf_renderoptions.h" |
| #include "core/fpdfapi/fpdf_render/include/cpdf_textrenderer.h" |
| #include "core/fxge/include/cfx_gemodule.h" |
| #include "core/fxge/include/cfx_pathdata.h" |
| #include "core/fxge/include/fx_ge.h" |
| |
| namespace { |
| |
| struct CPDF_UniqueKeyGen { |
| void Generate(int count, ...); |
| FX_CHAR m_Key[128]; |
| int m_KeyLen; |
| }; |
| |
| 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); |
| ((uint32_t*)m_Key)[i] = p; |
| } |
| va_end(argList); |
| m_KeyLen = count * sizeof(uint32_t); |
| } |
| |
| } // namespace |
| |
| CPDF_Type3Cache::CPDF_Type3Cache(CPDF_Type3Font* pFont) : m_pFont(pFont) {} |
| |
| CPDF_Type3Cache::~CPDF_Type3Cache() { |
| for (const auto& pair : m_SizeMap) { |
| delete pair.second; |
| } |
| m_SizeMap.clear(); |
| } |
| |
| CFX_GlyphBitmap* CPDF_Type3Cache::LoadGlyph(uint32_t charcode, |
| const CFX_Matrix* 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_ByteString FaceGlyphsKey(keygen.m_Key, keygen.m_KeyLen); |
| CPDF_Type3Glyphs* pSizeCache; |
| auto it = m_SizeMap.find(FaceGlyphsKey); |
| if (it == m_SizeMap.end()) { |
| pSizeCache = new CPDF_Type3Glyphs; |
| m_SizeMap[FaceGlyphsKey] = pSizeCache; |
| } else { |
| pSizeCache = it->second; |
| } |
| auto it2 = pSizeCache->m_GlyphMap.find(charcode); |
| if (it2 != pSizeCache->m_GlyphMap.end()) |
| return it2->second; |
| |
| CFX_GlyphBitmap* pGlyphBitmap = |
| RenderGlyph(pSizeCache, charcode, pMatrix, retinaScaleX, retinaScaleY); |
| pSizeCache->m_GlyphMap[charcode] = pGlyphBitmap; |
| return pGlyphBitmap; |
| } |
| |
| CPDF_Type3Glyphs::CPDF_Type3Glyphs() |
| : m_TopBlueCount(0), m_BottomBlueCount(0) {} |
| |
| CPDF_Type3Glyphs::~CPDF_Type3Glyphs() { |
| for (const auto& pair : m_GlyphMap) |
| delete pair.second; |
| } |
| |
| 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); |
| } |
| |
| 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; |
| } |
| return (width % 8) && (pBuf[width / 8] & (0xff << (8 - width % 8))); |
| } |
| |
| 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 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; |
| } |
| return -1; |
| } |
| |
| CFX_GlyphBitmap* CPDF_Type3Cache::RenderGlyph(CPDF_Type3Glyphs* pSize, |
| uint32_t charcode, |
| const CFX_Matrix* pMatrix, |
| FX_FLOAT retinaScaleX, |
| FX_FLOAT retinaScaleY) { |
| const CPDF_Type3Char* pChar = m_pFont->LoadChar(charcode); |
| if (!pChar || !pChar->m_pBitmap) |
| return nullptr; |
| |
| CFX_DIBitmap* pBitmap = pChar->m_pBitmap.get(); |
| CFX_Matrix 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); |
| std::unique_ptr<CFX_DIBitmap> pResBitmap; |
| int left = 0; |
| int top = 0; |
| 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 = _DetectFirstLastScan(pBitmap, TRUE); |
| int 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.reset(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); |
| } |
| } |
| } |
| if (!pResBitmap) { |
| image_matrix.Scale(retinaScaleX, retinaScaleY); |
| pResBitmap.reset(pBitmap->TransformTo(&image_matrix, left, top)); |
| } |
| if (!pResBitmap) |
| return nullptr; |
| |
| CFX_GlyphBitmap* pGlyph = new CFX_GlyphBitmap; |
| pGlyph->m_Left = left; |
| pGlyph->m_Top = -top; |
| pGlyph->m_Bitmap.TakeOver(pResBitmap.get()); |
| return pGlyph; |
| } |
| |
| FX_BOOL CPDF_RenderStatus::ProcessText(const CPDF_TextObject* textobj, |
| const CFX_Matrix* pObj2Device, |
| CFX_PathData* pClippingPath) { |
| if (textobj->m_nChars == 0) |
| return TRUE; |
| |
| const TextRenderingMode& text_render_mode = |
| textobj->m_TextState.GetObject()->m_TextMode; |
| if (text_render_mode == TextRenderingMode::MODE_INVISIBLE) |
| return TRUE; |
| |
| CPDF_Font* pFont = textobj->m_TextState.GetFont(); |
| if (pFont->IsType3Font()) |
| return ProcessType3Text(textobj, pObj2Device); |
| |
| bool bFill = false; |
| bool bStroke = false; |
| bool bClip = false; |
| if (pClippingPath) { |
| bClip = true; |
| } else { |
| switch (text_render_mode) { |
| case TextRenderingMode::MODE_FILL: |
| case TextRenderingMode::MODE_FILL_CLIP: |
| bFill = true; |
| break; |
| case TextRenderingMode::MODE_STROKE: |
| case TextRenderingMode::MODE_STROKE_CLIP: |
| if (pFont->GetFace() || |
| (pFont->GetSubstFont()->m_SubstFlags & FXFONT_SUBST_GLYPHPATH)) { |
| bStroke = true; |
| } else { |
| bFill = true; |
| } |
| break; |
| case TextRenderingMode::MODE_FILL_STROKE: |
| case TextRenderingMode::MODE_FILL_STROKE_CLIP: |
| bFill = true; |
| if (pFont->GetFace() || |
| (pFont->GetSubstFont()->m_SubstFlags & FXFONT_SUBST_GLYPHPATH)) { |
| bStroke = true; |
| } |
| break; |
| case TextRenderingMode::MODE_INVISIBLE: |
| // Already handled above, but the compiler is not smart enough to |
| // realize it. Fall through. |
| ASSERT(false); |
| case TextRenderingMode::MODE_CLIP: |
| return TRUE; |
| } |
| } |
| FX_ARGB stroke_argb = 0; |
| FX_ARGB fill_argb = 0; |
| bool bPattern = false; |
| if (bStroke) { |
| if (textobj->m_ColorState.GetStrokeColor()->IsPattern()) { |
| bPattern = true; |
| } else { |
| stroke_argb = GetStrokeArgb(textobj); |
| } |
| } |
| if (bFill) { |
| if (textobj->m_ColorState.GetFillColor()->IsPattern()) { |
| bPattern = true; |
| } else { |
| fill_argb = GetFillArgb(textobj); |
| } |
| } |
| CFX_Matrix text_matrix; |
| textobj->GetTextMatrix(&text_matrix); |
| if (!IsAvailableMatrix(text_matrix)) |
| 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_Matrix* pDeviceMatrix = pObj2Device; |
| CFX_Matrix device_matrix; |
| if (bStroke) { |
| const FX_FLOAT* pCTM = textobj->m_TextState.GetObject()->m_CTM; |
| if (pCTM[0] != 1.0f || pCTM[3] != 1.0f) { |
| CFX_Matrix 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 = |
| static_cast<const CPDF_PageObject*>(textobj) |
| ->m_GeneralState.GetObject(); |
| 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.GetObject(), 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) { |
| return nullptr; |
| } |
| 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) { |
| return; |
| } |
| pFont->m_pDocument->GetRenderData()->ReleaseCachedType3(pFont); |
| pFont->m_pDocument->GetPageData()->ReleaseFont(pFont->GetFontDict()); |
| } |
| |
| class CPDF_RefType3Cache { |
| public: |
| explicit CPDF_RefType3Cache(CPDF_Type3Font* pType3Font) |
| : m_dwCount(0), m_pType3Font(pType3Font) {} |
| ~CPDF_RefType3Cache() { |
| while (m_dwCount--) { |
| ReleaseCachedType3(m_pType3Font); |
| } |
| } |
| uint32_t m_dwCount; |
| CPDF_Type3Font* const m_pType3Font; |
| }; |
| |
| FX_BOOL CPDF_RenderStatus::ProcessType3Text(const CPDF_TextObject* textobj, |
| const CFX_Matrix* pObj2Device) { |
| CPDF_Type3Font* pType3Font = textobj->m_TextState.GetFont()->AsType3Font(); |
| for (int i = 0; i < m_Type3FontCache.GetSize(); ++i) { |
| if (m_Type3FontCache.GetAt(i) == pType3Font) |
| return TRUE; |
| } |
| |
| CFX_Matrix dCTM = m_pDevice->GetCTM(); |
| FX_FLOAT sa = FXSYS_fabs(dCTM.a); |
| FX_FLOAT sd = FXSYS_fabs(dCTM.d); |
| CFX_Matrix text_matrix; |
| textobj->GetTextMatrix(&text_matrix); |
| CFX_Matrix 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(); |
| std::vector<FXTEXT_GLYPHPOS> glyphs; |
| if (device_class == FXDC_DISPLAY) |
| glyphs.resize(textobj->m_nChars); |
| else if (fill_alpha < 255) |
| return FALSE; |
| |
| CPDF_RefType3Cache refTypeCache(pType3Font); |
| uint32_t* pChars = textobj->m_pCharCodes; |
| if (textobj->m_nChars == 1) |
| pChars = (uint32_t*)(&textobj->m_pCharCodes); |
| |
| for (int iChar = 0; iChar < textobj->m_nChars; iChar++) { |
| uint32_t charcode = pChars[iChar]; |
| if (charcode == (uint32_t)-1) |
| continue; |
| |
| CPDF_Type3Char* pType3Char = pType3Font->LoadChar(charcode); |
| if (!pType3Char) |
| continue; |
| |
| CFX_Matrix 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 (!glyphs.empty()) { |
| for (int i = 0; i < iChar; i++) { |
| const FXTEXT_GLYPHPOS& glyph = glyphs[i]; |
| if (!glyph.m_pGlyph) |
| 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); |
| } |
| glyphs.clear(); |
| } |
| 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 = nullptr; |
| if (pType3Char->m_pForm && pType3Char->m_pForm->m_pFormDict) { |
| pFormResource = |
| pType3Char->m_pForm->m_pFormDict->GetDictBy("Resources"); |
| } |
| if (fill_alpha == 255) { |
| CPDF_RenderStatus status; |
| status.Initialize(m_pContext, m_pDevice, nullptr, nullptr, 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.get(), &matrix); |
| m_pDevice->RestoreState(false); |
| } 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, |
| nullptr)) { |
| return TRUE; |
| } |
| bitmap_device.GetBitmap()->Clear(0); |
| CPDF_RenderStatus status; |
| status.Initialize(m_pContext, &bitmap_device, nullptr, nullptr, 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.get(), &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) |
| continue; |
| |
| int origin_x = FXSYS_round(matrix.e); |
| int origin_y = FXSYS_round(matrix.f); |
| if (glyphs.empty()) { |
| m_pDevice->SetBitMask(&pBitmap->m_Bitmap, origin_x + pBitmap->m_Left, |
| origin_y - pBitmap->m_Top, fill_argb); |
| } else { |
| glyphs[iChar].m_pGlyph = pBitmap; |
| glyphs[iChar].m_OriginX = origin_x; |
| glyphs[iChar].m_OriginY = origin_y; |
| } |
| } else { |
| CFX_Matrix image_matrix = pType3Char->m_ImageMatrix; |
| image_matrix.Concat(matrix); |
| CPDF_ImageRenderer renderer; |
| if (renderer.Start(this, pType3Char->m_pBitmap.get(), fill_argb, 255, |
| &image_matrix, 0, FALSE)) { |
| renderer.Continue(nullptr); |
| } |
| if (!renderer.m_Result) |
| return FALSE; |
| } |
| } |
| } |
| |
| if (glyphs.empty()) |
| return TRUE; |
| |
| FX_RECT rect = FXGE_GetGlyphsBBox(glyphs, 0, sa, sd); |
| CFX_DIBitmap bitmap; |
| if (!bitmap.Create(static_cast<int>(rect.Width() * sa), |
| static_cast<int>(rect.Height() * sd), FXDIB_8bppMask)) { |
| return TRUE; |
| } |
| bitmap.Clear(0); |
| for (const FXTEXT_GLYPHPOS& glyph : glyphs) { |
| if (!glyph.m_pGlyph) |
| continue; |
| |
| bitmap.TransferBitmap( |
| static_cast<int>( |
| (glyph.m_OriginX + glyph.m_pGlyph->m_Left - rect.left) * sa), |
| static_cast<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); |
| return TRUE; |
| } |
| |
| class CPDF_CharPosList { |
| public: |
| CPDF_CharPosList(); |
| ~CPDF_CharPosList(); |
| void Load(int nChars, |
| uint32_t* pCharCodes, |
| FX_FLOAT* pCharPos, |
| CPDF_Font* pFont, |
| FX_FLOAT font_size); |
| FXTEXT_CHARPOS* m_pCharPos; |
| uint32_t m_nChars; |
| }; |
| |
| CPDF_CharPosList::CPDF_CharPosList() { |
| m_pCharPos = nullptr; |
| } |
| |
| CPDF_CharPosList::~CPDF_CharPosList() { |
| FX_Free(m_pCharPos); |
| } |
| |
| void CPDF_CharPosList::Load(int nChars, |
| uint32_t* pCharCodes, |
| FX_FLOAT* pCharPos, |
| CPDF_Font* pFont, |
| FX_FLOAT FontSize) { |
| m_pCharPos = FX_Alloc(FXTEXT_CHARPOS, nChars); |
| m_nChars = 0; |
| CPDF_CIDFont* pCIDFont = pFont->AsCIDFont(); |
| FX_BOOL bVertWriting = pCIDFont && pCIDFont->IsVertWriting(); |
| for (int iChar = 0; iChar < nChars; iChar++) { |
| uint32_t CharCode = |
| nChars == 1 ? (uint32_t)(uintptr_t)pCharCodes : pCharCodes[iChar]; |
| if (CharCode == (uint32_t)-1) { |
| continue; |
| } |
| bool bVert = false; |
| FXTEXT_CHARPOS& charpos = m_pCharPos[m_nChars++]; |
| if (pCIDFont) { |
| charpos.m_bFontStyle = true; |
| } |
| charpos.m_GlyphIndex = pFont->GlyphFromCharCode(CharCode, &bVert); |
| #if _FXM_PLATFORM_ == _FXM_PLATFORM_APPLE_ |
| charpos.m_ExtGID = pFont->GlyphFromCharCodeExt(CharCode); |
| #endif |
| if (!pFont->IsEmbedded() && !pFont->IsCIDFont()) { |
| 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) { |
| continue; |
| } |
| uint16_t 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] = pCIDFont->CIDTransformToFloat(pTransform[0]); |
| charpos.m_AdjustMatrix[2] = pCIDFont->CIDTransformToFloat(pTransform[2]); |
| charpos.m_AdjustMatrix[1] = pCIDFont->CIDTransformToFloat(pTransform[1]); |
| charpos.m_AdjustMatrix[3] = pCIDFont->CIDTransformToFloat(pTransform[3]); |
| charpos.m_OriginX += |
| pCIDFont->CIDTransformToFloat(pTransform[4]) * FontSize; |
| charpos.m_OriginY += |
| pCIDFont->CIDTransformToFloat(pTransform[5]) * FontSize; |
| charpos.m_bGlyphAdjust = TRUE; |
| } |
| } |
| } |
| |
| // static |
| FX_BOOL CPDF_TextRenderer::DrawTextPath(CFX_RenderDevice* pDevice, |
| int nChars, |
| uint32_t* pCharCodes, |
| FX_FLOAT* pCharPos, |
| CPDF_Font* pFont, |
| FX_FLOAT font_size, |
| const CFX_Matrix* pText2User, |
| const CFX_Matrix* 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() |
| : nullptr; |
| CPDF_CharPosList CharPosList; |
| CharPosList.Load(nChars, pCharCodes, pCharPos, pFont, font_size); |
| return pDevice->DrawTextPathWithFlags( |
| CharPosList.m_nChars, CharPosList.m_pCharPos, &pFont->m_Font, pCache, |
| font_size, pText2User, pUser2Device, pGraphState, fill_argb, stroke_argb, |
| pClippingPath, nFlag); |
| } |
| |
| // static |
| void CPDF_TextRenderer::DrawTextString(CFX_RenderDevice* pDevice, |
| FX_FLOAT origin_x, |
| FX_FLOAT origin_y, |
| CPDF_Font* pFont, |
| FX_FLOAT font_size, |
| const CFX_Matrix* pMatrix, |
| const CFX_ByteString& str, |
| FX_ARGB fill_argb, |
| FX_ARGB stroke_argb, |
| const CFX_GraphStateData* pGraphState, |
| const CPDF_RenderOptions* pOptions) { |
| if (pFont->IsType3Font()) |
| return; |
| |
| int nChars = pFont->CountChar(str.c_str(), str.GetLength()); |
| if (nChars <= 0) |
| return; |
| |
| int offset = 0; |
| uint32_t* pCharCodes; |
| FX_FLOAT* pCharPos; |
| std::vector<uint32_t> codes; |
| std::vector<FX_FLOAT> positions; |
| if (nChars == 1) { |
| pCharCodes = reinterpret_cast<uint32_t*>( |
| pFont->GetNextChar(str.c_str(), str.GetLength(), offset)); |
| pCharPos = nullptr; |
| } else { |
| codes.resize(nChars); |
| positions.resize(nChars - 1); |
| FX_FLOAT cur_pos = 0; |
| for (int i = 0; i < nChars; i++) { |
| codes[i] = pFont->GetNextChar(str.c_str(), str.GetLength(), offset); |
| if (i) |
| positions[i - 1] = cur_pos; |
| cur_pos += pFont->GetCharWidthF(codes[i]) * font_size / 1000; |
| } |
| pCharCodes = codes.data(); |
| pCharPos = positions.data(); |
| } |
| CFX_Matrix matrix; |
| if (pMatrix) |
| matrix = *pMatrix; |
| |
| matrix.e = origin_x; |
| matrix.f = origin_y; |
| |
| 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, nullptr, pGraphState, fill_argb, stroke_argb, nullptr, |
| 0); |
| } |
| } |
| |
| // static |
| FX_BOOL CPDF_TextRenderer::DrawNormalText(CFX_RenderDevice* pDevice, |
| int nChars, |
| uint32_t* pCharCodes, |
| FX_FLOAT* pCharPos, |
| CPDF_Font* pFont, |
| FX_FLOAT font_size, |
| const CFX_Matrix* pText2Device, |
| FX_ARGB fill_argb, |
| const CPDF_RenderOptions* pOptions) { |
| CFX_FontCache* pCache = |
| pFont->m_pDocument ? pFont->m_pDocument->GetRenderData()->GetFontCache() |
| : nullptr; |
| CPDF_CharPosList CharPosList; |
| CharPosList.Load(nChars, pCharCodes, pCharPos, pFont, font_size); |
| int FXGE_flags = 0; |
| if (pOptions) { |
| uint32_t 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->IsCIDFont()) { |
| 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_Matrix* pObj2Device, |
| CPDF_Font* pFont, |
| FX_FLOAT font_size, |
| const CFX_Matrix* pTextMatrix, |
| FX_BOOL bFill, |
| FX_BOOL bStroke) { |
| if (!bStroke) { |
| CPDF_PathObject path; |
| std::vector<std::unique_ptr<CPDF_TextObject>> pCopy; |
| pCopy.push_back(std::unique_ptr<CPDF_TextObject>(textobj->Clone())); |
| path.m_bStroke = FALSE; |
| path.m_FillType = FXFILL_WINDING; |
| path.m_ClipPath.AppendTexts(&pCopy); |
| 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); |
| CFX_AutoFontCache autoFontCache(pCache, &pFont->m_Font); |
| CPDF_CharPosList CharPosList; |
| CharPosList.Load(textobj->m_nChars, textobj->m_pCharCodes, |
| textobj->m_pCharPos, pFont, font_size); |
| for (uint32_t 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) { |
| continue; |
| } |
| CPDF_PathObject path; |
| path.m_GraphState = textobj->m_GraphState; |
| path.m_ColorState = textobj->m_ColorState; |
| CFX_Matrix 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); |
| } |
| } |