// 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_scaledrenderbuffer.h" #include "core/fpdfapi/parser/cpdf_dictionary.h" #include "core/fpdfapi/render/cpdf_devicebuffer.h" #include "core/fpdfapi/render/cpdf_rendercontext.h" #include "core/fxge/cfx_defaultrenderdevice.h" #include "core/fxge/dib/cfx_dibitmap.h" namespace { constexpr size_t kImageSizeLimitBytes = 30 * 1024 * 1024; } // namespace CPDF_ScaledRenderBuffer::CPDF_ScaledRenderBuffer() = default; CPDF_ScaledRenderBuffer::~CPDF_ScaledRenderBuffer() = default; bool CPDF_ScaledRenderBuffer::Initialize(CPDF_RenderContext* pContext, CFX_RenderDevice* pDevice, const FX_RECT& rect, const CPDF_PageObject* pObj, const CPDF_RenderOptions* pOptions, int max_dpi) { m_pDevice = pDevice; if (m_pDevice->GetDeviceCaps(FXDC_RENDER_CAPS) & FXRC_GET_BITS) return true; m_Rect = rect; m_Matrix = CPDF_DeviceBuffer::CalculateMatrix(pDevice, rect, max_dpi, /*scale=*/true); m_pBitmapDevice = std::make_unique(); bool bIsAlpha = !!(m_pDevice->GetDeviceCaps(FXDC_RENDER_CAPS) & FXRC_ALPHA_OUTPUT); FXDIB_Format dibFormat = bIsAlpha ? FXDIB_Format::kArgb : FXDIB_Format::kRgb; while (true) { FX_RECT bitmap_rect = m_Matrix.TransformRect(CFX_FloatRect(rect)).GetOuterRect(); int32_t width = bitmap_rect.Width(); int32_t height = bitmap_rect.Height(); // Set to 0 to make CalculatePitchAndSize() calculate it. constexpr uint32_t kNoPitch = 0; absl::optional pitch_size = CFX_DIBitmap::CalculatePitchAndSize(width, height, dibFormat, kNoPitch); if (!pitch_size.has_value()) return false; if (pitch_size.value().size <= kImageSizeLimitBytes && m_pBitmapDevice->Create(width, height, dibFormat, nullptr)) { break; } m_Matrix.Scale(0.5f, 0.5f); } pContext->GetBackground(m_pBitmapDevice->GetBitmap(), pObj, pOptions, m_Matrix); return true; } CFX_RenderDevice* CPDF_ScaledRenderBuffer::GetDevice() const { return m_pBitmapDevice ? static_cast(m_pBitmapDevice.get()) : m_pDevice.get(); } void CPDF_ScaledRenderBuffer::OutputToDevice() { if (m_pBitmapDevice) { m_pDevice->StretchDIBits(m_pBitmapDevice->GetBitmap(), m_Rect.left, m_Rect.top, m_Rect.Width(), m_Rect.Height()); } }