1 /*
2 * Copyright 2016 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8 #include "SkCanvas.h"
9 #include "SkDrawable.h"
10 #include "SkFont.h"
11 #include "SkPictureRecorder.h"
12 #include "SkReadBuffer.h"
13 #include "SkRect.h"
14 #include "SkStream.h"
15 #include "SkWriteBuffer.h"
16 #include "Test.h"
17
18 class IntDrawable : public SkDrawable {
19 public:
IntDrawable(uint32_t a,uint32_t b,uint32_t c,uint32_t d)20 IntDrawable(uint32_t a, uint32_t b, uint32_t c, uint32_t d)
21 : fA(a)
22 , fB(b)
23 , fC(c)
24 , fD(d)
25 {}
26
flatten(SkWriteBuffer & buffer) const27 void flatten(SkWriteBuffer& buffer) const override {
28 buffer.writeUInt(fA);
29 buffer.writeUInt(fB);
30 buffer.writeUInt(fC);
31 buffer.writeUInt(fD);
32 }
33
CreateProc(SkReadBuffer & buffer)34 static sk_sp<SkFlattenable> CreateProc(SkReadBuffer& buffer) {
35 uint32_t a = buffer.readUInt();
36 uint32_t b = buffer.readUInt();
37 uint32_t c = buffer.readUInt();
38 uint32_t d = buffer.readUInt();
39 return sk_sp<IntDrawable>(new IntDrawable(a, b, c, d));
40 }
41
getFactory() const42 Factory getFactory() const override { return CreateProc; }
43
a() const44 uint32_t a() const { return fA; }
b() const45 uint32_t b() const { return fB; }
c() const46 uint32_t c() const { return fC; }
d() const47 uint32_t d() const { return fD; }
48
getTypeName() const49 const char* getTypeName() const override { return "IntDrawable"; }
50
51 protected:
onGetBounds()52 SkRect onGetBounds() override { return SkRect::MakeEmpty(); }
onDraw(SkCanvas *)53 void onDraw(SkCanvas*) override {}
54
55 private:
56 uint32_t fA;
57 uint32_t fB;
58 uint32_t fC;
59 uint32_t fD;
60 };
61
62 class PaintDrawable : public SkDrawable {
63 public:
PaintDrawable(const SkPaint & paint)64 PaintDrawable(const SkPaint& paint)
65 : fPaint(paint)
66 {}
67
flatten(SkWriteBuffer & buffer) const68 void flatten(SkWriteBuffer& buffer) const override {
69 buffer.writePaint(fPaint);
70 }
71
CreateProc(SkReadBuffer & buffer)72 static sk_sp<SkFlattenable> CreateProc(SkReadBuffer& buffer) {
73 SkPaint paint;
74 buffer.readPaint(&paint, nullptr);
75 return sk_sp<PaintDrawable>(new PaintDrawable(paint));
76 }
77
getFactory() const78 Factory getFactory() const override { return CreateProc; }
79
paint() const80 const SkPaint& paint() const { return fPaint; }
81
getTypeName() const82 const char* getTypeName() const override { return "PaintDrawable"; }
83
84 protected:
onGetBounds()85 SkRect onGetBounds() override { return SkRect::MakeEmpty(); }
onDraw(SkCanvas *)86 void onDraw(SkCanvas*) override {}
87
88 private:
89 SkPaint fPaint;
90 };
91
92 class CompoundDrawable : public SkDrawable {
93 public:
CompoundDrawable(uint32_t a,uint32_t b,uint32_t c,uint32_t d,const SkPaint & paint)94 CompoundDrawable(uint32_t a, uint32_t b, uint32_t c, uint32_t d, const SkPaint& paint)
95 : fIntDrawable(new IntDrawable(a, b, c, d))
96 , fPaintDrawable(new PaintDrawable(paint))
97 {}
98
CompoundDrawable(IntDrawable * intDrawable,PaintDrawable * paintDrawable)99 CompoundDrawable(IntDrawable* intDrawable, PaintDrawable* paintDrawable)
100 : fIntDrawable(SkRef(intDrawable))
101 , fPaintDrawable(SkRef(paintDrawable))
102 {}
103
flatten(SkWriteBuffer & buffer) const104 void flatten(SkWriteBuffer& buffer) const override {
105 buffer.writeFlattenable(fIntDrawable.get());
106 buffer.writeFlattenable(fPaintDrawable.get());
107 }
108
CreateProc(SkReadBuffer & buffer)109 static sk_sp<SkFlattenable> CreateProc(SkReadBuffer& buffer) {
110 sk_sp<SkFlattenable> intDrawable(
111 buffer.readFlattenable(SkFlattenable::kSkDrawable_Type));
112 SkASSERT(intDrawable);
113 SkASSERT(!strcmp("IntDrawable", intDrawable->getTypeName()));
114
115 sk_sp<SkFlattenable> paintDrawable(
116 buffer.readFlattenable(SkFlattenable::kSkDrawable_Type));
117 SkASSERT(paintDrawable);
118 SkASSERT(!strcmp("PaintDrawable", paintDrawable->getTypeName()));
119
120 return sk_sp<CompoundDrawable>(new CompoundDrawable((IntDrawable*) intDrawable.get(),
121 (PaintDrawable*) paintDrawable.get()));
122 }
123
getFactory() const124 Factory getFactory() const override { return CreateProc; }
125
intDrawable() const126 IntDrawable* intDrawable() const { return fIntDrawable.get(); }
paintDrawable() const127 PaintDrawable* paintDrawable() const { return fPaintDrawable.get(); }
128
getTypeName() const129 const char* getTypeName() const override { return "CompoundDrawable"; }
130
131 protected:
onGetBounds()132 SkRect onGetBounds() override { return SkRect::MakeEmpty(); }
onDraw(SkCanvas *)133 void onDraw(SkCanvas*) override {}
134
135 private:
136 sk_sp<IntDrawable> fIntDrawable;
137 sk_sp<PaintDrawable> fPaintDrawable;
138 };
139
140 class RootDrawable : public SkDrawable {
141 public:
RootDrawable(uint32_t a,uint32_t b,uint32_t c,uint32_t d,const SkPaint & paint,uint32_t e,uint32_t f,uint32_t g,uint32_t h,SkDrawable * drawable)142 RootDrawable(uint32_t a, uint32_t b, uint32_t c, uint32_t d, const SkPaint& paint,
143 uint32_t e, uint32_t f, uint32_t g, uint32_t h, SkDrawable* drawable)
144 : fCompoundDrawable(new CompoundDrawable(a, b, c, d, paint))
145 , fIntDrawable(new IntDrawable(e, f, g, h))
146 , fDrawable(SkRef(drawable))
147 {}
148
RootDrawable(CompoundDrawable * compoundDrawable,IntDrawable * intDrawable,SkDrawable * drawable)149 RootDrawable(CompoundDrawable* compoundDrawable, IntDrawable* intDrawable,
150 SkDrawable* drawable)
151 : fCompoundDrawable(SkRef(compoundDrawable))
152 , fIntDrawable(SkRef(intDrawable))
153 , fDrawable(SkRef(drawable))
154 {}
155
flatten(SkWriteBuffer & buffer) const156 void flatten(SkWriteBuffer& buffer) const override {
157 buffer.writeFlattenable(fCompoundDrawable.get());
158 buffer.writeFlattenable(fIntDrawable.get());
159 buffer.writeFlattenable(fDrawable.get());
160 }
161
CreateProc(SkReadBuffer & buffer)162 static sk_sp<SkFlattenable> CreateProc(SkReadBuffer& buffer) {
163 sk_sp<SkFlattenable> compoundDrawable(
164 buffer.readFlattenable(SkFlattenable::kSkDrawable_Type));
165 SkASSERT(compoundDrawable);
166 SkASSERT(!strcmp("CompoundDrawable", compoundDrawable->getTypeName()));
167
168 sk_sp<SkFlattenable> intDrawable(
169 buffer.readFlattenable(SkFlattenable::kSkDrawable_Type));
170 SkASSERT(intDrawable);
171 SkASSERT(!strcmp("IntDrawable", intDrawable->getTypeName()));
172
173 sk_sp<SkFlattenable> drawable(
174 buffer.readFlattenable(SkFlattenable::kSkDrawable_Type));
175 SkASSERT(drawable);
176
177 return sk_sp<RootDrawable>(new RootDrawable((CompoundDrawable*) compoundDrawable.get(),
178 (IntDrawable*) intDrawable.get(),
179 (SkDrawable*) drawable.get()));
180 }
181
getFactory() const182 Factory getFactory() const override { return CreateProc; }
183
compoundDrawable() const184 CompoundDrawable* compoundDrawable() const { return fCompoundDrawable.get(); }
intDrawable() const185 IntDrawable* intDrawable() const { return fIntDrawable.get(); }
drawable() const186 SkDrawable* drawable() const { return fDrawable.get(); }
187
getTypeName() const188 const char* getTypeName() const override { return "RootDrawable"; }
189
190 protected:
onGetBounds()191 SkRect onGetBounds() override { return SkRect::MakeEmpty(); }
onDraw(SkCanvas *)192 void onDraw(SkCanvas*) override {}
193
194 private:
195 sk_sp<CompoundDrawable> fCompoundDrawable;
196 sk_sp<IntDrawable> fIntDrawable;
197 sk_sp<SkDrawable> fDrawable;
198 };
199
200 // Register these drawables for deserialization some time before main().
201 static struct Initializer {
InitializerInitializer202 Initializer() {
203 SK_REGISTER_FLATTENABLE(IntDrawable);
204 SK_REGISTER_FLATTENABLE(PaintDrawable);
205 SK_REGISTER_FLATTENABLE(CompoundDrawable);
206 SK_REGISTER_FLATTENABLE(RootDrawable);
207 }
208 } initializer;
209
DEF_TEST(FlattenDrawable,r)210 DEF_TEST(FlattenDrawable, r) {
211 // Create and serialize the test drawable
212 sk_sp<SkDrawable> drawable(new IntDrawable(1, 2, 3, 4));
213 SkPaint paint;
214 paint.setColor(SK_ColorBLUE);
215 sk_sp<RootDrawable> root(new RootDrawable(5, 6, 7, 8, paint, 9, 10, 11, 12, drawable.get()));
216 SkBinaryWriteBuffer writeBuffer;
217 writeBuffer.writeFlattenable(root.get());
218
219 // Copy the contents of the write buffer into a read buffer
220 sk_sp<SkData> data = SkData::MakeUninitialized(writeBuffer.bytesWritten());
221 writeBuffer.writeToMemory(data->writable_data());
222 SkReadBuffer readBuffer(data->data(), data->size());
223
224 // Deserialize and verify the drawable
225 sk_sp<SkDrawable> out((SkDrawable*)readBuffer.readFlattenable(SkFlattenable::kSkDrawable_Type));
226 REPORTER_ASSERT(r, out);
227 REPORTER_ASSERT(r, !strcmp("RootDrawable", out->getTypeName()));
228
229 RootDrawable* rootOut = (RootDrawable*) out.get();
230 REPORTER_ASSERT(r, 5 == rootOut->compoundDrawable()->intDrawable()->a());
231 REPORTER_ASSERT(r, 6 == rootOut->compoundDrawable()->intDrawable()->b());
232 REPORTER_ASSERT(r, 7 == rootOut->compoundDrawable()->intDrawable()->c());
233 REPORTER_ASSERT(r, 8 == rootOut->compoundDrawable()->intDrawable()->d());
234 REPORTER_ASSERT(r, SK_ColorBLUE ==
235 rootOut->compoundDrawable()->paintDrawable()->paint().getColor());
236 REPORTER_ASSERT(r, 9 == rootOut->intDrawable()->a());
237 REPORTER_ASSERT(r, 10 == rootOut->intDrawable()->b());
238 REPORTER_ASSERT(r, 11 == rootOut->intDrawable()->c());
239 REPORTER_ASSERT(r, 12 == rootOut->intDrawable()->d());
240
241 // Note that we can still recognize the generic drawable as an IntDrawable
242 SkDrawable* generic = rootOut->drawable();
243 REPORTER_ASSERT(r, !strcmp("IntDrawable", generic->getTypeName()));
244 IntDrawable* integer = (IntDrawable*) generic;
245 REPORTER_ASSERT(r, 1 == integer->a());
246 REPORTER_ASSERT(r, 2 == integer->b());
247 REPORTER_ASSERT(r, 3 == integer->c());
248 REPORTER_ASSERT(r, 4 == integer->d());
249 }
250
DEF_TEST(FlattenRecordedDrawable,r)251 DEF_TEST(FlattenRecordedDrawable, r) {
252 // Record a set of canvas draw commands
253 SkPictureRecorder recorder;
254 SkCanvas* canvas = recorder.beginRecording(1000.0f, 1000.0f);
255 SkPaint paint;
256 paint.setColor(SK_ColorGREEN);
257 canvas->drawPoint(42.0f, 17.0f, paint);
258 paint.setColor(SK_ColorRED);
259 canvas->drawPaint(paint);
260 SkPaint textPaint;
261 textPaint.setColor(SK_ColorBLUE);
262 canvas->drawString("TEXT", 467.0f, 100.0f, SkFont(), textPaint);
263
264 // Draw some drawables as well
265 sk_sp<SkDrawable> drawable(new IntDrawable(1, 2, 3, 4));
266 sk_sp<RootDrawable> root(new RootDrawable(5, 6, 7, 8, paint, 9, 10, 11, 12, drawable.get()));
267 canvas->drawDrawable(root.get(), 747.0f, 242.0f);
268 sk_sp<PaintDrawable> paintDrawable(new PaintDrawable(paint));
269 canvas->drawDrawable(paintDrawable.get(), 500.0, 500.0f);
270 sk_sp<CompoundDrawable> comDrawable(new CompoundDrawable(13, 14, 15, 16, textPaint));
271 canvas->drawDrawable(comDrawable.get(), 10.0f, 10.0f);
272
273 // Serialize the recorded drawable
274 sk_sp<SkDrawable> recordedDrawable = recorder.finishRecordingAsDrawable();
275 SkBinaryWriteBuffer writeBuffer;
276 writeBuffer.writeFlattenable(recordedDrawable.get());
277
278 // Copy the contents of the write buffer into a read buffer
279 sk_sp<SkData> data = SkData::MakeUninitialized(writeBuffer.bytesWritten());
280 writeBuffer.writeToMemory(data->writable_data());
281 SkReadBuffer readBuffer(data->data(), data->size());
282
283 // Deserialize and verify the drawable
284 sk_sp<SkDrawable> out((SkDrawable*)readBuffer.readFlattenable(SkFlattenable::kSkDrawable_Type));
285 REPORTER_ASSERT(r, out);
286 REPORTER_ASSERT(r, !strcmp("SkRecordedDrawable", out->getTypeName()));
287 }
288
289 // be sure these constructs compile, don't assert, and return null
DEF_TEST(Flattenable_EmptyDeserialze,reporter)290 DEF_TEST(Flattenable_EmptyDeserialze, reporter) {
291 auto data = SkData::MakeEmpty();
292
293 #define test(name) REPORTER_ASSERT(reporter, !name::Deserialize(data->data(), data->size()))
294 test(SkPathEffect);
295 test(SkMaskFilter);
296 test(SkShaderBase); // todo: make this just be shader!
297 test(SkColorFilter);
298 test(SkImageFilter);
299 test(SkDrawLooper);
300 #undef test
301 }
302
303