| // Copyright 2016 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/render/cpdf_scaledrenderbuffer.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<CFX_DefaultRenderDevice>(); |
| 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<CFX_DIBitmap::PitchAndSize> 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 ? 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()); |
| } |
| } |