blob: 5c5d8a7a79b93cca11341d1817fa2b9cf7b7cd52 [file] [log] [blame]
// 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 "xfa/fgas/graphics/cfgas_gegraphics.h"
#include <cmath>
#include <memory>
#include "core/fxge/cfx_defaultrenderdevice.h"
#include "core/fxge/cfx_renderdevice.h"
#include "core/fxge/cfx_unicodeencoding.h"
#include "core/fxge/dib/cfx_dibitmap.h"
#include "third_party/base/check.h"
#include "third_party/base/stl_util.h"
#include "xfa/fgas/graphics/cfgas_gecolor.h"
#include "xfa/fgas/graphics/cfgas_gepath.h"
#include "xfa/fgas/graphics/cfgas_gepattern.h"
#include "xfa/fgas/graphics/cfgas_geshading.h"
namespace {
struct FX_HATCHDATA {
int32_t width;
int32_t height;
uint8_t maskBits[64];
};
const FX_HATCHDATA kHatchBitmapData[] = {
{16, // Horizontal
16,
{
0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff,
0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
}},
{16, // Vertical
16,
{
0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00,
0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80,
0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x80,
0x80, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00,
0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00,
0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00,
}},
{16, // ForwardDiagonal
16,
{
0x80, 0x80, 0x00, 0x00, 0x40, 0x40, 0x00, 0x00, 0x20, 0x20, 0x00,
0x00, 0x10, 0x10, 0x00, 0x00, 0x08, 0x08, 0x00, 0x00, 0x04, 0x04,
0x00, 0x00, 0x02, 0x02, 0x00, 0x00, 0x01, 0x01, 0x00, 0x00, 0x80,
0x80, 0x00, 0x00, 0x40, 0x40, 0x00, 0x00, 0x20, 0x20, 0x00, 0x00,
0x10, 0x10, 0x00, 0x00, 0x08, 0x08, 0x00, 0x00, 0x04, 0x04, 0x00,
0x00, 0x02, 0x02, 0x00, 0x00, 0x01, 0x01, 0x00, 0x00,
}},
{16, // BackwardDiagonal
16,
{
0x01, 0x01, 0x00, 0x00, 0x02, 0x02, 0x00, 0x00, 0x04, 0x04, 0x00,
0x00, 0x08, 0x08, 0x00, 0x00, 0x10, 0x10, 0x00, 0x00, 0x20, 0x20,
0x00, 0x00, 0x40, 0x40, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x01,
0x01, 0x00, 0x00, 0x02, 0x02, 0x00, 0x00, 0x04, 0x04, 0x00, 0x00,
0x08, 0x08, 0x00, 0x00, 0x10, 0x10, 0x00, 0x00, 0x20, 0x20, 0x00,
0x00, 0x40, 0x40, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00,
}},
{16, // Cross
16,
{
0xff, 0xff, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00,
0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80,
0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0xff,
0xff, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00,
0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00,
0x00, 0x80, 0x80, 0x00, 0x00, 0x80, 0x80, 0x00, 0x00,
}},
{16, // DiagonalCross
16,
{
0x81, 0x81, 0x00, 0x00, 0x42, 0x42, 0x00, 0x00, 0x24, 0x24, 0x00,
0x00, 0x18, 0x18, 0x00, 0x00, 0x18, 0x18, 0x00, 0x00, 0x24, 0x24,
0x00, 0x00, 0x42, 0x42, 0x00, 0x00, 0x81, 0x81, 0x00, 0x00, 0x81,
0x81, 0x00, 0x00, 0x42, 0x42, 0x00, 0x00, 0x24, 0x24, 0x00, 0x00,
0x18, 0x18, 0x00, 0x00, 0x18, 0x18, 0x00, 0x00, 0x24, 0x24, 0x00,
0x00, 0x42, 0x42, 0x00, 0x00, 0x81, 0x81, 0x00, 0x00,
}},
};
const FX_HATCHDATA kHatchPlaceHolder = {
0,
0,
{
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
}};
const FX_HATCHDATA& GetHatchBitmapData(size_t index) {
return index < pdfium::size(kHatchBitmapData) ? kHatchBitmapData[index]
: kHatchPlaceHolder;
}
} // namespace
CFGAS_GEGraphics::CFGAS_GEGraphics(CFX_RenderDevice* renderDevice)
: m_renderDevice(renderDevice) {
DCHECK(m_renderDevice);
}
CFGAS_GEGraphics::~CFGAS_GEGraphics() = default;
void CFGAS_GEGraphics::SaveGraphState() {
m_renderDevice->SaveState();
m_infoStack.push_back(std::make_unique<TInfo>(m_info));
}
void CFGAS_GEGraphics::RestoreGraphState() {
m_renderDevice->RestoreState(false);
if (m_infoStack.empty())
return;
m_info = *m_infoStack.back();
m_infoStack.pop_back();
return;
}
void CFGAS_GEGraphics::SetLineCap(CFX_GraphStateData::LineCap lineCap) {
m_info.graphState.m_LineCap = lineCap;
}
void CFGAS_GEGraphics::SetLineDash(float dashPhase,
pdfium::span<const float> dashArray) {
DCHECK(!dashArray.empty());
float scale = m_info.isActOnDash ? m_info.graphState.m_LineWidth : 1.0;
m_info.graphState.m_DashPhase = dashPhase;
m_info.graphState.m_DashArray.resize(dashArray.size());
for (size_t i = 0; i < dashArray.size(); ++i)
m_info.graphState.m_DashArray[i] = dashArray[i] * scale;
}
void CFGAS_GEGraphics::SetSolidLineDash() {
m_info.graphState.m_DashArray.clear();
}
void CFGAS_GEGraphics::SetLineWidth(float lineWidth) {
m_info.graphState.m_LineWidth = lineWidth;
}
void CFGAS_GEGraphics::EnableActOnDash() {
m_info.isActOnDash = true;
}
void CFGAS_GEGraphics::SetStrokeColor(const CFGAS_GEColor& color) {
m_info.strokeColor = color;
}
void CFGAS_GEGraphics::SetFillColor(const CFGAS_GEColor& color) {
m_info.fillColor = color;
}
void CFGAS_GEGraphics::StrokePath(CFGAS_GEPath* path,
const CFX_Matrix* matrix) {
if (path)
RenderDeviceStrokePath(path, matrix);
}
void CFGAS_GEGraphics::FillPath(CFGAS_GEPath* path,
CFX_FillRenderOptions::FillType fill_type,
const CFX_Matrix* matrix) {
if (path)
RenderDeviceFillPath(path, fill_type, matrix);
}
void CFGAS_GEGraphics::ConcatMatrix(const CFX_Matrix* matrix) {
if (matrix)
m_info.CTM.Concat(*matrix);
}
const CFX_Matrix* CFGAS_GEGraphics::GetMatrix() const {
return &m_info.CTM;
}
CFX_RectF CFGAS_GEGraphics::GetClipRect() const {
FX_RECT r = m_renderDevice->GetClipBox();
return CFX_RectF(r.left, r.top, r.Width(), r.Height());
}
void CFGAS_GEGraphics::SetClipRect(const CFX_RectF& rect) {
m_renderDevice->SetClip_Rect(
FX_RECT(FXSYS_roundf(rect.left), FXSYS_roundf(rect.top),
FXSYS_roundf(rect.right()), FXSYS_roundf(rect.bottom())));
}
CFX_RenderDevice* CFGAS_GEGraphics::GetRenderDevice() {
return m_renderDevice;
}
void CFGAS_GEGraphics::RenderDeviceStrokePath(const CFGAS_GEPath* path,
const CFX_Matrix* matrix) {
if (m_info.strokeColor.GetType() != CFGAS_GEColor::Solid)
return;
CFX_Matrix m = m_info.CTM;
if (matrix)
m.Concat(*matrix);
m_renderDevice->DrawPath(path->GetPathData(), &m, &m_info.graphState, 0x0,
m_info.strokeColor.GetArgb(),
CFX_FillRenderOptions());
}
void CFGAS_GEGraphics::RenderDeviceFillPath(
const CFGAS_GEPath* path,
CFX_FillRenderOptions::FillType fill_type,
const CFX_Matrix* matrix) {
CFX_Matrix m = m_info.CTM;
if (matrix)
m.Concat(*matrix);
const CFX_FillRenderOptions fill_options(fill_type);
switch (m_info.fillColor.GetType()) {
case CFGAS_GEColor::Solid:
m_renderDevice->DrawPath(path->GetPathData(), &m, &m_info.graphState,
m_info.fillColor.GetArgb(), 0x0, fill_options);
return;
case CFGAS_GEColor::Pattern:
FillPathWithPattern(path, fill_options, m);
return;
case CFGAS_GEColor::Shading:
FillPathWithShading(path, fill_options, m);
return;
default:
return;
}
}
void CFGAS_GEGraphics::FillPathWithPattern(
const CFGAS_GEPath* path,
const CFX_FillRenderOptions& fill_options,
const CFX_Matrix& matrix) {
RetainPtr<CFX_DIBitmap> bitmap = m_renderDevice->GetBitmap();
int32_t width = bitmap->GetWidth();
int32_t height = bitmap->GetHeight();
auto bmp = pdfium::MakeRetain<CFX_DIBitmap>();
bmp->Create(width, height, FXDIB_Format::kArgb);
m_renderDevice->GetDIBits(bmp, 0, 0);
FX_HatchStyle hatchStyle = m_info.fillColor.GetPattern()->m_hatchStyle;
const FX_HATCHDATA& data =
GetHatchBitmapData(static_cast<size_t>(hatchStyle));
auto mask = pdfium::MakeRetain<CFX_DIBitmap>();
mask->Create(data.width, data.height, FXDIB_Format::k1bppMask);
memcpy(mask->GetBuffer(), data.maskBits, mask->GetPitch() * data.height);
const CFX_FloatRect rectf =
matrix.TransformRect(path->GetPathData()->GetBoundingBox());
const FX_RECT rect = rectf.ToRoundedFxRect();
CFX_DefaultRenderDevice device;
device.Attach(bmp, false, nullptr, false);
device.FillRect(rect, m_info.fillColor.GetPattern()->m_backArgb);
for (int32_t j = rect.bottom; j < rect.top; j += mask->GetHeight()) {
for (int32_t i = rect.left; i < rect.right; i += mask->GetWidth())
device.SetBitMask(mask, i, j, m_info.fillColor.GetPattern()->m_foreArgb);
}
CFX_RenderDevice::StateRestorer restorer(m_renderDevice);
m_renderDevice->SetClip_PathFill(path->GetPathData(), &matrix, fill_options);
SetDIBitsWithMatrix(bmp, CFX_Matrix());
}
void CFGAS_GEGraphics::FillPathWithShading(
const CFGAS_GEPath* path,
const CFX_FillRenderOptions& fill_options,
const CFX_Matrix& matrix) {
RetainPtr<CFX_DIBitmap> bitmap = m_renderDevice->GetBitmap();
int32_t width = bitmap->GetWidth();
int32_t height = bitmap->GetHeight();
float start_x = m_info.fillColor.GetShading()->m_beginPoint.x;
float start_y = m_info.fillColor.GetShading()->m_beginPoint.y;
float end_x = m_info.fillColor.GetShading()->m_endPoint.x;
float end_y = m_info.fillColor.GetShading()->m_endPoint.y;
auto bmp = pdfium::MakeRetain<CFX_DIBitmap>();
bmp->Create(width, height, FXDIB_Format::kArgb);
m_renderDevice->GetDIBits(bmp, 0, 0);
int32_t pitch = bmp->GetPitch();
bool result = false;
switch (m_info.fillColor.GetShading()->m_type) {
case FX_SHADING_Axial: {
float x_span = end_x - start_x;
float y_span = end_y - start_y;
float axis_len_square = (x_span * x_span) + (y_span * y_span);
for (int32_t row = 0; row < height; row++) {
uint32_t* dib_buf =
reinterpret_cast<uint32_t*>(bmp->GetBuffer() + row * pitch);
for (int32_t column = 0; column < width; column++) {
float scale = 0.0f;
if (axis_len_square) {
float y = static_cast<float>(row);
float x = static_cast<float>(column);
scale = (((x - start_x) * x_span) + ((y - start_y) * y_span)) /
axis_len_square;
if (std::isnan(scale) || scale < 0.0f) {
if (!m_info.fillColor.GetShading()->m_isExtendedBegin)
continue;
scale = 0.0f;
} else if (scale > 1.0f) {
if (!m_info.fillColor.GetShading()->m_isExtendedEnd)
continue;
scale = 1.0f;
}
}
int32_t index = static_cast<int32_t>(scale * (FX_SHADING_Steps - 1));
dib_buf[column] = m_info.fillColor.GetShading()->m_argbArray[index];
}
}
result = true;
break;
}
case FX_SHADING_Radial: {
float start_r = m_info.fillColor.GetShading()->m_beginRadius;
float end_r = m_info.fillColor.GetShading()->m_endRadius;
float a = ((start_x - end_x) * (start_x - end_x)) +
((start_y - end_y) * (start_y - end_y)) -
((start_r - end_r) * (start_r - end_r));
for (int32_t row = 0; row < height; row++) {
uint32_t* dib_buf = (uint32_t*)(bmp->GetBuffer() + row * pitch);
for (int32_t column = 0; column < width; column++) {
float x = (float)(column);
float y = (float)(row);
float b = -2 * (((x - start_x) * (end_x - start_x)) +
((y - start_y) * (end_y - start_y)) +
(start_r * (end_r - start_r)));
float c = ((x - start_x) * (x - start_x)) +
((y - start_y) * (y - start_y)) - (start_r * start_r);
float s;
if (a == 0) {
s = -c / b;
} else {
float b2_4ac = (b * b) - 4 * (a * c);
if (b2_4ac < 0) {
continue;
}
float root = (sqrt(b2_4ac));
float s1, s2;
if (a > 0) {
s1 = (-b - root) / (2 * a);
s2 = (-b + root) / (2 * a);
} else {
s2 = (-b - root) / (2 * a);
s1 = (-b + root) / (2 * a);
}
if (s2 <= 1.0f || m_info.fillColor.GetShading()->m_isExtendedEnd) {
s = (s2);
} else {
s = (s1);
}
if ((start_r) + s * (end_r - start_r) < 0) {
continue;
}
}
if (std::isnan(s) || s < 0.0f) {
if (!m_info.fillColor.GetShading()->m_isExtendedBegin)
continue;
s = 0.0f;
}
if (s > 1.0f) {
if (!m_info.fillColor.GetShading()->m_isExtendedEnd)
continue;
s = 1.0f;
}
int index = static_cast<int32_t>(s * (FX_SHADING_Steps - 1));
dib_buf[column] = m_info.fillColor.GetShading()->m_argbArray[index];
}
}
result = true;
break;
}
default: {
result = false;
break;
}
}
if (result) {
CFX_RenderDevice::StateRestorer restorer(m_renderDevice);
m_renderDevice->SetClip_PathFill(path->GetPathData(), &matrix,
fill_options);
SetDIBitsWithMatrix(bmp, matrix);
}
}
void CFGAS_GEGraphics::SetDIBitsWithMatrix(const RetainPtr<CFX_DIBBase>& source,
const CFX_Matrix& matrix) {
if (matrix.IsIdentity()) {
m_renderDevice->SetDIBits(source, 0, 0);
} else {
CFX_Matrix m((float)source->GetWidth(), 0, 0, (float)source->GetHeight(), 0,
0);
m.Concat(matrix);
int32_t left;
int32_t top;
RetainPtr<CFX_DIBitmap> bmp1 = source->FlipImage(false, true);
RetainPtr<CFX_DIBitmap> bmp2 = bmp1->TransformTo(m, &left, &top);
m_renderDevice->SetDIBits(bmp2, left, top);
}
}
CFGAS_GEGraphics::TInfo::TInfo() = default;
CFGAS_GEGraphics::TInfo::TInfo(const TInfo& info)
: graphState(info.graphState),
CTM(info.CTM),
isActOnDash(info.isActOnDash),
strokeColor(info.strokeColor),
fillColor(info.fillColor) {}
CFGAS_GEGraphics::TInfo& CFGAS_GEGraphics::TInfo::operator=(
const TInfo& other) {
graphState = other.graphState;
CTM = other.CTM;
isActOnDash = other.isActOnDash;
strokeColor = other.strokeColor;
fillColor = other.fillColor;
return *this;
}