blob: cf929b4488422816ae93f32a3dc085c6be490b80 [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
vandebo@chromium.orgf66025d2010-10-01 23:26:55 +00002/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00003 * Copyright 2010 The Android Open Source Project
vandebo@chromium.orgf66025d2010-10-01 23:26:55 +00004 *
epoger@google.comec3ed6a2011-07-28 14:26:00 +00005 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
vandebo@chromium.orgf66025d2010-10-01 23:26:55 +00007 */
8
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +00009#include "Test.h"
edisonn@google.comd9dfa182013-04-24 13:01:01 +000010#include "SkBitmap.h"
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +000011#include "SkCanvas.h"
reed@google.com8a85d0c2011-06-24 19:12:12 +000012#include "SkData.h"
vandebo@chromium.org421d6442011-07-20 17:39:01 +000013#include "SkFlate.h"
edisonn@google.comd9dfa182013-04-24 13:01:01 +000014#include "SkImageEncoder.h"
15#include "SkMatrix.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"
20#include "SkScalar.h"
21#include "SkStream.h"
vandebo@chromium.org421d6442011-07-20 17:39:01 +000022#include "SkTypes.h"
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000023
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +000024class SkPDFTestDict : public SkPDFDict {
25public:
edisonn@google.com6addb192013-04-02 15:33:08 +000026 virtual void getResources(const SkTSet<SkPDFObject*>& knownResourceObjects,
27 SkTSet<SkPDFObject*>* newResourceObjects) {
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +000028 for (int i = 0; i < fResources.count(); i++) {
edisonn@google.com6addb192013-04-02 15:33:08 +000029 newResourceObjects->add(fResources[i]);
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +000030 fResources[i]->ref();
31 }
32 }
33
34 void addResource(SkPDFObject* object) {
35 fResources.append(1, &object);
36 }
37
38private:
39 SkTDArray<SkPDFObject*> fResources;
40};
41
edisonn@google.comd9dfa182013-04-24 13:01:01 +000042static bool encode_to_dct_stream(SkWStream* stream, const SkBitmap& bitmap, const SkIRect& rect) {
43 stream->writeText("DCT compessed stream.");
44 return true;
45}
46
reed@google.com8a85d0c2011-06-24 19:12:12 +000047static bool stream_equals(const SkDynamicMemoryWStream& stream, size_t offset,
48 const void* buffer, size_t len) {
49 SkAutoDataUnref data(stream.copyToData());
robertphillips@google.com59f46b82012-07-10 17:30:58 +000050 if (offset + len > data->size()) {
reed@google.com8a85d0c2011-06-24 19:12:12 +000051 return false;
52 }
robertphillips@google.com59f46b82012-07-10 17:30:58 +000053 return memcmp(data->bytes() + offset, buffer, len) == 0;
reed@google.com8a85d0c2011-06-24 19:12:12 +000054}
55
edisonn@google.comd9dfa182013-04-24 13:01:01 +000056static bool stream_contains(const SkDynamicMemoryWStream& stream,
57 const char* buffer) {
58 SkAutoDataUnref data(stream.copyToData());
59 int len = strlen(buffer); // our buffer does not have EOSs.
60
61 for (int offset = 0 ; offset < (int)data->size() - len; offset++) {
62 if (memcmp(data->bytes() + offset, buffer, len) == 0) {
63 return true;
64 }
65 }
66
67 return false;
68}
69
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000070static void CheckObjectOutput(skiatest::Reporter* reporter, SkPDFObject* obj,
vandebo@chromium.org421d6442011-07-20 17:39:01 +000071 const char* expectedData, size_t expectedSize,
72 bool indirect, bool compression) {
73 SkPDFDocument::Flags docFlags = (SkPDFDocument::Flags) 0;
74 if (!compression) {
edisonn@google.com8c020612013-03-12 19:53:16 +000075 docFlags = SkTBitOr(docFlags, SkPDFDocument::kFavorSpeedOverSize_Flags);
vandebo@chromium.org421d6442011-07-20 17:39:01 +000076 }
77 SkPDFCatalog catalog(docFlags);
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +000078 size_t directSize = obj->getOutputSize(&catalog, false);
vandebo@chromium.org421d6442011-07-20 17:39:01 +000079 REPORTER_ASSERT(reporter, directSize == expectedSize);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000080
81 SkDynamicMemoryWStream buffer;
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +000082 obj->emit(&buffer, &catalog, false);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000083 REPORTER_ASSERT(reporter, directSize == buffer.getOffset());
vandebo@chromium.org421d6442011-07-20 17:39:01 +000084 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedData,
reed@google.com8a85d0c2011-06-24 19:12:12 +000085 directSize));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000086
87 if (indirect) {
88 // Indirect output.
89 static char header[] = "1 0 obj\n";
90 static size_t headerLen = strlen(header);
91 static char footer[] = "\nendobj\n";
92 static size_t footerLen = strlen(footer);
93
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +000094 catalog.addObject(obj, false);
95
96 size_t indirectSize = obj->getOutputSize(&catalog, true);
97 REPORTER_ASSERT(reporter,
98 indirectSize == directSize + headerLen + footerLen);
99
100 buffer.reset();
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000101 obj->emit(&buffer, &catalog, true);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000102 REPORTER_ASSERT(reporter, indirectSize == buffer.getOffset());
reed@google.com8a85d0c2011-06-24 19:12:12 +0000103 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, header, headerLen));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000104 REPORTER_ASSERT(reporter, stream_equals(buffer, headerLen, expectedData,
reed@google.com8a85d0c2011-06-24 19:12:12 +0000105 directSize));
106 REPORTER_ASSERT(reporter, stream_equals(buffer, headerLen + directSize,
107 footer, footerLen));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000108 }
109}
110
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000111static void SimpleCheckObjectOutput(skiatest::Reporter* reporter,
112 SkPDFObject* obj,
robertphillips@google.com4debcac2012-05-14 16:33:36 +0000113 const char* expectedResult) {
114 CheckObjectOutput(reporter, obj, expectedResult,
115 strlen(expectedResult), true, false);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000116}
117
118static void TestPDFStream(skiatest::Reporter* reporter) {
119 char streamBytes[] = "Test\nFoo\tBar";
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000120 SkAutoTUnref<SkMemoryStream> streamData(new SkMemoryStream(
121 streamBytes, strlen(streamBytes), true));
122 SkAutoTUnref<SkPDFStream> stream(new SkPDFStream(streamData.get()));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000123 SimpleCheckObjectOutput(
124 reporter, stream.get(),
125 "<</Length 12\n>> stream\nTest\nFoo\tBar\nendstream");
126 stream->insert("Attribute", new SkPDFInt(42))->unref();
127 SimpleCheckObjectOutput(reporter, stream.get(),
128 "<</Length 12\n/Attribute 42\n>> stream\n"
129 "Test\nFoo\tBar\nendstream");
130
131 if (SkFlate::HaveFlate()) {
132 char streamBytes2[] = "This is a longer string, so that compression "
133 "can do something with it. With shorter strings, "
134 "the short circuit logic cuts in and we end up "
135 "with an uncompressed string.";
136 SkAutoDataUnref streamData2(SkData::NewWithCopy(streamBytes2,
137 strlen(streamBytes2)));
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000138 SkAutoTUnref<SkPDFStream> stream(new SkPDFStream(streamData2.get()));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000139
140 SkDynamicMemoryWStream compressedByteStream;
141 SkFlate::Deflate(streamData2.get(), &compressedByteStream);
142 SkAutoDataUnref compressedData(compressedByteStream.copyToData());
143
144 // Check first without compression.
145 SkDynamicMemoryWStream expectedResult1;
146 expectedResult1.writeText("<</Length 167\n>> stream\n");
147 expectedResult1.writeText(streamBytes2);
148 expectedResult1.writeText("\nendstream");
149 SkAutoDataUnref expectedResultData1(expectedResult1.copyToData());
150 CheckObjectOutput(reporter, stream.get(),
robertphillips@google.com59f46b82012-07-10 17:30:58 +0000151 (const char*) expectedResultData1->data(),
152 expectedResultData1->size(), true, false);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000153
154 // Then again with compression.
155 SkDynamicMemoryWStream expectedResult2;
156 expectedResult2.writeText("<</Filter /FlateDecode\n/Length 116\n"
157 ">> stream\n");
robertphillips@google.com59f46b82012-07-10 17:30:58 +0000158 expectedResult2.write(compressedData->data(), compressedData->size());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000159 expectedResult2.writeText("\nendstream");
160 SkAutoDataUnref expectedResultData2(expectedResult2.copyToData());
161 CheckObjectOutput(reporter, stream.get(),
robertphillips@google.com59f46b82012-07-10 17:30:58 +0000162 (const char*) expectedResultData2->data(),
163 expectedResultData2->size(), true, true);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000164 }
165}
166
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000167static void TestCatalog(skiatest::Reporter* reporter) {
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000168 SkPDFCatalog catalog((SkPDFDocument::Flags)0);
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000169 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
170 SkAutoTUnref<SkPDFInt> int2(new SkPDFInt(2));
171 SkAutoTUnref<SkPDFInt> int3(new SkPDFInt(3));
172 int1.get()->ref();
173 SkAutoTUnref<SkPDFInt> int1Again(int1.get());
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000174
175 catalog.addObject(int1.get(), false);
176 catalog.addObject(int2.get(), false);
177 catalog.addObject(int3.get(), false);
178
179 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int1.get()) == 3);
180 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int2.get()) == 3);
181 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int3.get()) == 3);
182
183 SkDynamicMemoryWStream buffer;
184 catalog.emitObjectNumber(&buffer, int1.get());
185 catalog.emitObjectNumber(&buffer, int2.get());
186 catalog.emitObjectNumber(&buffer, int3.get());
187 catalog.emitObjectNumber(&buffer, int1Again.get());
188 char expectedResult[] = "1 02 03 01 0";
reed@google.com8a85d0c2011-06-24 19:12:12 +0000189 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
190 strlen(expectedResult)));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000191}
192
193static void TestObjectRef(skiatest::Reporter* reporter) {
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000194 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
195 SkAutoTUnref<SkPDFInt> int2(new SkPDFInt(2));
196 SkAutoTUnref<SkPDFObjRef> int2ref(new SkPDFObjRef(int2.get()));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000197
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000198 SkPDFCatalog catalog((SkPDFDocument::Flags)0);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000199 catalog.addObject(int1.get(), false);
200 catalog.addObject(int2.get(), false);
201 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int1.get()) == 3);
202 REPORTER_ASSERT(reporter, catalog.getObjectNumberSize(int2.get()) == 3);
203
204 char expectedResult[] = "2 0 R";
205 SkDynamicMemoryWStream buffer;
206 int2ref->emitObject(&buffer, &catalog, false);
207 REPORTER_ASSERT(reporter, buffer.getOffset() == strlen(expectedResult));
reed@google.com8a85d0c2011-06-24 19:12:12 +0000208 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
209 buffer.getOffset()));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000210}
211
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000212static void TestSubstitute(skiatest::Reporter* reporter) {
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000213 SkAutoTUnref<SkPDFTestDict> proxy(new SkPDFTestDict());
214 SkAutoTUnref<SkPDFTestDict> stub(new SkPDFTestDict());
215 SkAutoTUnref<SkPDFInt> int33(new SkPDFInt(33));
216 SkAutoTUnref<SkPDFDict> stubResource(new SkPDFDict());
217 SkAutoTUnref<SkPDFInt> int44(new SkPDFInt(44));
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000218
219 stub->insert("Value", int33.get());
220 stubResource->insert("InnerValue", int44.get());
221 stub->addResource(stubResource.get());
222
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000223 SkPDFCatalog catalog((SkPDFDocument::Flags)0);
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000224 catalog.addObject(proxy.get(), false);
225 catalog.setSubstitute(proxy.get(), stub.get());
226
227 SkDynamicMemoryWStream buffer;
228 proxy->emit(&buffer, &catalog, false);
229 catalog.emitSubstituteResources(&buffer, false);
230
vandebo@chromium.orgd3a094c2011-07-25 22:22:25 +0000231 char objectResult[] = "2 0 obj\n<</Value 33\n>>\nendobj\n";
232 REPORTER_ASSERT(
233 reporter,
234 catalog.setFileOffset(proxy.get(), 0) == strlen(objectResult));
235
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000236 char expectedResult[] =
237 "<</Value 33\n>>1 0 obj\n<</InnerValue 44\n>>\nendobj\n";
238 REPORTER_ASSERT(reporter, buffer.getOffset() == strlen(expectedResult));
239 REPORTER_ASSERT(reporter, stream_equals(buffer, 0, expectedResult,
240 buffer.getOffset()));
241}
242
edisonn@google.comd9dfa182013-04-24 13:01:01 +0000243// Create a bitmap that would be easier to be compressed in a JPEG than ZIP.
244static void setup_jpegBitmap(SkBitmap* bitmap, int width, int height) {
245 bitmap->setConfig(SkBitmap::kARGB_8888_Config, width, height);
246 bitmap->allocPixels();
247 for (int y = 0; y < bitmap->height(); y++) {
248 for (int x = 0; x < bitmap->width(); x++) {
249 *bitmap->getAddr32(x, y) =
250 SkColorSetRGB(0 + y % 20 + 128 + 100 * cos(x * 0.01F),
251 1 + y % 20 + 128 + 100 * cos(x * 0.1F),
252 2 + y % 20 + 128 + 100 * cos(x * 1.0F));
253 }
254 }
255}
256
257// Create a bitmap that would be very eficiently compressed in a ZIP.
258static void setup_solidBitmap(SkBitmap* bitmap, int width, int height) {
259 bitmap->setConfig(SkBitmap::kARGB_8888_Config, width, height);
260 bitmap->allocPixels();
261 bitmap->eraseColor(SK_ColorWHITE);
262}
263
264static void TestImage(skiatest::Reporter* reporter, const SkBitmap& bitmap,
265 const char* expected, bool useDCTEncoder) {
266 SkISize pageSize = SkISize::Make(bitmap.width(), bitmap.height());
267 SkPDFDevice* dev = new SkPDFDevice(pageSize, pageSize, SkMatrix::I());
268
269 if (useDCTEncoder) {
270 dev->setDCTEncoder(encode_to_dct_stream);
271 }
272
273 SkCanvas c(dev);
274 c.drawBitmap(bitmap, 0, 0, NULL);
275
276 SkPDFDocument doc;
277 doc.appendPage(dev);
278
279 SkDynamicMemoryWStream stream;
280 doc.emitPDF(&stream);
281
282 REPORTER_ASSERT(reporter, stream_contains(stream, expected));
283}
284
285static void TestUncompressed(skiatest::Reporter* reporter) {
286 SkBitmap bitmap;
287 setup_solidBitmap(&bitmap, 1, 1);
288 TestImage(reporter, bitmap,
289 "/Subtype /Image\n"
290 "/Width 1\n"
291 "/Height 1\n"
292 "/ColorSpace /DeviceRGB\n"
293 "/BitsPerComponent 8\n"
294 "/Length 3\n"
295 ">> stream",
296 true);
297}
298
299static void TestFlateDecode(skiatest::Reporter* reporter) {
300 if (!SkFlate::HaveFlate()) {
301 return;
302 }
303 SkBitmap bitmap;
304 setup_solidBitmap(&bitmap, 10, 10);
305 TestImage(reporter, bitmap,
306 "/Subtype /Image\n"
307 "/Width 10\n"
308 "/Height 10\n"
309 "/ColorSpace /DeviceRGB\n"
310 "/BitsPerComponent 8\n"
311 "/Filter /FlateDecode\n"
312 "/Length 13\n"
313 ">> stream",
314 false);
315}
316
317static void TestDCTDecode(skiatest::Reporter* reporter) {
318 SkBitmap bitmap;
319 setup_jpegBitmap(&bitmap, 32, 32);
320 TestImage(reporter, bitmap,
321 "/Subtype /Image\n"
322 "/Width 32\n"
323 "/Height 32\n"
324 "/ColorSpace /DeviceRGB\n"
325 "/BitsPerComponent 8\n"
326 "/Filter /DCTDecode\n"
327 "/ColorTransform 0\n"
328 "/Length 21\n"
329 ">> stream",
330 true);
331}
332
333static void TestImages(skiatest::Reporter* reporter) {
334 TestUncompressed(reporter);
335 TestFlateDecode(reporter);
336 TestDCTDecode(reporter);
337}
338
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000339// This test used to assert without the fix submitted for
340// http://code.google.com/p/skia/issues/detail?id=1083.
341// SKP files might have invalid glyph ids. This test ensures they are ignored,
342// and there is no assert on input data in Debug mode.
sugoi@google.com54f0d1b2013-02-27 19:17:41 +0000343static void test_issue1083() {
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000344 SkISize pageSize = SkISize::Make(100, 100);
robertphillips@google.com35300c42013-03-21 17:38:49 +0000345 SkAutoTUnref<SkPDFDevice> dev(new SkPDFDevice(pageSize, pageSize, SkMatrix::I()));
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000346
347 SkCanvas c(dev);
348 SkPaint paint;
349 paint.setTextEncoding(SkPaint::kGlyphID_TextEncoding);
350
351 uint16_t glyphID = 65000;
352 c.drawText(&glyphID, 2, 0, 0, paint);
353
354 SkPDFDocument doc;
355 doc.appendPage(dev);
356
357 SkDynamicMemoryWStream stream;
358 doc.emitPDF(&stream);
359}
360
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000361static void TestPDFPrimitives(skiatest::Reporter* reporter) {
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000362 SkAutoTUnref<SkPDFInt> int42(new SkPDFInt(42));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000363 SimpleCheckObjectOutput(reporter, int42.get(), "42");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000364
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000365 SkAutoTUnref<SkPDFScalar> realHalf(new SkPDFScalar(SK_ScalarHalf));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000366 SimpleCheckObjectOutput(reporter, realHalf.get(), "0.5");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000367
vandebo@chromium.org6cc26da2011-05-18 17:08:05 +0000368#if defined(SK_SCALAR_IS_FLOAT)
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000369 SkAutoTUnref<SkPDFScalar> bigScalar(new SkPDFScalar(110999.75f));
vandebo@chromium.org6cc26da2011-05-18 17:08:05 +0000370#if !defined(SK_ALLOW_LARGE_PDF_SCALARS)
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000371 SimpleCheckObjectOutput(reporter, bigScalar.get(), "111000");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000372#else
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000373 SimpleCheckObjectOutput(reporter, bigScalar.get(), "110999.75");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000374
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000375 SkAutoTUnref<SkPDFScalar> biggerScalar(new SkPDFScalar(50000000.1));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000376 SimpleCheckObjectOutput(reporter, biggerScalar.get(), "50000000");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000377
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000378 SkAutoTUnref<SkPDFScalar> smallestScalar(new SkPDFScalar(1.0/65536));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000379 SimpleCheckObjectOutput(reporter, smallestScalar.get(), "0.00001526");
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000380#endif
vandebo@chromium.org6cc26da2011-05-18 17:08:05 +0000381#endif
vandebo@chromium.org094316b2011-03-04 03:15:13 +0000382
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000383 SkAutoTUnref<SkPDFString> stringSimple(
384 new SkPDFString("test ) string ( foo"));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000385 SimpleCheckObjectOutput(reporter, stringSimple.get(),
386 "(test \\) string \\( foo)");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000387 SkAutoTUnref<SkPDFString> stringComplex(
388 new SkPDFString("\ttest ) string ( foo"));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000389 SimpleCheckObjectOutput(reporter, stringComplex.get(),
390 "<0974657374202920737472696E67202820666F6F>");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000391
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000392 SkAutoTUnref<SkPDFName> name(new SkPDFName("Test name\twith#tab"));
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000393 const char expectedResult[] = "/Test#20name#09with#23tab";
394 CheckObjectOutput(reporter, name.get(), expectedResult,
395 strlen(expectedResult), false, false);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000396
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000397 SkAutoTUnref<SkPDFName> escapedName(new SkPDFName("A#/%()<>[]{}B"));
vandebo@chromium.org251a7662012-09-21 17:50:50 +0000398 const char escapedNameExpected[] = "/A#23#2F#25#28#29#3C#3E#5B#5D#7B#7DB";
399 CheckObjectOutput(reporter, escapedName.get(), escapedNameExpected,
400 strlen(escapedNameExpected), false, false);
401
vandebo@chromium.orgc0376fe2012-03-05 18:44:33 +0000402 // Test that we correctly handle characters with the high-bit set.
bungeman@google.comf8aa18c2012-03-19 21:04:52 +0000403 const unsigned char highBitCString[] = {0xDE, 0xAD, 'b', 'e', 0xEF, 0};
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000404 SkAutoTUnref<SkPDFName> highBitName(
405 new SkPDFName((const char*)highBitCString));
vandebo@chromium.orgc0376fe2012-03-05 18:44:33 +0000406 const char highBitExpectedResult[] = "/#DE#ADbe#EF";
407 CheckObjectOutput(reporter, highBitName.get(), highBitExpectedResult,
408 strlen(highBitExpectedResult), false, false);
409
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000410 SkAutoTUnref<SkPDFArray> array(new SkPDFArray);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000411 SimpleCheckObjectOutput(reporter, array.get(), "[]");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000412 array->append(int42.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000413 SimpleCheckObjectOutput(reporter, array.get(), "[42]");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000414 array->append(realHalf.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000415 SimpleCheckObjectOutput(reporter, array.get(), "[42 0.5]");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000416 SkAutoTUnref<SkPDFInt> int0(new SkPDFInt(0));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000417 array->append(int0.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000418 SimpleCheckObjectOutput(reporter, array.get(), "[42 0.5 0]");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000419 SkAutoTUnref<SkPDFInt> int1(new SkPDFInt(1));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000420 array->setAt(0, int1.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000421 SimpleCheckObjectOutput(reporter, array.get(), "[1 0.5 0]");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000422
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000423 SkAutoTUnref<SkPDFDict> dict(new SkPDFDict);
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000424 SimpleCheckObjectOutput(reporter, dict.get(), "<<>>");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000425 SkAutoTUnref<SkPDFName> n1(new SkPDFName("n1"));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000426 dict->insert(n1.get(), int42.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000427 SimpleCheckObjectOutput(reporter, dict.get(), "<</n1 42\n>>");
vandebo@chromium.orgd96d17b2013-01-04 19:31:24 +0000428 SkAutoTUnref<SkPDFName> n2(new SkPDFName("n2"));
429 SkAutoTUnref<SkPDFName> n3(new SkPDFName("n3"));
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000430 dict->insert(n2.get(), realHalf.get());
431 dict->insert(n3.get(), array.get());
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000432 SimpleCheckObjectOutput(reporter, dict.get(),
433 "<</n1 42\n/n2 0.5\n/n3 [1 0.5 0]\n>>");
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000434
vandebo@chromium.org421d6442011-07-20 17:39:01 +0000435 TestPDFStream(reporter);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000436
437 TestCatalog(reporter);
438
439 TestObjectRef(reporter);
vandebo@chromium.org2ef12d42011-07-06 23:31:24 +0000440
441 TestSubstitute(reporter);
vandebo@chromium.org4e1cc6a2013-01-25 19:27:23 +0000442
sugoi@google.com54f0d1b2013-02-27 19:17:41 +0000443 test_issue1083();
edisonn@google.comd9dfa182013-04-24 13:01:01 +0000444
445 TestImages(reporter);
vandebo@chromium.org8459d4e2010-09-24 22:25:30 +0000446}
447
448#include "TestClassDef.h"
449DEFINE_TESTCLASS("PDFPrimitives", PDFPrimitivesTestClass, TestPDFPrimitives)