// 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.
|
|
#include "core/fxge/include/fx_ge.h"
|
#include "core/fxge/skia/fx_skia_device.h"
|
#include "fpdfsdk/include/fsdk_define.h"
|
#include "public/fpdfview.h"
|
#include "testing/fx_string_testhelpers.h"
|
#include "testing/gtest/include/gtest/gtest.h"
|
#include "third_party/skia/include/core/SkPictureRecorder.h"
|
|
namespace {
|
|
struct State {
|
enum class Change { kNo, kYes };
|
enum class Save { kNo, kYes };
|
enum class Clip { kNo, kSame, kDifferentPath, kDifferentMatrix };
|
enum class Graphic { kNone, kPath, kText };
|
|
Change m_change;
|
Save m_save;
|
Clip m_clip;
|
Graphic m_graphic;
|
uint32_t m_pixel;
|
};
|
|
void EmptyTest(CFX_SkiaDeviceDriver* driver, const State&) {
|
driver->SaveState();
|
driver->RestoreState(true);
|
driver->RestoreState(false);
|
}
|
|
void CommonTest(CFX_SkiaDeviceDriver* driver, const State& state) {
|
FXTEXT_CHARPOS charPos[] = {{1, 0, 1, 4, false, {0, 0, 0, 0}, false}};
|
CFX_Font font;
|
FX_FLOAT fontSize = 1;
|
CFX_FontCache cache;
|
CFX_PathData clipPath, clipPath2;
|
clipPath.AppendRect(0, 0, 3, 1);
|
clipPath2.AppendRect(0, 0, 2, 1);
|
CFX_Matrix clipMatrix;
|
CFX_Matrix clipMatrix2(1, 0, 0, 1, 0, 1);
|
driver->SaveState();
|
CFX_PathData path1;
|
path1.AppendRect(0, 0, 1, 2);
|
CFX_Matrix matrix, matrix2;
|
matrix2.Translate(1, 0);
|
CFX_GraphStateData graphState;
|
if (state.m_save == State::Save::kYes)
|
driver->SaveState();
|
if (state.m_clip != State::Clip::kNo)
|
driver->SetClip_PathFill(&clipPath, &clipMatrix, 0);
|
if (state.m_graphic == State::Graphic::kPath) {
|
driver->DrawPath(&path1, &matrix, &graphState, 0xFF112233, 0,
|
FXFILL_WINDING, 0);
|
} else if (state.m_graphic == State::Graphic::kText) {
|
driver->DrawDeviceText(SK_ARRAY_COUNT(charPos), charPos, &font, &cache,
|
&matrix, fontSize, 0xFF445566);
|
}
|
if (state.m_save == State::Save::kYes)
|
driver->RestoreState(true);
|
CFX_PathData path2;
|
path2.AppendRect(0, 0, 2, 2);
|
if (state.m_change == State::Change::kYes) {
|
if (state.m_graphic == State::Graphic::kPath)
|
graphState.m_LineCap = CFX_GraphStateData::LineCapRound;
|
else if (state.m_graphic == State::Graphic::kText)
|
fontSize = 2;
|
}
|
if (state.m_clip == State::Clip::kSame)
|
driver->SetClip_PathFill(&clipPath, &clipMatrix, 0);
|
else if (state.m_clip == State::Clip::kDifferentPath)
|
driver->SetClip_PathFill(&clipPath2, &clipMatrix, 0);
|
else if (state.m_clip == State::Clip::kDifferentMatrix)
|
driver->SetClip_PathFill(&clipPath, &clipMatrix2, 0);
|
if (state.m_graphic == State::Graphic::kPath) {
|
driver->DrawPath(&path2, &matrix2, &graphState, 0xFF112233, 0,
|
FXFILL_WINDING, 0);
|
} else if (state.m_graphic == State::Graphic::kText) {
|
driver->DrawDeviceText(SK_ARRAY_COUNT(charPos), charPos, &font, &cache,
|
&matrix2, fontSize, 0xFF445566);
|
}
|
if (state.m_save == State::Save::kYes)
|
driver->RestoreState(false);
|
driver->RestoreState(false);
|
}
|
|
void OutOfSequenceClipTest(CFX_SkiaDeviceDriver* driver, const State&) {
|
CFX_PathData clipPath;
|
clipPath.AppendRect(1, 0, 3, 1);
|
CFX_Matrix clipMatrix;
|
driver->SaveState();
|
driver->SetClip_PathFill(&clipPath, &clipMatrix, 0);
|
driver->RestoreState(true);
|
driver->SaveState();
|
driver->SetClip_PathFill(&clipPath, &clipMatrix, 0);
|
driver->RestoreState(false);
|
driver->RestoreState(false);
|
|
driver->SaveState();
|
driver->SaveState();
|
driver->SetClip_PathFill(&clipPath, &clipMatrix, 0);
|
driver->RestoreState(true);
|
driver->SetClip_PathFill(&clipPath, &clipMatrix, 0);
|
driver->RestoreState(false);
|
driver->RestoreState(false);
|
}
|
|
void Harness(void (*Test)(CFX_SkiaDeviceDriver*, const State&),
|
const State& state) {
|
int h = 1;
|
int w = 4;
|
FPDF_BITMAP bitmap = FPDFBitmap_Create(w, h, 1);
|
EXPECT_NE(nullptr, bitmap);
|
if (!bitmap)
|
return;
|
FPDFBitmap_FillRect(bitmap, 0, 0, w, h, 0x00000000);
|
CFX_FxgeDevice geDevice;
|
CFX_DIBitmap* pBitmap = CFXBitmapFromFPDFBitmap(bitmap);
|
geDevice.Attach(pBitmap, false, nullptr, false);
|
CFX_SkiaDeviceDriver* driver =
|
static_cast<CFX_SkiaDeviceDriver*>(geDevice.GetDeviceDriver());
|
(*Test)(driver, state);
|
driver->Flush();
|
uint32_t pixel = pBitmap->GetPixel(0, 0);
|
EXPECT_EQ(state.m_pixel, pixel);
|
#ifdef SK_DEBUG
|
if (!driver) // force dump to be linked in so it can be called from debugger
|
driver->Dump();
|
#endif
|
}
|
|
} // namespace
|
|
TEST(fxge, SkiaStateEmpty) {
|
Harness(&EmptyTest, {});
|
}
|
|
TEST(fxge, SkiaStatePath) {
|
Harness(&CommonTest, {State::Change::kNo, State::Save::kYes,
|
State::Clip::kSame, State::Graphic::kPath, 0xFF112233});
|
Harness(&CommonTest,
|
{State::Change::kNo, State::Save::kYes, State::Clip::kDifferentPath,
|
State::Graphic::kPath, 0xFF112233});
|
Harness(&CommonTest, {State::Change::kNo, State::Save::kYes, State::Clip::kNo,
|
State::Graphic::kPath, 0xFF112233});
|
Harness(&CommonTest, {State::Change::kYes, State::Save::kNo, State::Clip::kNo,
|
State::Graphic::kPath, 0xFF112233});
|
Harness(&CommonTest, {State::Change::kNo, State::Save::kNo, State::Clip::kNo,
|
State::Graphic::kPath, 0xFF112233});
|
}
|
|
TEST(fxge, SkiaStateText) {
|
Harness(&CommonTest,
|
{State::Change::kNo, State::Save::kYes, State::Clip::kDifferentMatrix,
|
State::Graphic::kText, 0xFF445566});
|
Harness(&CommonTest, {State::Change::kNo, State::Save::kYes,
|
State::Clip::kSame, State::Graphic::kText, 0xFF445566});
|
}
|
|
TEST(fxge, SkiaStateOOSClip) {
|
Harness(&OutOfSequenceClipTest, {});
|
}
|