blob: 7861ef0fe145dc7e7359ce5d4361150283fa8aec [file] [log] [blame]
vandebo@chromium.orgf66025d2010-10-01 23:26:55 +00001/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00002 * Copyright 2010 The Android Open Source Project
vandebo@chromium.orgf66025d2010-10-01 23:26:55 +00003 *
epoger@google.comec3ed6a2011-07-28 14:26:00 +00004 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
vandebo@chromium.orgf66025d2010-10-01 23:26:55 +00006 */
7
edisonn@google.comd9dfa182013-04-24 13:01:01 +00008#include "SkBitmap.h"
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +00009#include "SkCanvas.h"
reed@google.com8a85d0c2011-06-24 19:12:12 +000010#include "SkData.h"
halcanary3d32d502015-03-01 06:55:20 -080011#include "SkDocument.h"
vandebo@chromium.org421d6442011-07-20 17:39:01 +000012#include "SkFlate.h"
edisonn@google.comd9dfa182013-04-24 13:01:01 +000013#include "SkImageEncoder.h"
14#include "SkMatrix.h"
halcanarya1f1ee92015-02-20 06:17:26 -080015#include "SkPDFCanon.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000016#include "SkPDFCatalog.h"
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +000017#include "SkPDFDevice.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000018#include "SkPDFStream.h"
19#include "SkPDFTypes.h"
senorblancob0e89dc2014-10-20 14:03:12 -070020#include "SkReadBuffer.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000021#include "SkScalar.h"
22#include "SkStream.h"
vandebo@chromium.org421d6442011-07-20 17:39:01 +000023#include "SkTypes.h"
tfarina@chromium.org8f6884a2014-01-24 20:56:26 +000024#include "Test.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000025
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +000026class SkPDFTestDict : public SkPDFDict {
27public:
edisonn@google.com6addb192013-04-02 15:33:08 +000028 virtual void getResources(const SkTSet<SkPDFObject*>& knownResourceObjects,
29 SkTSet<SkPDFObject*>* newResourceObjects) {
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +000030 for (int i = 0; i < fResources.count(); i++) {
edisonn@google.com6addb192013-04-02 15:33:08 +000031 newResourceObjects->add(fResources[i]);
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +000032 fResources[i]->ref();
33 }
34 }
35
36 void addResource(SkPDFObject* object) {
37 fResources.append(1, &object);
38 }
39
40private:
41 SkTDArray<SkPDFObject*> fResources;
42};
43
commit-bot@chromium.org608ea652013-10-03 19:29:21 +000044#define DUMMY_TEXT "DCT compessed stream."
45
reed@google.com8a85d0c2011-06-24 19:12:12 +000046static bool stream_equals(const SkDynamicMemoryWStream& stream, size_t offset,
47 const void* buffer, size_t len) {
48 SkAutoDataUnref data(stream.copyToData());
robertphillips@google.com59f46b82012-07-10 17:30:58 +000049 if (offset + len > data->size()) {
reed@google.com8a85d0c2011-06-24 19:12:12 +000050 return false;
51 }
robertphillips@google.com59f46b82012-07-10 17:30:58 +000052 return memcmp(data->bytes() + offset, buffer, len) == 0;
reed@google.com8a85d0c2011-06-24 19:12:12 +000053}
54
halcanary6a144342015-01-23 11:45:10 -080055static void emit_object(SkPDFObject* object,
56 SkWStream* stream,
57 SkPDFCatalog* catalog,
58 bool indirect) {
59 SkPDFObject* realObject = catalog->getSubstituteObject(object);
60 if (indirect) {
halcanarybf799cd2015-02-10 13:32:09 -080061 stream->writeDecAsText(catalog->getObjectNumber(object));
62 stream->writeText(" 0 obj\n"); // Generation number is always 0.
halcanary6a144342015-01-23 11:45:10 -080063 realObject->emitObject(stream, catalog);
64 stream->writeText("\nendobj\n");
65 } else {
66 realObject->emitObject(stream, catalog);
67 }
68}
69
70static size_t get_output_size(SkPDFObject* object,
71 SkPDFCatalog* catalog,
72 bool indirect) {
73 SkDynamicMemoryWStream buffer;
74 emit_object(object, &buffer, catalog, indirect);
75 return buffer.getOffset();
76}
77
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000078static void CheckObjectOutput(skiatest::Reporter* reporter, SkPDFObject* obj,
vandebo@chromium.org421d6442011-07-20 17:39:01 +000079 const char* expectedData, size_t expectedSize,
mtkleincabc08c2015-02-19 08:29:24 -080080 bool indirect) {
81 SkPDFCatalog catalog;
halcanary6a144342015-01-23 11:45:10 -080082 size_t directSize = get_output_size(obj, &catalog, false);
vandebo@chromium.org421d6442011-07-20 17:39:01 +000083 REPORTER_ASSERT(reporter, directSize == expectedSize);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000084
85 SkDynamicMemoryWStream buffer;
halcanary6a144342015-01-23 11:45:10 -080086 emit_object(obj, &buffer, &catalog, false);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000087 REPORTER_ASSERT(reporter, directSize == buffer.getOffset());
vandebo@chromium.org421d6442011-07-20 17:39:01 +000088 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedData,
reed@google.com8a85d0c2011-06-24 19:12:12 +000089 directSize));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000090
91 if (indirect) {
92 // Indirect output.
93 static char header[] = "1 0 obj\n";
94 static size_t headerLen = strlen(header);
95 static char footer[] = "\nendobj\n";
96 static size_t footerLen = strlen(footer);
97
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000098 catalog.addObject(obj, false);
99
halcanary6a144342015-01-23 11:45:10 -0800100 size_t indirectSize = get_output_size(obj, &catalog, true);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000101 REPORTER_ASSERT(reporter,
102 indirectSize == directSize + headerLen + footerLen);
103
104 buffer.reset();
halcanary6a144342015-01-23 11:45:10 -0800105 emit_object(obj, &buffer, &catalog, true);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000106 REPORTER_ASSERT(reporter, indirectSize == buffer.getOffset());
reed@google.com8a85d0c2011-06-24 19:12:12 +0000107 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, header, headerLen));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000108 REPORTER_ASSERT(reporter, stream_equals(buffer, headerLen, expectedData,
reed@google.com8a85d0c2011-06-24 19:12:12 +0000109 directSize));
110 REPORTER_ASSERT(reporter, stream_equals(buffer, headerLen + directSize,
111 footer, footerLen));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000112 }
113}
114
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000115static void SimpleCheckObjectOutput(skiatest::Reporter* reporter,
116 SkPDFObject* obj,
robertphillips@google.com4debcac2012-05-14 16:33:36 +0000117 const char* expectedResult) {
118 CheckObjectOutput(reporter, obj, expectedResult,
mtkleincabc08c2015-02-19 08:29:24 -0800119 strlen(expectedResult), true);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000120}
121
122static void TestPDFStream(skiatest::Reporter* reporter) {
123 char streamBytes[] = "Test\nFoo\tBar";
scroggoa1193e42015-01-21 12:09:53 -0800124 SkAutoTDelete<SkMemoryStream> streamData(new SkMemoryStream(
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000125 streamBytes, strlen(streamBytes), true));
126 SkAutoTUnref<SkPDFStream> stream(new SkPDFStream(streamData.get()));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000127 SimpleCheckObjectOutput(
128 reporter, stream.get(),
129 "<</Length 12\n>> stream\nTest\nFoo\tBar\nendstream");
130 stream->insert("Attribute", new SkPDFInt(42))->unref();
131 SimpleCheckObjectOutput(reporter, stream.get(),
132 "<</Length 12\n/Attribute 42\n>> stream\n"
133 "Test\nFoo\tBar\nendstream");
134
halcanary91d1d622015-02-17 14:43:06 -0800135#ifndef SK_NO_FLATE
136 {
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000137 char streamBytes2[] = "This is a longer string, so that compression "
138 "can do something with it. With shorter strings, "
139 "the short circuit logic cuts in and we end up "
140 "with an uncompressed string.";
141 SkAutoDataUnref streamData2(SkData::NewWithCopy(streamBytes2,
142 strlen(streamBytes2)));
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000143 SkAutoTUnref<SkPDFStream> stream(new SkPDFStream(streamData2.get()));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000144
145 SkDynamicMemoryWStream compressedByteStream;
146 SkFlate::Deflate(streamData2.get(), &compressedByteStream);
147 SkAutoDataUnref compressedData(compressedByteStream.copyToData());
148
mtkleincabc08c2015-02-19 08:29:24 -0800149 SkDynamicMemoryWStream expected;
150 expected.writeText("<</Filter /FlateDecode\n/Length 116\n"
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000151 ">> stream\n");
mtkleincabc08c2015-02-19 08:29:24 -0800152 expected.write(compressedData->data(), compressedData->size());
153 expected.writeText("\nendstream");
154 SkAutoDataUnref expectedResultData2(expected.copyToData());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000155 CheckObjectOutput(reporter, stream.get(),
robertphillips@google.com59f46b82012-07-10 17:30:58 +0000156 (const char*) expectedResultData2->data(),
mtkleincabc08c2015-02-19 08:29:24 -0800157 expectedResultData2->size(), true);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000158 }
halcanary91d1d622015-02-17 14:43:06 -0800159#endif // SK_NO_FLATE
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000160}
161
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000162static void TestCatalog(skiatest::Reporter* reporter) {
mtkleincabc08c2015-02-19 08:29:24 -0800163 SkPDFCatalog catalog;
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000164 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
165 SkAutoTUnref<SkPDFInt> int2(new SkPDFInt(2));
166 SkAutoTUnref<SkPDFInt> int3(new SkPDFInt(3));
167 int1.get()->ref();
168 SkAutoTUnref<SkPDFInt> int1Again(int1.get());
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000169
170 catalog.addObject(int1.get(), false);
171 catalog.addObject(int2.get(), false);
172 catalog.addObject(int3.get(), false);
173
halcanarybf799cd2015-02-10 13:32:09 -0800174 REPORTER_ASSERT(reporter, catalog.getObjectNumber(int1.get()) == 1);
175 REPORTER_ASSERT(reporter, catalog.getObjectNumber(int2.get()) == 2);
176 REPORTER_ASSERT(reporter, catalog.getObjectNumber(int3.get()) == 3);
177 REPORTER_ASSERT(reporter, catalog.getObjectNumber(int1Again.get()) == 1);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000178}
179
180static void TestObjectRef(skiatest::Reporter* reporter) {
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000181 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
182 SkAutoTUnref<SkPDFInt> int2(new SkPDFInt(2));
183 SkAutoTUnref<SkPDFObjRef> int2ref(new SkPDFObjRef(int2.get()));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000184
mtkleincabc08c2015-02-19 08:29:24 -0800185 SkPDFCatalog catalog;
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000186 catalog.addObject(int1.get(), false);
187 catalog.addObject(int2.get(), false);
halcanarybf799cd2015-02-10 13:32:09 -0800188 REPORTER_ASSERT(reporter, catalog.getObjectNumber(int1.get()) == 1);
189 REPORTER_ASSERT(reporter, catalog.getObjectNumber(int2.get()) == 2);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000190
191 char expectedResult[] = "2 0 R";
192 SkDynamicMemoryWStream buffer;
halcanary4fc48af2015-01-12 10:07:50 -0800193 int2ref->emitObject(&buffer, &catalog);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000194 REPORTER_ASSERT(reporter, buffer.getOffset() == strlen(expectedResult));
reed@google.com8a85d0c2011-06-24 19:12:12 +0000195 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
196 buffer.getOffset()));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000197}
198
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000199static void TestSubstitute(skiatest::Reporter* reporter) {
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000200 SkAutoTUnref<SkPDFTestDict> proxy(new SkPDFTestDict());
201 SkAutoTUnref<SkPDFTestDict> stub(new SkPDFTestDict());
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000202
halcanarybf799cd2015-02-10 13:32:09 -0800203 proxy->insert("Value", new SkPDFInt(33))->unref();
204 stub->insert("Value", new SkPDFInt(44))->unref();
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000205
mtkleincabc08c2015-02-19 08:29:24 -0800206 SkPDFCatalog catalog;
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000207 catalog.addObject(proxy.get(), false);
208 catalog.setSubstitute(proxy.get(), stub.get());
209
halcanarybf799cd2015-02-10 13:32:09 -0800210 REPORTER_ASSERT(reporter, stub.get() == catalog.getSubstituteObject(proxy));
211 REPORTER_ASSERT(reporter, proxy.get() != catalog.getSubstituteObject(stub));
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000212}
213
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000214// This test used to assert without the fix submitted for
215// http://code.google.com/p/skia/issues/detail?id=1083.
216// SKP files might have invalid glyph ids. This test ensures they are ignored,
217// and there is no assert on input data in Debug mode.
sugoi@google.com54f0d1b2013-02-27 19:17:41 +0000218static void test_issue1083() {
halcanary3d32d502015-03-01 06:55:20 -0800219 SkDynamicMemoryWStream outStream;
220 SkAutoTUnref<SkDocument> doc(SkDocument::CreatePDF(&outStream));
221 SkCanvas* canvas = doc->beginPage(100.0f, 100.0f);
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000222 SkPaint paint;
223 paint.setTextEncoding(SkPaint::kGlyphID_TextEncoding);
224
225 uint16_t glyphID = 65000;
halcanary3d32d502015-03-01 06:55:20 -0800226 canvas->drawText(&glyphID, 2, 0, 0, paint);
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000227
halcanary3d32d502015-03-01 06:55:20 -0800228 doc->close();
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000229}
230
tfarina@chromium.orge4fafb12013-12-12 21:11:12 +0000231DEF_TEST(PDFPrimitives, reporter) {
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000232 SkAutoTUnref<SkPDFInt> int42(new SkPDFInt(42));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000233 SimpleCheckObjectOutput(reporter, int42.get(), "42");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000234
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000235 SkAutoTUnref<SkPDFScalar> realHalf(new SkPDFScalar(SK_ScalarHalf));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000236 SimpleCheckObjectOutput(reporter, realHalf.get(), "0.5");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000237
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000238 SkAutoTUnref<SkPDFScalar> bigScalar(new SkPDFScalar(110999.75f));
vandebo@chromium.org6cc26da2011-05-18 17:08:05 +0000239#if !defined(SK_ALLOW_LARGE_PDF_SCALARS)
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000240 SimpleCheckObjectOutput(reporter, bigScalar.get(), "111000");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000241#else
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000242 SimpleCheckObjectOutput(reporter, bigScalar.get(), "110999.75");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000243
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000244 SkAutoTUnref<SkPDFScalar> biggerScalar(new SkPDFScalar(50000000.1));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000245 SimpleCheckObjectOutput(reporter, biggerScalar.get(), "50000000");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000246
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000247 SkAutoTUnref<SkPDFScalar> smallestScalar(new SkPDFScalar(1.0/65536));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000248 SimpleCheckObjectOutput(reporter, smallestScalar.get(), "0.00001526");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000249#endif
250
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000251 SkAutoTUnref<SkPDFString> stringSimple(
252 new SkPDFString("test ) string ( foo"));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000253 SimpleCheckObjectOutput(reporter, stringSimple.get(),
254 "(test \\) string \\( foo)");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000255 SkAutoTUnref<SkPDFString> stringComplex(
256 new SkPDFString("\ttest ) string ( foo"));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000257 SimpleCheckObjectOutput(reporter, stringComplex.get(),
258 "<0974657374202920737472696E67202820666F6F>");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000259
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000260 SkAutoTUnref<SkPDFName> name(new SkPDFName("Test name\twith#tab"));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000261 const char expectedResult[] = "/Test#20name#09with#23tab";
262 CheckObjectOutput(reporter, name.get(), expectedResult,
mtkleincabc08c2015-02-19 08:29:24 -0800263 strlen(expectedResult), false);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000264
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000265 SkAutoTUnref<SkPDFName> escapedName(new SkPDFName("A#/%()<>[]{}B"));
vandebo@chromium.org251a7662012-09-21 17:50:50 +0000266 const char escapedNameExpected[] = "/A#23#2F#25#28#29#3C#3E#5B#5D#7B#7DB";
267 CheckObjectOutput(reporter, escapedName.get(), escapedNameExpected,
mtkleincabc08c2015-02-19 08:29:24 -0800268 strlen(escapedNameExpected), false);
vandebo@chromium.org251a7662012-09-21 17:50:50 +0000269
vandebo@chromium.orgc0376fe2012-03-05 18:44:33 +0000270 // Test that we correctly handle characters with the high-bit set.
bungeman@google.comf8aa18c2012-03-19 21:04:52 +0000271 const unsigned char highBitCString[] = {0xDE, 0xAD, 'b', 'e', 0xEF, 0};
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000272 SkAutoTUnref<SkPDFName> highBitName(
273 new SkPDFName((const char*)highBitCString));
vandebo@chromium.orgc0376fe2012-03-05 18:44:33 +0000274 const char highBitExpectedResult[] = "/#DE#ADbe#EF";
275 CheckObjectOutput(reporter, highBitName.get(), highBitExpectedResult,
mtkleincabc08c2015-02-19 08:29:24 -0800276 strlen(highBitExpectedResult), false);
vandebo@chromium.orgc0376fe2012-03-05 18:44:33 +0000277
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000278 SkAutoTUnref<SkPDFArray> array(new SkPDFArray);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000279 SimpleCheckObjectOutput(reporter, array.get(), "[]");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000280 array->append(int42.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000281 SimpleCheckObjectOutput(reporter, array.get(), "[42]");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000282 array->append(realHalf.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000283 SimpleCheckObjectOutput(reporter, array.get(), "[42 0.5]");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000284 SkAutoTUnref<SkPDFInt> int0(new SkPDFInt(0));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000285 array->append(int0.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000286 SimpleCheckObjectOutput(reporter, array.get(), "[42 0.5 0]");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000287 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000288 array->setAt(0, int1.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000289 SimpleCheckObjectOutput(reporter, array.get(), "[1 0.5 0]");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000290
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000291 SkAutoTUnref<SkPDFDict> dict(new SkPDFDict);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000292 SimpleCheckObjectOutput(reporter, dict.get(), "<<>>");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000293 SkAutoTUnref<SkPDFName> n1(new SkPDFName("n1"));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000294 dict->insert(n1.get(), int42.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000295 SimpleCheckObjectOutput(reporter, dict.get(), "<</n1 42\n>>");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000296 SkAutoTUnref<SkPDFName> n2(new SkPDFName("n2"));
297 SkAutoTUnref<SkPDFName> n3(new SkPDFName("n3"));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000298 dict->insert(n2.get(), realHalf.get());
299 dict->insert(n3.get(), array.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000300 SimpleCheckObjectOutput(reporter, dict.get(),
301 "<</n1 42\n/n2 0.5\n/n3 [1 0.5 0]\n>>");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000302
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000303 TestPDFStream(reporter);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000304
305 TestCatalog(reporter);
306
307 TestObjectRef(reporter);
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000308
309 TestSubstitute(reporter);
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000310
sugoi@google.com54f0d1b2013-02-27 19:17:41 +0000311 test_issue1083();
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000312}
senorblancob0e89dc2014-10-20 14:03:12 -0700313
314namespace {
315
316class DummyImageFilter : public SkImageFilter {
317public:
318 DummyImageFilter(bool visited = false) : SkImageFilter(0, NULL), fVisited(visited) {}
mtklein72c9faa2015-01-09 10:06:39 -0800319 ~DummyImageFilter() SK_OVERRIDE {}
senorblancob0e89dc2014-10-20 14:03:12 -0700320 virtual bool onFilterImage(Proxy*, const SkBitmap& src, const Context&,
rmistry465206a2015-02-02 12:08:18 -0800321 SkBitmap* result, SkIPoint* offset) const SK_OVERRIDE {
senorblancob0e89dc2014-10-20 14:03:12 -0700322 fVisited = true;
323 offset->fX = offset->fY = 0;
324 *result = src;
325 return true;
326 }
robertphillipsf3f5bad2014-12-19 13:49:15 -0800327 SK_TO_STRING_OVERRIDE()
senorblancob0e89dc2014-10-20 14:03:12 -0700328 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(DummyImageFilter)
senorblancob0e89dc2014-10-20 14:03:12 -0700329 bool visited() const { return fVisited; }
330
331private:
332 mutable bool fVisited;
333};
334
335SkFlattenable* DummyImageFilter::CreateProc(SkReadBuffer& buffer) {
336 SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 0);
337 bool visited = buffer.readBool();
338 return SkNEW_ARGS(DummyImageFilter, (visited));
339}
340
robertphillipsf3f5bad2014-12-19 13:49:15 -0800341#ifndef SK_IGNORE_TO_STRING
342void DummyImageFilter::toString(SkString* str) const {
343 str->appendf("DummyImageFilter: (");
344 str->append(")");
345}
346#endif
347
senorblancob0e89dc2014-10-20 14:03:12 -0700348};
349
350// Check that PDF rendering of image filters successfully falls back to
351// CPU rasterization.
352DEF_TEST(PDFImageFilter, reporter) {
halcanary3d32d502015-03-01 06:55:20 -0800353 SkDynamicMemoryWStream stream;
354 SkAutoTUnref<SkDocument> doc(SkDocument::CreatePDF(&stream));
355 SkCanvas* canvas = doc->beginPage(100.0f, 100.0f);
356
senorblancob0e89dc2014-10-20 14:03:12 -0700357 SkAutoTUnref<DummyImageFilter> filter(new DummyImageFilter());
358
359 // Filter just created; should be unvisited.
360 REPORTER_ASSERT(reporter, !filter->visited());
361 SkPaint paint;
362 paint.setImageFilter(filter.get());
halcanary3d32d502015-03-01 06:55:20 -0800363 canvas->drawRect(SkRect::MakeWH(100, 100), paint);
364 doc->close();
senorblancob0e89dc2014-10-20 14:03:12 -0700365
366 // Filter was used in rendering; should be visited.
367 REPORTER_ASSERT(reporter, filter->visited());
368}