2015-08-11 17:30:12 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2013 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
2013-10-22 14:49:03 +00:00
|
|
|
#include "Test.h"
|
|
|
|
|
2015-12-10 16:59:43 +00:00
|
|
|
#include "Resources.h"
|
2013-10-22 14:49:03 +00:00
|
|
|
#include "SkCanvas.h"
|
|
|
|
#include "SkDocument.h"
|
|
|
|
#include "SkOSFile.h"
|
|
|
|
#include "SkStream.h"
|
2015-12-10 16:59:43 +00:00
|
|
|
#include "SkPixelSerializer.h"
|
2013-10-22 14:49:03 +00:00
|
|
|
|
|
|
|
static void test_empty(skiatest::Reporter* reporter) {
|
|
|
|
SkDynamicMemoryWStream stream;
|
|
|
|
|
2016-04-27 14:45:18 +00:00
|
|
|
sk_sp<SkDocument> doc(SkDocument::MakePDF(&stream));
|
2013-10-22 14:49:03 +00:00
|
|
|
|
|
|
|
doc->close();
|
|
|
|
|
|
|
|
REPORTER_ASSERT(reporter, stream.bytesWritten() == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_abort(skiatest::Reporter* reporter) {
|
|
|
|
SkDynamicMemoryWStream stream;
|
2016-04-27 14:45:18 +00:00
|
|
|
sk_sp<SkDocument> doc(SkDocument::MakePDF(&stream));
|
2013-10-22 14:49:03 +00:00
|
|
|
|
|
|
|
SkCanvas* canvas = doc->beginPage(100, 100);
|
|
|
|
canvas->drawColor(SK_ColorRED);
|
|
|
|
doc->endPage();
|
|
|
|
|
|
|
|
doc->abort();
|
|
|
|
|
2016-03-21 20:45:05 +00:00
|
|
|
// Test that only the header is written, not the full document.
|
|
|
|
REPORTER_ASSERT(reporter, stream.bytesWritten() < 256);
|
2013-10-22 14:49:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void test_abortWithFile(skiatest::Reporter* reporter) {
|
2015-01-20 17:30:20 +00:00
|
|
|
SkString tmpDir = skiatest::GetTmpDir();
|
2013-10-22 14:49:03 +00:00
|
|
|
|
|
|
|
if (tmpDir.isEmpty()) {
|
|
|
|
return; // TODO(edisonn): unfortunatelly this pattern is used in other
|
|
|
|
// tests, but if GetTmpDir() starts returning and empty dir
|
|
|
|
// allways, then all these tests will be disabled.
|
|
|
|
}
|
|
|
|
|
2014-07-29 02:26:58 +00:00
|
|
|
SkString path = SkOSPath::Join(tmpDir.c_str(), "aborted.pdf");
|
2013-10-22 14:49:03 +00:00
|
|
|
|
|
|
|
// Make sure doc's destructor is called to flush.
|
|
|
|
{
|
2016-04-27 14:45:18 +00:00
|
|
|
sk_sp<SkDocument> doc(SkDocument::MakePDF(path.c_str()));
|
2013-10-22 14:49:03 +00:00
|
|
|
|
|
|
|
SkCanvas* canvas = doc->beginPage(100, 100);
|
|
|
|
canvas->drawColor(SK_ColorRED);
|
|
|
|
doc->endPage();
|
|
|
|
|
|
|
|
doc->abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
FILE* file = fopen(path.c_str(), "r");
|
|
|
|
// The created file should be empty.
|
|
|
|
char buffer[100];
|
|
|
|
REPORTER_ASSERT(reporter, fread(buffer, 1, 1, file) == 0);
|
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_file(skiatest::Reporter* reporter) {
|
2015-01-20 17:30:20 +00:00
|
|
|
SkString tmpDir = skiatest::GetTmpDir();
|
2013-10-22 14:49:03 +00:00
|
|
|
if (tmpDir.isEmpty()) {
|
|
|
|
return; // TODO(edisonn): unfortunatelly this pattern is used in other
|
|
|
|
// tests, but if GetTmpDir() starts returning and empty dir
|
|
|
|
// allways, then all these tests will be disabled.
|
|
|
|
}
|
|
|
|
|
2014-07-29 02:26:58 +00:00
|
|
|
SkString path = SkOSPath::Join(tmpDir.c_str(), "file.pdf");
|
2013-10-22 14:49:03 +00:00
|
|
|
|
2016-04-27 14:45:18 +00:00
|
|
|
sk_sp<SkDocument> doc(SkDocument::MakePDF(path.c_str()));
|
2013-10-22 14:49:03 +00:00
|
|
|
|
|
|
|
SkCanvas* canvas = doc->beginPage(100, 100);
|
|
|
|
|
|
|
|
canvas->drawColor(SK_ColorRED);
|
|
|
|
doc->endPage();
|
|
|
|
doc->close();
|
|
|
|
|
|
|
|
FILE* file = fopen(path.c_str(), "r");
|
2015-08-27 14:41:13 +00:00
|
|
|
REPORTER_ASSERT(reporter, file != nullptr);
|
2013-10-22 14:49:03 +00:00
|
|
|
char header[100];
|
2013-10-22 18:33:21 +00:00
|
|
|
REPORTER_ASSERT(reporter, fread(header, 4, 1, file) != 0);
|
2013-10-22 14:49:03 +00:00
|
|
|
REPORTER_ASSERT(reporter, strncmp(header, "%PDF", 4) == 0);
|
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_close(skiatest::Reporter* reporter) {
|
|
|
|
SkDynamicMemoryWStream stream;
|
2016-04-27 14:45:18 +00:00
|
|
|
sk_sp<SkDocument> doc(SkDocument::MakePDF(&stream));
|
2013-10-22 14:49:03 +00:00
|
|
|
|
|
|
|
SkCanvas* canvas = doc->beginPage(100, 100);
|
|
|
|
canvas->drawColor(SK_ColorRED);
|
|
|
|
doc->endPage();
|
|
|
|
|
|
|
|
doc->close();
|
|
|
|
|
|
|
|
REPORTER_ASSERT(reporter, stream.bytesWritten() != 0);
|
|
|
|
}
|
|
|
|
|
2016-09-09 18:41:59 +00:00
|
|
|
DEF_TEST(SkPDF_document_tests, reporter) {
|
2015-08-11 20:35:12 +00:00
|
|
|
REQUIRE_PDF_DOCUMENT(document_tests, reporter);
|
2013-10-22 14:49:03 +00:00
|
|
|
test_empty(reporter);
|
|
|
|
test_abort(reporter);
|
|
|
|
test_abortWithFile(reporter);
|
|
|
|
test_file(reporter);
|
|
|
|
test_close(reporter);
|
|
|
|
}
|
2015-12-10 16:59:43 +00:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
class JPEGSerializer final : public SkPixelSerializer {
|
|
|
|
bool onUseEncodedData(const void*, size_t) override { return true; }
|
|
|
|
SkData* onEncode(const SkPixmap& pixmap) override {
|
|
|
|
SkBitmap bm;
|
|
|
|
return bm.installPixels(pixmap.info(),
|
|
|
|
pixmap.writable_addr(),
|
|
|
|
pixmap.rowBytes(),
|
|
|
|
pixmap.ctable(),
|
|
|
|
nullptr, nullptr)
|
|
|
|
? SkImageEncoder::EncodeData(bm, SkImageEncoder::kJPEG_Type, 85)
|
|
|
|
: nullptr;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
size_t count_bytes(const SkBitmap& bm, bool useDCT) {
|
|
|
|
SkDynamicMemoryWStream stream;
|
2016-04-27 14:45:18 +00:00
|
|
|
sk_sp<SkDocument> doc;
|
2015-12-10 16:59:43 +00:00
|
|
|
if (useDCT) {
|
2016-04-27 14:45:18 +00:00
|
|
|
doc = SkDocument::MakePDF(&stream, SK_ScalarDefaultRasterDPI,
|
|
|
|
SkDocument::PDFMetadata(),
|
|
|
|
sk_make_sp<JPEGSerializer>(), false);
|
2015-12-10 16:59:43 +00:00
|
|
|
} else {
|
2016-04-27 14:45:18 +00:00
|
|
|
doc = SkDocument::MakePDF(&stream);
|
2015-12-10 16:59:43 +00:00
|
|
|
}
|
|
|
|
SkCanvas* canvas = doc->beginPage(64, 64);
|
|
|
|
canvas->drawBitmap(bm, 0, 0);
|
|
|
|
doc->endPage();
|
|
|
|
doc->close();
|
|
|
|
return stream.bytesWritten();
|
|
|
|
}
|
|
|
|
|
2016-09-09 18:41:59 +00:00
|
|
|
DEF_TEST(SkPDF_document_dct_encoder, r) {
|
|
|
|
REQUIRE_PDF_DOCUMENT(SkPDF_document_dct_encoder, r);
|
2015-12-10 16:59:43 +00:00
|
|
|
SkBitmap bm;
|
|
|
|
if (GetResourceAsBitmap("mandrill_64.png", &bm)) {
|
|
|
|
// Lossy encoding works better on photographs.
|
|
|
|
REPORTER_ASSERT(r, count_bytes(bm, true) < count_bytes(bm, false));
|
|
|
|
}
|
|
|
|
}
|
2016-01-06 17:02:25 +00:00
|
|
|
|
2016-09-09 18:41:59 +00:00
|
|
|
DEF_TEST(SkPDF_document_skbug_4734, r) {
|
|
|
|
REQUIRE_PDF_DOCUMENT(SkPDF_document_skbug_4734, r);
|
2016-01-06 17:02:25 +00:00
|
|
|
SkDynamicMemoryWStream stream;
|
2016-04-27 14:45:18 +00:00
|
|
|
sk_sp<SkDocument> doc(SkDocument::MakePDF(&stream));
|
2016-01-06 17:02:25 +00:00
|
|
|
SkCanvas* canvas = doc->beginPage(64, 64);
|
|
|
|
canvas->scale(10000.0f, 10000.0f);
|
|
|
|
canvas->translate(20.0f, 10.0f);
|
|
|
|
canvas->rotate(30.0f);
|
|
|
|
const char text[] = "HELLO";
|
|
|
|
canvas->drawText(text, strlen(text), 0, 0, SkPaint());
|
|
|
|
}
|
2016-09-09 18:41:59 +00:00
|
|
|
|
|
|
|
static bool contains(const uint8_t* result, size_t size, const char expectation[]) {
|
|
|
|
size_t len = strlen(expectation);
|
|
|
|
size_t N = 1 + size - len;
|
|
|
|
for (size_t i = 0; i < N; ++i) {
|
|
|
|
if (0 == memcmp(result + i, expectation, len)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// verify that the PDFA flag does something.
|
|
|
|
DEF_TEST(SkPDF_pdfa_document, r) {
|
|
|
|
REQUIRE_PDF_DOCUMENT(SkPDF_pdfa_document, r);
|
|
|
|
|
|
|
|
SkDocument::PDFMetadata pdfMetadata;
|
|
|
|
pdfMetadata.fTitle = "test document";
|
|
|
|
pdfMetadata.fCreation.fEnabled = true;
|
|
|
|
pdfMetadata.fCreation.fDateTime = {0, 1999, 12, 5, 31, 23, 59, 59};
|
|
|
|
|
|
|
|
SkDynamicMemoryWStream buffer;
|
|
|
|
auto doc = SkDocument::MakePDF(&buffer, SK_ScalarDefaultRasterDPI,
|
|
|
|
pdfMetadata, nullptr, /* pdfa = */ true);
|
|
|
|
doc->beginPage(64, 64)->drawColor(SK_ColorRED);
|
|
|
|
doc->close();
|
2016-09-12 19:01:44 +00:00
|
|
|
sk_sp<SkData> data(buffer.detachAsData());
|
|
|
|
|
2016-09-09 18:41:59 +00:00
|
|
|
static const char* expectations[] = {
|
|
|
|
"sRGB IEC61966-2.1",
|
|
|
|
"<dc:title><rdf:Alt><rdf:li xml:lang=\"x-default\">test document",
|
|
|
|
"<xmp:CreateDate>1999-12-31T23:59:59+00:00</xmp:CreateDate>",
|
|
|
|
"/Subtype /XML",
|
|
|
|
"/CreationDate (D:19991231235959+00'00')>>",
|
|
|
|
};
|
|
|
|
for (const char* expectation : expectations) {
|
|
|
|
if (!contains(data->bytes(), data->size(), expectation)) {
|
|
|
|
ERRORF(r, "PDFA expectation missing: '%s'.", expectation);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pdfMetadata.fProducer = "phoney library";
|
|
|
|
doc = SkDocument::MakePDF(&buffer, SK_ScalarDefaultRasterDPI,
|
|
|
|
pdfMetadata, nullptr, /* pdfa = */ true);
|
|
|
|
doc->beginPage(64, 64)->drawColor(SK_ColorRED);
|
|
|
|
doc->close();
|
2016-09-12 19:01:44 +00:00
|
|
|
data = buffer.detachAsData();
|
2016-09-09 18:41:59 +00:00
|
|
|
|
|
|
|
static const char* moreExpectations[] = {
|
|
|
|
"/Producer (phoney library)",
|
|
|
|
"/ProductionLibrary (Skia/PDF m",
|
|
|
|
"<!-- <skia:ProductionLibrary>Skia/PDF m",
|
|
|
|
"<pdf:Producer>phoney library</pdf:Producer>",
|
|
|
|
};
|
|
|
|
for (const char* expectation : moreExpectations) {
|
|
|
|
if (!contains(data->bytes(), data->size(), expectation)) {
|
|
|
|
ERRORF(r, "PDFA expectation missing: '%s'.", expectation);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|