// Copyright 2016 The PDFium Authors // 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/render/cpdf_renderstatus.h" #include #include #include #include #include #include #include #include "build/build_config.h" #include "constants/transparency.h" #include "core/fpdfapi/font/cpdf_font.h" #include "core/fpdfapi/font/cpdf_type3char.h" #include "core/fpdfapi/font/cpdf_type3font.h" #include "core/fpdfapi/page/cpdf_docpagedata.h" #include "core/fpdfapi/page/cpdf_form.h" #include "core/fpdfapi/page/cpdf_formobject.h" #include "core/fpdfapi/page/cpdf_function.h" #include "core/fpdfapi/page/cpdf_graphicstates.h" #include "core/fpdfapi/page/cpdf_image.h" #include "core/fpdfapi/page/cpdf_imageobject.h" #include "core/fpdfapi/page/cpdf_occontext.h" #include "core/fpdfapi/page/cpdf_page.h" #include "core/fpdfapi/page/cpdf_pageimagecache.h" #include "core/fpdfapi/page/cpdf_pageobject.h" #include "core/fpdfapi/page/cpdf_pathobject.h" #include "core/fpdfapi/page/cpdf_shadingobject.h" #include "core/fpdfapi/page/cpdf_shadingpattern.h" #include "core/fpdfapi/page/cpdf_textobject.h" #include "core/fpdfapi/page/cpdf_tilingpattern.h" #include "core/fpdfapi/page/cpdf_transferfunc.h" #include "core/fpdfapi/parser/cpdf_array.h" #include "core/fpdfapi/parser/cpdf_document.h" #include "core/fpdfapi/parser/cpdf_stream.h" #include "core/fpdfapi/parser/fpdf_parser_utility.h" #include "core/fpdfapi/render/charposlist.h" #include "core/fpdfapi/render/cpdf_docrenderdata.h" #include "core/fpdfapi/render/cpdf_imagerenderer.h" #include "core/fpdfapi/render/cpdf_rendercontext.h" #include "core/fpdfapi/render/cpdf_renderoptions.h" #include "core/fpdfapi/render/cpdf_rendershading.h" #include "core/fpdfapi/render/cpdf_rendertiling.h" #include "core/fpdfapi/render/cpdf_textrenderer.h" #include "core/fpdfapi/render/cpdf_type3cache.h" #include "core/fxcrt/autorestorer.h" #include "core/fxcrt/check.h" #include "core/fxcrt/compiler_specific.h" #include "core/fxcrt/containers/contains.h" #include "core/fxcrt/data_vector.h" #include "core/fxcrt/fx_2d_size.h" #include "core/fxcrt/fx_safe_types.h" #include "core/fxcrt/fx_system.h" #include "core/fxcrt/notreached.h" #include "core/fxcrt/span.h" #include "core/fxcrt/stl_util.h" #include "core/fxcrt/unowned_ptr.h" #include "core/fxge/agg/cfx_agg_imagerenderer.h" #include "core/fxge/cfx_defaultrenderdevice.h" #include "core/fxge/cfx_fillrenderoptions.h" #include "core/fxge/cfx_glyphbitmap.h" #include "core/fxge/cfx_path.h" #include "core/fxge/dib/cfx_dibitmap.h" #include "core/fxge/fx_font.h" #include "core/fxge/renderdevicedriver_iface.h" #include "core/fxge/text_char_pos.h" #include "core/fxge/text_glyph_pos.h" #if BUILDFLAG(IS_WIN) #include "core/fpdfapi/render/cpdf_scaledrenderbuffer.h" #endif namespace { constexpr int kRenderMaxRecursionDepth = 64; int g_CurrentRecursionDepth = 0; CFX_FillRenderOptions GetFillOptionsForDrawPathWithBlend( const CPDF_RenderOptions::Options& options, const CPDF_PathObject* path_obj, CFX_FillRenderOptions::FillType fill_type, bool is_stroke, bool is_type3_char) { CFX_FillRenderOptions fill_options(fill_type); if (fill_type != CFX_FillRenderOptions::FillType::kNoFill && options.bRectAA) { fill_options.rect_aa = true; } if (options.bNoPathSmooth) { fill_options.aliased_path = true; } if (path_obj->general_state().GetStrokeAdjust()) { fill_options.adjust_stroke = true; } if (is_stroke) { fill_options.stroke = true; } if (is_type3_char) { fill_options.text_mode = true; } return fill_options; } CFX_FillRenderOptions GetFillOptionsForDrawTextPath( const CPDF_RenderOptions::Options& options, const CPDF_TextObject* text_obj, bool is_stroke, bool is_fill) { CFX_FillRenderOptions fill_options; if (is_stroke && is_fill) { fill_options.stroke = true; fill_options.stroke_text_mode = true; } if (text_obj->general_state().GetStrokeAdjust()) { fill_options.adjust_stroke = true; } if (options.bNoTextSmooth) { fill_options.aliased_path = true; } return fill_options; } FXDIB_Format GetFormatForLuminosity(bool is_luminosity) { if (!is_luminosity) return FXDIB_Format::k8bppMask; #if BUILDFLAG(IS_APPLE) return FXDIB_Format::kBgrx; #else if (CFX_DefaultRenderDevice::UseSkiaRenderer()) { return FXDIB_Format::kBgrx; } return FXDIB_Format::kBgr; #endif } bool IsAvailableMatrix(const CFX_Matrix& matrix) { if (matrix.a == 0 || matrix.d == 0) return matrix.b != 0 && matrix.c != 0; if (matrix.b == 0 || matrix.c == 0) return matrix.a != 0 && matrix.d != 0; return true; } bool MissingFillColor(const CPDF_ColorState* pColorState) { return !pColorState->HasRef() || pColorState->GetFillColor()->IsNull(); } bool MissingStrokeColor(const CPDF_ColorState* pColorState) { return !pColorState->HasRef() || pColorState->GetStrokeColor()->IsNull(); } bool Type3CharMissingFillColor(const CPDF_Type3Char* pChar, const CPDF_ColorState* pColorState) { return pChar && (!pChar->colored() || MissingFillColor(pColorState)); } bool Type3CharMissingStrokeColor(const CPDF_Type3Char* pChar, const CPDF_ColorState* pColorState) { return pChar && (!pChar->colored() || MissingStrokeColor(pColorState)); } } // namespace CPDF_RenderStatus::CPDF_RenderStatus(CPDF_RenderContext* pContext, CFX_RenderDevice* pDevice) : m_pContext(pContext), m_pDevice(pDevice) {} CPDF_RenderStatus::~CPDF_RenderStatus() = default; void CPDF_RenderStatus::Initialize(const CPDF_RenderStatus* pParentStatus, const CPDF_GraphicStates* pInitialStates) { #if BUILDFLAG(IS_WIN) m_bPrint = m_pDevice->GetDeviceType() == DeviceType::kPrinter; #endif m_pPageResource.Reset(m_pContext->GetPageResources()); if (pInitialStates && !m_pType3Char) { m_InitialStates = *pInitialStates; if (pParentStatus) { if (!m_InitialStates.color_state().HasFillColor()) { m_InitialStates.mutable_color_state().SetFillColorRef( pParentStatus->m_InitialStates.color_state().GetFillColorRef()); *m_InitialStates.mutable_color_state().GetMutableFillColor() = *pParentStatus->m_InitialStates.color_state().GetFillColor(); } if (!m_InitialStates.color_state().HasStrokeColor()) { m_InitialStates.mutable_color_state().SetStrokeColorRef( pParentStatus->m_InitialStates.color_state().GetFillColorRef()); *m_InitialStates.mutable_color_state().GetMutableStrokeColor() = *pParentStatus->m_InitialStates.color_state().GetStrokeColor(); } } } else { m_InitialStates.SetDefaultStates(); } } void CPDF_RenderStatus::RenderObjectList( const CPDF_PageObjectHolder* pObjectHolder, const CFX_Matrix& mtObj2Device) { CFX_FloatRect clip_rect = mtObj2Device.GetInverse().TransformRect( CFX_FloatRect(m_pDevice->GetClipBox())); for (const auto& pCurObj : *pObjectHolder) { if (pCurObj.get() == m_pStopObj) { m_bStopped = true; return; } if (!pCurObj || !pCurObj->IsActive()) { continue; } if (pCurObj->GetRect().left > clip_rect.right || pCurObj->GetRect().right < clip_rect.left || pCurObj->GetRect().bottom > clip_rect.top || pCurObj->GetRect().top < clip_rect.bottom) { continue; } RenderSingleObject(pCurObj.get(), mtObj2Device); if (m_bStopped) return; } } void CPDF_RenderStatus::RenderSingleObject(CPDF_PageObject* pObj, const CFX_Matrix& mtObj2Device) { AutoRestorer restorer(&g_CurrentRecursionDepth); if (++g_CurrentRecursionDepth > kRenderMaxRecursionDepth) { return; } m_pCurObj = pObj; if (!m_Options.CheckPageObjectVisible(pObj)) { return; } ProcessClipPath(pObj->clip_path(), mtObj2Device); if (ProcessTransparency(pObj, mtObj2Device)) { return; } ProcessObjectNoClip(pObj, mtObj2Device); } bool CPDF_RenderStatus::ContinueSingleObject(CPDF_PageObject* pObj, const CFX_Matrix& mtObj2Device, PauseIndicatorIface* pPause) { if (m_pImageRenderer) { if (m_pImageRenderer->Continue(pPause)) return true; if (!m_pImageRenderer->GetResult()) DrawObjWithBackground(pObj, mtObj2Device); m_pImageRenderer.reset(); return false; } m_pCurObj = pObj; if (!m_Options.CheckPageObjectVisible(pObj)) return false; ProcessClipPath(pObj->clip_path(), mtObj2Device); if (ProcessTransparency(pObj, mtObj2Device)) return false; if (!pObj->IsImage()) { ProcessObjectNoClip(pObj, mtObj2Device); return false; } m_pImageRenderer = std::make_unique(this); if (!m_pImageRenderer->Start(pObj->AsImage(), mtObj2Device, /*bStdCS=*/false)) { if (!m_pImageRenderer->GetResult()) DrawObjWithBackground(pObj, mtObj2Device); m_pImageRenderer.reset(); return false; } return ContinueSingleObject(pObj, mtObj2Device, pPause); } FX_RECT CPDF_RenderStatus::GetObjectClippedRect( const CPDF_PageObject* pObj, const CFX_Matrix& mtObj2Device) const { FX_RECT rect = pObj->GetTransformedBBox(mtObj2Device); rect.Intersect(m_pDevice->GetClipBox()); return rect; } void CPDF_RenderStatus::ProcessObjectNoClip(CPDF_PageObject* pObj, const CFX_Matrix& mtObj2Device) { bool bRet = false; switch (pObj->GetType()) { case CPDF_PageObject::Type::kText: bRet = ProcessText(pObj->AsText(), mtObj2Device, nullptr); break; case CPDF_PageObject::Type::kPath: bRet = ProcessPath(pObj->AsPath(), mtObj2Device); break; case CPDF_PageObject::Type::kImage: bRet = ProcessImage(pObj->AsImage(), mtObj2Device); break; case CPDF_PageObject::Type::kShading: ProcessShading(pObj->AsShading(), mtObj2Device); return; case CPDF_PageObject::Type::kForm: bRet = ProcessForm(pObj->AsForm(), mtObj2Device); break; } if (!bRet) DrawObjWithBackground(pObj, mtObj2Device); } bool CPDF_RenderStatus::DrawObjWithBlend(CPDF_PageObject* pObj, const CFX_Matrix& mtObj2Device) { switch (pObj->GetType()) { case CPDF_PageObject::Type::kPath: return ProcessPath(pObj->AsPath(), mtObj2Device); case CPDF_PageObject::Type::kImage: return ProcessImage(pObj->AsImage(), mtObj2Device); case CPDF_PageObject::Type::kForm: return ProcessForm(pObj->AsForm(), mtObj2Device); case CPDF_PageObject::Type::kText: case CPDF_PageObject::Type::kShading: return false; } } void CPDF_RenderStatus::DrawObjWithBackground(CPDF_PageObject* pObj, const CFX_Matrix& mtObj2Device) { FX_RECT rect = GetObjectClippedRect(pObj, mtObj2Device); if (rect.IsEmpty()) return; const bool needs_buffer = !(m_pDevice->GetDeviceCaps(FXDC_RENDER_CAPS) & FXRC_GET_BITS); if (!needs_buffer) { DrawObjWithBackgroundToDevice(pObj, mtObj2Device, m_pDevice, CFX_Matrix()); return; } #if BUILDFLAG(IS_WIN) CPDF_ScaledRenderBuffer buffer(m_pDevice, rect); int res = (pObj->IsImage() && IsPrint()) ? 0 : 300; if (!buffer.Initialize(m_pContext, pObj, m_Options, res)) { return; } DrawObjWithBackgroundToDevice(pObj, mtObj2Device, buffer.GetDevice(), buffer.GetMatrix()); buffer.OutputToDevice(); #else NOTREACHED_NORETURN(); #endif } void CPDF_RenderStatus::DrawObjWithBackgroundToDevice( CPDF_PageObject* obj, const CFX_Matrix& object_to_device, CFX_RenderDevice* device, const CFX_Matrix& device_matrix) { RetainPtr pFormResource; const CPDF_FormObject* pFormObj = obj->AsForm(); if (pFormObj) { pFormResource = pFormObj->form()->GetDict()->GetDictFor("Resources"); } CPDF_RenderStatus status(m_pContext, device); status.SetOptions(m_Options); status.SetDeviceMatrix(device_matrix); status.SetTransparency(m_Transparency); status.SetDropObjects(m_bDropObjects); status.SetFormResource(std::move(pFormResource)); status.SetInGroup(m_bInGroup); status.Initialize(nullptr, nullptr); status.RenderSingleObject(obj, object_to_device * device_matrix); } bool CPDF_RenderStatus::ProcessForm(const CPDF_FormObject* pFormObj, const CFX_Matrix& mtObj2Device) { RetainPtr pOC = pFormObj->form()->GetDict()->GetDictFor("OC"); if (pOC && !m_Options.CheckOCGDictVisible(pOC.Get())) return true; CFX_Matrix matrix = pFormObj->form_matrix() * mtObj2Device; RetainPtr pResources = pFormObj->form()->GetDict()->GetDictFor("Resources"); CPDF_RenderStatus status(m_pContext, m_pDevice); status.SetOptions(m_Options); status.SetStopObject(m_pStopObj); status.SetTransparency(m_Transparency); status.SetDropObjects(m_bDropObjects); status.SetFormResource(std::move(pResources)); status.SetInGroup(m_bInGroup); status.Initialize(this, &pFormObj->graphic_states()); { CFX_RenderDevice::StateRestorer restorer(m_pDevice); status.RenderObjectList(pFormObj->form(), matrix); m_bStopped = status.m_bStopped; } return true; } bool CPDF_RenderStatus::ProcessPath(CPDF_PathObject* path_obj, const CFX_Matrix& mtObj2Device) { CFX_FillRenderOptions::FillType fill_type = path_obj->filltype(); bool stroke = path_obj->stroke(); ProcessPathPattern(path_obj, mtObj2Device, &fill_type, &stroke); if (fill_type == CFX_FillRenderOptions::FillType::kNoFill && !stroke) return true; // If the option to convert fill paths to stroke is enabled for forced color, // set |fill_type| to FillType::kNoFill and |stroke| to true. CPDF_RenderOptions::Options& options = m_Options.GetOptions(); if (m_Options.ColorModeIs(CPDF_RenderOptions::Type::kForcedColor) && options.bConvertFillToStroke && fill_type != CFX_FillRenderOptions::FillType::kNoFill) { stroke = true; fill_type = CFX_FillRenderOptions::FillType::kNoFill; } uint32_t fill_argb = fill_type != CFX_FillRenderOptions::FillType::kNoFill ? GetFillArgb(path_obj) : 0; uint32_t stroke_argb = stroke ? GetStrokeArgb(path_obj) : 0; CFX_Matrix path_matrix = path_obj->matrix() * mtObj2Device; if (!IsAvailableMatrix(path_matrix)) { return true; } return m_pDevice->DrawPath( *path_obj->path().GetObject(), &path_matrix, path_obj->graph_state().GetObject(), fill_argb, stroke_argb, GetFillOptionsForDrawPathWithBlend(options, path_obj, fill_type, stroke, m_pType3Char)); } RetainPtr CPDF_RenderStatus::GetTransferFunc( RetainPtr pObj) const { DCHECK(pObj); auto* pDocCache = CPDF_DocRenderData::FromDocument(m_pContext->GetDocument()); return pDocCache ? pDocCache->GetTransferFunc(std::move(pObj)) : nullptr; } FX_ARGB CPDF_RenderStatus::GetFillArgb(CPDF_PageObject* pObj) const { if (Type3CharMissingFillColor(m_pType3Char, &pObj->color_state())) { return m_T3FillColor; } return GetFillArgbForType3(pObj); } FX_ARGB CPDF_RenderStatus::GetFillArgbForType3(CPDF_PageObject* pObj) const { const CPDF_ColorState* pColorState = &pObj->color_state(); if (MissingFillColor(pColorState)) pColorState = &m_InitialStates.color_state(); FX_COLORREF colorref = pColorState->GetFillColorRef(); if (colorref == 0xFFFFFFFF) return 0; int32_t alpha = static_cast((pObj->general_state().GetFillAlpha() * 255)); RetainPtr pTR = pObj->general_state().GetTR(); if (pTR) { if (!pObj->general_state().GetTransferFunc()) { pObj->mutable_general_state().SetTransferFunc( GetTransferFunc(std::move(pTR))); } if (pObj->general_state().GetTransferFunc()) { colorref = pObj->general_state().GetTransferFunc()->TranslateColor(colorref); } } return m_Options.TranslateObjectFillColor( AlphaAndColorRefToArgb(alpha, colorref), pObj->GetType()); } FX_ARGB CPDF_RenderStatus::GetStrokeArgb(CPDF_PageObject* pObj) const { const CPDF_ColorState* pColorState = &pObj->color_state(); if (Type3CharMissingStrokeColor(m_pType3Char, pColorState)) return m_T3FillColor; if (MissingStrokeColor(pColorState)) pColorState = &m_InitialStates.color_state(); FX_COLORREF colorref = pColorState->GetStrokeColorRef(); if (colorref == 0xFFFFFFFF) return 0; int32_t alpha = static_cast(pObj->general_state().GetStrokeAlpha() * 255); // not rounded. RetainPtr pTR = pObj->general_state().GetTR(); if (pTR) { if (!pObj->general_state().GetTransferFunc()) { pObj->mutable_general_state().SetTransferFunc( GetTransferFunc(std::move(pTR))); } if (pObj->general_state().GetTransferFunc()) { colorref = pObj->general_state().GetTransferFunc()->TranslateColor(colorref); } } return m_Options.TranslateObjectStrokeColor( AlphaAndColorRefToArgb(alpha, colorref), pObj->GetType()); } void CPDF_RenderStatus::ProcessClipPath(const CPDF_ClipPath& ClipPath, const CFX_Matrix& mtObj2Device) { if (!ClipPath.HasRef()) { if (m_LastClipPath.HasRef()) { m_pDevice->RestoreState(true); m_LastClipPath.SetNull(); } return; } if (m_LastClipPath == ClipPath) return; m_LastClipPath = ClipPath; m_pDevice->RestoreState(true); for (size_t i = 0; i < ClipPath.GetPathCount(); ++i) { const CFX_Path* pPath = ClipPath.GetPath(i).GetObject(); if (!pPath) continue; if (pPath->GetPoints().empty()) { CFX_Path empty_path; empty_path.AppendRect(-1, -1, 0, 0); m_pDevice->SetClip_PathFill(empty_path, nullptr, CFX_FillRenderOptions::WindingOptions()); } else { m_pDevice->SetClip_PathFill( *pPath, &mtObj2Device, CFX_FillRenderOptions(ClipPath.GetClipType(i))); } } if (ClipPath.GetTextCount() == 0) return; if (!IsPrint() && !(m_pDevice->GetDeviceCaps(FXDC_RENDER_CAPS) & FXRC_SOFT_CLIP)) { return; } std::unique_ptr pTextClippingPath; for (size_t i = 0; i < ClipPath.GetTextCount(); ++i) { CPDF_TextObject* pText = ClipPath.GetText(i); if (pText) { if (!pTextClippingPath) pTextClippingPath = std::make_unique(); ProcessText(pText, mtObj2Device, pTextClippingPath.get()); continue; } if (!pTextClippingPath) continue; CFX_FillRenderOptions fill_options(CFX_FillRenderOptions::WindingOptions()); if (m_Options.GetOptions().bNoTextSmooth) { fill_options.aliased_path = true; } m_pDevice->SetClip_PathFill(*pTextClippingPath, nullptr, fill_options); pTextClippingPath.reset(); } } bool CPDF_RenderStatus::ClipPattern(const CPDF_PageObject* page_obj, const CFX_Matrix& mtObj2Device, bool stroke) { if (page_obj->IsPath()) return SelectClipPath(page_obj->AsPath(), mtObj2Device, stroke); if (page_obj->IsImage()) { m_pDevice->SetClip_Rect(page_obj->GetTransformedBBox(mtObj2Device)); return true; } return false; } bool CPDF_RenderStatus::SelectClipPath(const CPDF_PathObject* path_obj, const CFX_Matrix& mtObj2Device, bool stroke) { CFX_Matrix path_matrix = path_obj->matrix() * mtObj2Device; if (stroke) { return m_pDevice->SetClip_PathStroke(*path_obj->path().GetObject(), &path_matrix, path_obj->graph_state().GetObject()); } CFX_FillRenderOptions fill_options(path_obj->filltype()); if (m_Options.GetOptions().bNoPathSmooth) { fill_options.aliased_path = true; } return m_pDevice->SetClip_PathFill(*path_obj->path().GetObject(), &path_matrix, fill_options); } bool CPDF_RenderStatus::ProcessTransparency(CPDF_PageObject* pPageObj, const CFX_Matrix& mtObj2Device) { const BlendMode blend_type = pPageObj->general_state().GetBlendType(); RetainPtr pSMaskDict = pPageObj->mutable_general_state().GetMutableSoftMask(); if (pSMaskDict) { if (pPageObj->IsImage() && pPageObj->AsImage()->GetImage()->GetDict()->KeyExist("SMask")) { pSMaskDict = nullptr; } } RetainPtr pFormResource; float group_alpha = 1.0f; float initial_alpha = 1.0f; CPDF_Transparency transparency = m_Transparency; bool bGroupTransparent = false; const CPDF_FormObject* pFormObj = pPageObj->AsForm(); if (pFormObj) { group_alpha = pFormObj->general_state().GetFillAlpha(); transparency = pFormObj->form()->GetTransparency(); bGroupTransparent = transparency.IsIsolated(); pFormResource = pFormObj->form()->GetDict()->GetDictFor("Resources"); initial_alpha = m_InitialStates.general_state().GetFillAlpha(); } bool bTextClip = !IsPrint() && pPageObj->clip_path().HasRef() && pPageObj->clip_path().GetTextCount() > 0 && !(m_pDevice->GetDeviceCaps(FXDC_RENDER_CAPS) & FXRC_SOFT_CLIP); if (!pSMaskDict && group_alpha == 1.0f && blend_type == BlendMode::kNormal && !bTextClip && !bGroupTransparent && initial_alpha == 1.0f) { return false; } #if BUILDFLAG(IS_WIN) if (IsPrint()) { DrawObjWithBackground(pPageObj, mtObj2Device); return true; } #endif FX_RECT rect = pPageObj->GetTransformedBBox(mtObj2Device); rect.Intersect(m_pDevice->GetClipBox()); if (rect.IsEmpty()) return true; const int width = rect.Width(); const int height = rect.Height(); CFX_DefaultRenderDevice bitmap_device; RetainPtr backdrop; if (!transparency.IsIsolated() && (m_pDevice->GetRenderCaps() & FXRC_GET_BITS)) { backdrop = pdfium::MakeRetain(); if (!m_pDevice->CreateCompatibleBitmap(backdrop, width, height)) return true; m_pDevice->GetDIBits(backdrop, rect.left, rect.top); } if (!bitmap_device.CreateWithBackdrop( width, height, GetCompatibleArgbFormat(), std::move(backdrop))) { return true; } CFX_Matrix new_matrix = mtObj2Device; new_matrix.Translate(-rect.left, -rect.top); RetainPtr text_mask_bitmap; if (bTextClip) { text_mask_bitmap = pdfium::MakeRetain(); if (!text_mask_bitmap->Create(width, height, FXDIB_Format::k8bppMask)) { return true; } CFX_DefaultRenderDevice text_device; text_device.Attach(text_mask_bitmap); for (size_t i = 0; i < pPageObj->clip_path().GetTextCount(); ++i) { CPDF_TextObject* textobj = pPageObj->clip_path().GetText(i); if (!textobj) break; // TODO(thestig): Should we check the return value here? CPDF_TextRenderer::DrawTextPath( &text_device, textobj->GetCharCodes(), textobj->GetCharPositions(), textobj->text_state().GetFont().Get(), textobj->text_state().GetFontSize(), textobj->GetTextMatrix(), &new_matrix, textobj->graph_state().GetObject(), 0xffffffff, 0, nullptr, CFX_FillRenderOptions()); } } CPDF_RenderStatus bitmap_render(m_pContext, &bitmap_device); bitmap_render.SetOptions(m_Options); bitmap_render.SetStopObject(m_pStopObj); bitmap_render.SetStdCS(true); bitmap_render.SetDropObjects(m_bDropObjects); bitmap_render.SetFormResource(std::move(pFormResource)); bitmap_render.SetInGroup(transparency.IsGroup()); bitmap_render.Initialize(nullptr, nullptr); bitmap_render.ProcessObjectNoClip(pPageObj, new_matrix); m_bStopped = bitmap_render.m_bStopped; if (pSMaskDict) { CFX_Matrix smask_matrix = *pPageObj->general_state().GetSMaskMatrix() * mtObj2Device; RetainPtr smask_bitmap = LoadSMask(pSMaskDict.Get(), rect, smask_matrix); if (smask_bitmap) { bitmap_device.MultiplyAlphaMask(std::move(smask_bitmap)); } } if (text_mask_bitmap) { bitmap_device.MultiplyAlphaMask(std::move(text_mask_bitmap)); } if (transparency.IsGroup()) { bitmap_device.MultiplyAlpha(group_alpha); } if (initial_alpha != 1.0f && !m_bInGroup) { bitmap_device.MultiplyAlpha(initial_alpha); } transparency = m_Transparency; if (pPageObj->IsForm()) { transparency.SetGroup(); } CompositeDIBitmap(bitmap_device.GetBitmap(), rect.left, rect.top, /*mask_argb=*/0, /*alpha=*/1.0f, blend_type, transparency); return true; } FX_RECT CPDF_RenderStatus::GetClippedBBox(const FX_RECT& rect) const { FX_RECT bbox = rect; bbox.Intersect(m_pDevice->GetClipBox()); return bbox; } RetainPtr CPDF_RenderStatus::GetBackdrop( const CPDF_PageObject* pObj, const FX_RECT& bbox, bool bBackAlphaRequired) { int width = bbox.Width(); int height = bbox.Height(); auto backdrop = pdfium::MakeRetain(); if (bBackAlphaRequired && !m_bDropObjects) { // TODO(crbug.com/42271020): Consider adding support for // `FXDIB_Format::kBgraPremul` if (!backdrop->Create(width, height, FXDIB_Format::kBgra)) { return nullptr; } } else { if (!m_pDevice->CreateCompatibleBitmap(backdrop, width, height)) { return nullptr; } } const int cap_to_check = backdrop->IsAlphaFormat() ? FXRC_ALPHA_OUTPUT : FXRC_GET_BITS; if (m_pDevice->GetRenderCaps() & cap_to_check) { m_pDevice->GetDIBits(backdrop, bbox.left, bbox.top); return backdrop; } CFX_Matrix FinalMatrix = m_DeviceMatrix; FinalMatrix.Translate(-bbox.left, -bbox.top); if (!backdrop->IsAlphaFormat()) { backdrop->Clear(0xffffffff); } CFX_DefaultRenderDevice device; device.Attach(backdrop); m_pContext->Render(&device, pObj, &m_Options, &FinalMatrix); return backdrop; } std::unique_ptr CPDF_RenderStatus::CloneObjStates( const CPDF_GraphicStates* pSrcStates, bool stroke) { if (!pSrcStates) return nullptr; auto pStates = std::make_unique(*pSrcStates); const CPDF_Color* pObjColor = stroke ? pSrcStates->color_state().GetStrokeColor() : pSrcStates->color_state().GetFillColor(); if (!pObjColor->IsNull()) { pStates->mutable_color_state().SetFillColorRef( stroke ? pSrcStates->color_state().GetStrokeColorRef() : pSrcStates->color_state().GetFillColorRef()); pStates->mutable_color_state().SetStrokeColorRef( pStates->color_state().GetFillColorRef()); } return pStates; } bool CPDF_RenderStatus::ProcessText(CPDF_TextObject* textobj, const CFX_Matrix& mtObj2Device, CFX_Path* clipping_path) { if (textobj->GetCharCodes().empty()) return true; const TextRenderingMode text_render_mode = textobj->text_state().GetTextMode(); if (text_render_mode == TextRenderingMode::MODE_INVISIBLE) return true; RetainPtr pFont = textobj->text_state().GetFont(); if (pFont->IsType3Font()) return ProcessType3Text(textobj, mtObj2Device); bool is_fill = false; bool is_stroke = false; bool is_clip = false; if (clipping_path) { is_clip = true; } else { switch (text_render_mode) { case TextRenderingMode::MODE_FILL: case TextRenderingMode::MODE_FILL_CLIP: is_fill = true; break; case TextRenderingMode::MODE_STROKE: case TextRenderingMode::MODE_STROKE_CLIP: if (pFont->HasFace()) is_stroke = true; else is_fill = true; break; case TextRenderingMode::MODE_FILL_STROKE: case TextRenderingMode::MODE_FILL_STROKE_CLIP: is_fill = true; if (pFont->HasFace()) is_stroke = true; break; case TextRenderingMode::MODE_INVISIBLE: // Already handled above, but the compiler is not smart enough to // realize it. NOTREACHED_NORETURN(); case TextRenderingMode::MODE_CLIP: return true; case TextRenderingMode::MODE_UNKNOWN: NOTREACHED_NORETURN(); } } FX_ARGB stroke_argb = 0; FX_ARGB fill_argb = 0; bool bPattern = false; if (is_stroke) { if (textobj->color_state().GetStrokeColor()->IsPattern()) { bPattern = true; } else { stroke_argb = GetStrokeArgb(textobj); } } if (is_fill) { if (textobj->color_state().GetFillColor()->IsPattern()) { bPattern = true; } else { fill_argb = GetFillArgb(textobj); } } CFX_Matrix text_matrix = textobj->GetTextMatrix(); if (!IsAvailableMatrix(text_matrix)) return true; float font_size = textobj->text_state().GetFontSize(); if (bPattern) { DrawTextPathWithPattern(textobj, mtObj2Device, pFont.Get(), font_size, text_matrix, is_fill, is_stroke); return true; } if (is_clip || is_stroke) { const CFX_Matrix* pDeviceMatrix = &mtObj2Device; CFX_Matrix device_matrix; if (is_stroke) { pdfium::span pCTM = textobj->text_state().GetCTM(); if (pCTM[0] != 1.0f || pCTM[3] != 1.0f) { CFX_Matrix ctm(pCTM[0], pCTM[1], pCTM[2], pCTM[3], 0, 0); text_matrix *= ctm.GetInverse(); device_matrix = ctm * mtObj2Device; pDeviceMatrix = &device_matrix; } } return CPDF_TextRenderer::DrawTextPath( m_pDevice, textobj->GetCharCodes(), textobj->GetCharPositions(), pFont.Get(), font_size, text_matrix, pDeviceMatrix, textobj->graph_state().GetObject(), fill_argb, stroke_argb, clipping_path, GetFillOptionsForDrawTextPath(m_Options.GetOptions(), textobj, is_stroke, is_fill)); } text_matrix.Concat(mtObj2Device); return CPDF_TextRenderer::DrawNormalText( m_pDevice, textobj->GetCharCodes(), textobj->GetCharPositions(), pFont.Get(), font_size, text_matrix, fill_argb, m_Options); } // TODO(npm): Font fallback for type 3 fonts? (Completely separate code!!) bool CPDF_RenderStatus::ProcessType3Text(CPDF_TextObject* textobj, const CFX_Matrix& mtObj2Device) { CPDF_Type3Font* pType3Font = textobj->text_state().GetFont()->AsType3Font(); if (pdfium::Contains(m_Type3FontCache, pType3Font)) return true; FX_ARGB fill_argb = GetFillArgbForType3(textobj); int fill_alpha = FXARGB_A(fill_argb); #if BUILDFLAG(IS_WIN) if (IsPrint() && fill_alpha < 255) { return false; } #endif CFX_Matrix text_matrix = textobj->GetTextMatrix(); CFX_Matrix char_matrix = pType3Font->GetFontMatrix(); float font_size = textobj->text_state().GetFontSize(); char_matrix.Scale(font_size, font_size); // Must come before |glyphs|, because |glyphs| points into |refTypeCache|. std::set> refTypeCache; std::vector glyphs; if (!IsPrint()) { glyphs.resize(textobj->GetCharCodes().size()); } for (size_t iChar = 0; iChar < textobj->GetCharCodes().size(); ++iChar) { uint32_t charcode = textobj->GetCharCodes()[iChar]; if (charcode == static_cast(-1)) continue; CPDF_Type3Char* pType3Char = pType3Font->LoadChar(charcode); if (!pType3Char) continue; CFX_Matrix matrix = char_matrix; matrix.e += iChar > 0 ? textobj->GetCharPositions()[iChar - 1] : 0; matrix.Concat(text_matrix); matrix.Concat(mtObj2Device); if (!pType3Char->LoadBitmapFromSoleImageOfForm()) { if (!glyphs.empty()) { for (size_t i = 0; i < iChar; ++i) { const TextGlyphPos& glyph = glyphs[i]; if (!glyph.m_pGlyph) continue; std::optional point = glyph.GetOrigin({0, 0}); if (!point.has_value()) continue; m_pDevice->SetBitMask(glyph.m_pGlyph->GetBitmap(), point->x, point->y, fill_argb); } glyphs.clear(); } std::unique_ptr pStates = CloneObjStates(&textobj->graphic_states(), false); CPDF_RenderOptions options = m_Options; options.GetOptions().bForceHalftone = true; options.GetOptions().bRectAA = true; const auto* pForm = static_cast(pType3Char->form()); RetainPtr pFormResource = pForm->GetDict()->GetDictFor("Resources"); if (fill_alpha == 255) { CPDF_RenderStatus status(m_pContext, m_pDevice); status.SetOptions(options); status.SetTransparency(pForm->GetTransparency()); status.SetType3Char(pType3Char); status.SetFillColor(fill_argb); status.SetDropObjects(m_bDropObjects); status.SetFormResource(std::move(pFormResource)); status.Initialize(this, pStates.get()); status.m_Type3FontCache = m_Type3FontCache; status.m_Type3FontCache.emplace_back(pType3Font); CFX_RenderDevice::StateRestorer restorer(m_pDevice); status.RenderObjectList(pForm, matrix); } else { FX_RECT rect = matrix.TransformRect(pForm->CalcBoundingBox()).GetOuterRect(); if (!rect.Valid()) continue; CFX_DefaultRenderDevice bitmap_device; // TODO(crbug.com/42271020): Consider adding support for // `FXDIB_Format::kBgraPremul` if (!bitmap_device.Create(rect.Width(), rect.Height(), FXDIB_Format::kBgra)) { return true; } CPDF_RenderStatus status(m_pContext, &bitmap_device); status.SetOptions(options); status.SetTransparency(pForm->GetTransparency()); status.SetType3Char(pType3Char); status.SetFillColor(fill_argb); status.SetDropObjects(m_bDropObjects); status.SetFormResource(std::move(pFormResource)); status.Initialize(this, pStates.get()); status.m_Type3FontCache = m_Type3FontCache; status.m_Type3FontCache.emplace_back(pType3Font); matrix.Translate(-rect.left, -rect.top); status.RenderObjectList(pForm, matrix); m_pDevice->SetDIBits(bitmap_device.GetBitmap(), rect.left, rect.top); } } else if (pType3Char->GetBitmap()) { #if BUILDFLAG(IS_WIN) if (IsPrint()) { CFX_Matrix image_matrix = pType3Char->matrix() * matrix; CPDF_ImageRenderer renderer(this); if (renderer.Start(pType3Char->GetBitmap(), fill_argb, image_matrix, FXDIB_ResampleOptions(), false)) { renderer.Continue(nullptr); } if (!renderer.GetResult()) { return false; } continue; } #endif CPDF_Document* pDoc = pType3Font->GetDocument(); RetainPtr pCache = CPDF_DocRenderData::FromDocument(pDoc)->GetCachedType3(pType3Font); const CFX_GlyphBitmap* pBitmap = pCache->LoadGlyph(charcode, matrix); if (!pBitmap) { continue; } refTypeCache.insert(std::move(pCache)); CFX_Point origin(FXSYS_roundf(matrix.e), FXSYS_roundf(matrix.f)); if (glyphs.empty()) { FX_SAFE_INT32 left = origin.x; left += pBitmap->left(); if (!left.IsValid()) { continue; } FX_SAFE_INT32 top = origin.y; top -= pBitmap->top(); if (!top.IsValid()) { continue; } m_pDevice->SetBitMask(pBitmap->GetBitmap(), left.ValueOrDie(), top.ValueOrDie(), fill_argb); } else { glyphs[iChar].m_pGlyph = pBitmap; glyphs[iChar].m_Origin = origin; } } } if (glyphs.empty()) return true; FX_RECT rect = GetGlyphsBBox(glyphs, 0); auto bitmap = pdfium::MakeRetain(); if (!bitmap->Create(rect.Width(), rect.Height(), FXDIB_Format::k8bppMask)) { return true; } for (const TextGlyphPos& glyph : glyphs) { if (!glyph.m_pGlyph || !glyph.m_pGlyph->GetBitmap()->IsMaskFormat()) continue; std::optional point = glyph.GetOrigin({rect.left, rect.top}); if (!point.has_value()) continue; bitmap->CompositeMask( point->x, point->y, glyph.m_pGlyph->GetBitmap()->GetWidth(), glyph.m_pGlyph->GetBitmap()->GetHeight(), glyph.m_pGlyph->GetBitmap(), fill_argb, 0, 0, BlendMode::kNormal, nullptr, false); } m_pDevice->SetBitMask(std::move(bitmap), rect.left, rect.top, fill_argb); return true; } void CPDF_RenderStatus::DrawTextPathWithPattern(const CPDF_TextObject* textobj, const CFX_Matrix& mtObj2Device, CPDF_Font* pFont, float font_size, const CFX_Matrix& mtTextMatrix, bool fill, bool stroke) { if (!stroke) { std::vector> pCopy; pCopy.push_back(textobj->Clone()); CPDF_PathObject path; path.set_filltype(CFX_FillRenderOptions::FillType::kWinding); path.mutable_clip_path().CopyClipPath(m_LastClipPath); path.mutable_clip_path().AppendTexts(&pCopy); path.mutable_color_state() = textobj->color_state(); path.mutable_general_state() = textobj->general_state(); path.path().AppendFloatRect(textobj->GetRect()); path.SetRect(textobj->GetRect()); AutoRestorer> restorer2(&m_pCurObj); RenderSingleObject(&path, mtObj2Device); return; } std::vector char_pos_list = GetCharPosList( textobj->GetCharCodes(), textobj->GetCharPositions(), pFont, font_size); for (const TextCharPos& charpos : char_pos_list) { auto* font = charpos.m_FallbackFontPosition == -1 ? pFont->GetFont() : pFont->GetFontFallback(charpos.m_FallbackFontPosition); const CFX_Path* pPath = font->LoadGlyphPath(charpos.m_GlyphIndex, charpos.m_FontCharWidth); if (!pPath) continue; CPDF_PathObject path; path.mutable_graph_state() = textobj->graph_state(); path.mutable_color_state() = textobj->color_state(); CFX_Matrix matrix = charpos.GetEffectiveMatrix(CFX_Matrix( font_size, 0, 0, font_size, charpos.m_Origin.x, charpos.m_Origin.y)); matrix.Concat(mtTextMatrix); path.set_stroke(stroke); path.set_filltype(fill ? CFX_FillRenderOptions::FillType::kWinding : CFX_FillRenderOptions::FillType::kNoFill); path.path().Append(*pPath, &matrix); path.SetPathMatrix(CFX_Matrix()); ProcessPath(&path, mtObj2Device); } } void CPDF_RenderStatus::DrawShadingPattern(CPDF_ShadingPattern* pattern, const CPDF_PageObject* pPageObj, const CFX_Matrix& mtObj2Device, bool stroke) { if (!pattern->Load()) return; CFX_RenderDevice::StateRestorer restorer(m_pDevice); if (!ClipPattern(pPageObj, mtObj2Device, stroke)) return; FX_RECT rect = GetObjectClippedRect(pPageObj, mtObj2Device); if (rect.IsEmpty()) return; CFX_Matrix matrix = pattern->pattern_to_form() * mtObj2Device; int alpha = FXSYS_roundf(255 * (stroke ? pPageObj->general_state().GetStrokeAlpha() : pPageObj->general_state().GetFillAlpha())); CPDF_RenderShading::Draw(m_pDevice, m_pContext, m_pCurObj, pattern, matrix, rect, alpha, m_Options); } void CPDF_RenderStatus::ProcessShading(const CPDF_ShadingObject* pShadingObj, const CFX_Matrix& mtObj2Device) { FX_RECT rect = GetObjectClippedRect(pShadingObj, mtObj2Device); if (rect.IsEmpty()) return; CFX_Matrix matrix = pShadingObj->matrix() * mtObj2Device; CPDF_RenderShading::Draw( m_pDevice, m_pContext, m_pCurObj, pShadingObj->pattern(), matrix, rect, FXSYS_roundf(255 * pShadingObj->general_state().GetFillAlpha()), m_Options); } void CPDF_RenderStatus::DrawTilingPattern(CPDF_TilingPattern* pattern, CPDF_PageObject* pPageObj, const CFX_Matrix& mtObj2Device, bool stroke) { const std::unique_ptr pPatternForm = pattern->Load(pPageObj); if (!pPatternForm) return; CFX_RenderDevice::StateRestorer restorer(m_pDevice); if (!ClipPattern(pPageObj, mtObj2Device, stroke)) return; FX_RECT clip_box = m_pDevice->GetClipBox(); if (clip_box.IsEmpty()) return; RetainPtr screen = CPDF_RenderTiling::Draw(this, pPageObj, pattern, pPatternForm.get(), mtObj2Device, clip_box, stroke); if (!screen) { return; } constexpr FX_ARGB kMask = 0; CompositeDIBitmap(std::move(screen), clip_box.left, clip_box.top, kMask, /*alpha=*/1.0f, BlendMode::kNormal, CPDF_Transparency()); } void CPDF_RenderStatus::DrawPathWithPattern(CPDF_PathObject* path_obj, const CFX_Matrix& mtObj2Device, const CPDF_Color* pColor, bool stroke) { RetainPtr pattern = pColor->GetPattern(); if (!pattern) return; if (CPDF_TilingPattern* pTilingPattern = pattern->AsTilingPattern()) DrawTilingPattern(pTilingPattern, path_obj, mtObj2Device, stroke); else if (CPDF_ShadingPattern* pShadingPattern = pattern->AsShadingPattern()) DrawShadingPattern(pShadingPattern, path_obj, mtObj2Device, stroke); } void CPDF_RenderStatus::ProcessPathPattern( CPDF_PathObject* path_obj, const CFX_Matrix& mtObj2Device, CFX_FillRenderOptions::FillType* fill_type, bool* stroke) { DCHECK(fill_type); DCHECK(stroke); if (*fill_type != CFX_FillRenderOptions::FillType::kNoFill) { const CPDF_Color& FillColor = *path_obj->color_state().GetFillColor(); if (FillColor.IsPattern()) { DrawPathWithPattern(path_obj, mtObj2Device, &FillColor, false); *fill_type = CFX_FillRenderOptions::FillType::kNoFill; } } if (*stroke) { const CPDF_Color& StrokeColor = *path_obj->color_state().GetStrokeColor(); if (StrokeColor.IsPattern()) { DrawPathWithPattern(path_obj, mtObj2Device, &StrokeColor, true); *stroke = false; } } } bool CPDF_RenderStatus::ProcessImage(CPDF_ImageObject* pImageObj, const CFX_Matrix& mtObj2Device) { CPDF_ImageRenderer render(this); if (render.Start(pImageObj, mtObj2Device, m_bStdCS)) { render.Continue(nullptr); } return render.GetResult(); } void CPDF_RenderStatus::CompositeDIBitmap( RetainPtr bitmap, int left, int top, FX_ARGB mask_argb, float alpha, BlendMode blend_mode, const CPDF_Transparency& transparency) { CHECK(bitmap); if (blend_mode == BlendMode::kNormal) { if (bitmap->IsMaskFormat()) { #if BUILDFLAG(IS_WIN) FX_ARGB fill_argb = m_Options.TranslateColor(mask_argb); if (alpha != 1.0f) { auto& bgra = reinterpret_cast&>(fill_argb); bgra.alpha *= FXSYS_roundf(alpha * 255) / 255; } if (m_pDevice->SetBitMask(bitmap, left, top, fill_argb)) { return; } #else NOTREACHED_NORETURN(); #endif } else { if (alpha != 1.0f) { if (CFX_DefaultRenderDevice::UseSkiaRenderer()) { CFX_Matrix matrix = CFX_RenderDevice::GetFlipMatrix( bitmap->GetWidth(), bitmap->GetHeight(), left, top); m_pDevice->StartDIBits(std::move(bitmap), alpha, /*argb=*/0, matrix, FXDIB_ResampleOptions()); return; } bitmap->MultiplyAlpha(alpha); } if (m_pDevice->SetDIBits(bitmap, left, top)) { return; } } } bool bIsolated = transparency.IsIsolated(); bool bBackAlphaRequired = blend_mode != BlendMode::kNormal && bIsolated && !m_bDropObjects; bool bGetBackGround = ((m_pDevice->GetRenderCaps() & FXRC_ALPHA_OUTPUT)) || (!(m_pDevice->GetRenderCaps() & FXRC_ALPHA_OUTPUT) && (m_pDevice->GetRenderCaps() & FXRC_GET_BITS) && !bBackAlphaRequired); if (bGetBackGround) { if (bIsolated || !transparency.IsGroup()) { if (!bitmap->IsMaskFormat()) { m_pDevice->SetDIBitsWithBlend(std::move(bitmap), left, top, blend_mode); } return; } FX_RECT rect(left, top, left + bitmap->GetWidth(), top + bitmap->GetHeight()); rect.Intersect(m_pDevice->GetClipBox()); RetainPtr pClone; if (m_pDevice->GetBackDrop() && m_pDevice->GetBitmap()) { pClone = m_pDevice->GetBackDrop()->ClipTo(rect); if (!pClone) return; pClone->CompositeBitmap(0, 0, pClone->GetWidth(), pClone->GetHeight(), m_pDevice->GetBitmap(), rect.left, rect.top, BlendMode::kNormal, nullptr, false); left = std::min(left, 0); top = std::min(top, 0); if (bitmap->IsMaskFormat()) { #if BUILDFLAG(IS_WIN) pClone->CompositeMask(0, 0, pClone->GetWidth(), pClone->GetHeight(), bitmap, mask_argb, left, top, blend_mode, nullptr, false); #else NOTREACHED_NORETURN(); #endif } else { pClone->CompositeBitmap(0, 0, pClone->GetWidth(), pClone->GetHeight(), bitmap, left, top, blend_mode, nullptr, false); } } else { pClone = bitmap; } if (m_pDevice->GetBackDrop()) { m_pDevice->SetDIBits(pClone, rect.left, rect.top); } else { if (!bitmap->IsMaskFormat()) { m_pDevice->SetDIBitsWithBlend(std::move(bitmap), rect.left, rect.top, blend_mode); } } return; } FX_RECT bbox = GetClippedBBox( FX_RECT(left, top, left + bitmap->GetWidth(), top + bitmap->GetHeight())); RetainPtr backdrop = GetBackdrop( m_pCurObj, bbox, blend_mode != BlendMode::kNormal && bIsolated); if (!backdrop) { return; } const int width = bitmap->GetWidth(); const int height = bitmap->GetHeight(); if (bitmap->IsMaskFormat()) { #if BUILDFLAG(IS_WIN) backdrop->CompositeMask(left - bbox.left, top - bbox.top, width, height, std::move(bitmap), mask_argb, 0, 0, blend_mode, nullptr, false); #else NOTREACHED_NORETURN(); #endif } else { backdrop->CompositeBitmap(left - bbox.left, top - bbox.top, width, height, std::move(bitmap), 0, 0, blend_mode, nullptr, false); } auto new_backdrop = pdfium::MakeRetain(); CHECK(new_backdrop->Create(backdrop->GetWidth(), backdrop->GetHeight(), FXDIB_Format::kBgrx)); new_backdrop->Clear(0xffffffff); new_backdrop->CompositeBitmap(0, 0, new_backdrop->GetWidth(), new_backdrop->GetHeight(), std::move(backdrop), 0, 0, BlendMode::kNormal, nullptr, false); m_pDevice->SetDIBits(std::move(new_backdrop), bbox.left, bbox.top); } RetainPtr CPDF_RenderStatus::LoadSMask( CPDF_Dictionary* smask_dict, const FX_RECT& clip_rect, const CFX_Matrix& smask_matrix) { RetainPtr pGroup = smask_dict->GetMutableStreamFor(pdfium::transparency::kG); if (!pGroup) return nullptr; std::unique_ptr pFunc; RetainPtr pFuncObj = smask_dict->GetDirectObjectFor(pdfium::transparency::kTR); if (pFuncObj && (pFuncObj->IsDictionary() || pFuncObj->IsStream())) pFunc = CPDF_Function::Load(std::move(pFuncObj)); CFX_Matrix matrix = smask_matrix; matrix.Translate(-clip_rect.left, -clip_rect.top); CPDF_Form form(m_pContext->GetDocument(), m_pContext->GetMutablePageResources(), pGroup); form.ParseContent(); CFX_DefaultRenderDevice bitmap_device; bool bLuminosity = smask_dict->GetByteStringFor(pdfium::transparency::kSoftMaskSubType) != pdfium::transparency::kAlpha; const int width = clip_rect.Width(); const int height = clip_rect.Height(); const FXDIB_Format format = GetFormatForLuminosity(bLuminosity); if (!bitmap_device.Create(width, height, format)) { return nullptr; } CPDF_ColorSpace::Family nCSFamily = CPDF_ColorSpace::Family::kUnknown; const FX_ARGB background_color = bLuminosity ? GetBackgroundColor(smask_dict, pGroup->GetDict().Get(), &nCSFamily) : 0; bitmap_device.Clear(background_color); RetainPtr pFormResource = form.GetDict()->GetDictFor("Resources"); CPDF_RenderOptions options; options.SetColorMode(bLuminosity ? CPDF_RenderOptions::kNormal : CPDF_RenderOptions::kAlpha); CPDF_RenderStatus status(m_pContext, &bitmap_device); status.SetOptions(options); status.SetGroupFamily(nCSFamily); status.SetLoadMask(bLuminosity); status.SetStdCS(true); status.SetFormResource(std::move(pFormResource)); status.SetDropObjects(m_bDropObjects); status.Initialize(nullptr, nullptr); status.RenderObjectList(&form, matrix); auto result_mask = pdfium::MakeRetain(); if (!result_mask->Create(width, height, FXDIB_Format::k8bppMask)) { return nullptr; } pdfium::span dest_buf = result_mask->GetWritableBuffer(); RetainPtr bitmap = bitmap_device.GetBitmap(); pdfium::span src_buf = bitmap->GetBuffer(); const int dest_pitch = result_mask->GetPitch(); const int src_pitch = bitmap->GetPitch(); DataVector transfers(256); if (pFunc) { std::vector results(pFunc->OutputCount()); for (size_t i = 0; i < transfers.size(); ++i) { float input = i / 255.0f; pFunc->Call(pdfium::span_from_ref(input), results); transfers[i] = FXSYS_roundf(results[0] * 255); } } else { // Fill |transfers| with 0, 1, ... N. std::iota(transfers.begin(), transfers.end(), 0); } if (bLuminosity) { const int bytes_per_pixel = bitmap->GetBPP() / 8; for (int row = 0; row < height; row++) { const size_t dest_offset = Fx2DSizeOrDie(row, dest_pitch); const size_t src_offset = Fx2DSizeOrDie(row, src_pitch); uint8_t* dest_pos = dest_buf.subspan(dest_offset).data(); const uint8_t* src_pos = src_buf.subspan(src_offset).data(); for (int col = 0; col < width; col++) { UNSAFE_TODO({ *dest_pos++ = transfers[FXRGB2GRAY(src_pos[2], src_pos[1], *src_pos)]; src_pos += bytes_per_pixel; }); } } } else if (pFunc) { int size = dest_pitch * height; for (int i = 0; i < size; i++) { dest_buf[i] = transfers[src_buf[i]]; } } else { fxcrt::Copy(src_buf.first(dest_pitch * height), dest_buf); } return result_mask; } FX_ARGB CPDF_RenderStatus::GetBackgroundColor( const CPDF_Dictionary* pSMaskDict, const CPDF_Dictionary* pGroupDict, CPDF_ColorSpace::Family* pCSFamily) { static constexpr FX_ARGB kDefaultColor = ArgbEncode(255, 0, 0, 0); RetainPtr pBC = pSMaskDict->GetArrayFor(pdfium::transparency::kBC); if (!pBC) return kDefaultColor; RetainPtr pCSObj; RetainPtr pGroup = pGroupDict ? pGroupDict->GetDictFor("Group") : nullptr; if (pGroup) pCSObj = pGroup->GetDirectObjectFor(pdfium::transparency::kCS); RetainPtr pCS = CPDF_DocPageData::FromDocument(m_pContext->GetDocument()) ->GetColorSpace(pCSObj.Get(), nullptr); if (!pCS) return kDefaultColor; CPDF_ColorSpace::Family family = pCS->GetFamily(); if (family == CPDF_ColorSpace::Family::kLab || pCS->IsSpecial() || (family == CPDF_ColorSpace::Family::kICCBased && !pCS->IsNormal())) { return kDefaultColor; } // Store Color Space Family to use in CPDF_RenderStatus::Initialize(). *pCSFamily = family; uint32_t comps = std::max(8u, pCS->ComponentCount()); size_t count = std::min(8, pBC->size()); std::vector floats = ReadArrayElementsToVector(pBC.Get(), count); floats.resize(comps); auto rgb = pCS->GetRGBOrZerosOnError(floats); return ArgbEncode(255, static_cast(rgb.red * 255), static_cast(rgb.green * 255), static_cast(rgb.blue * 255)); } FXDIB_Format CPDF_RenderStatus::GetCompatibleArgbFormat() const { #if defined(PDF_USE_SKIA) if (m_pDevice->GetDeviceCaps(FXDC_RENDER_CAPS) & FXRC_PREMULTIPLIED_ALPHA) { return FXDIB_Format::kBgraPremul; } #endif return FXDIB_Format::kBgra; }