1 /*
2 * Copyright 2011 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 "SkDrawLooper.h"
10 #include "SkTypes.h"
11 #include "Test.h"
12
13 /*
14 * Subclass of looper that just draws once, with an offset in X.
15 */
16 class TestLooper : public SkDrawLooper {
17 public:
18
createContext(SkCanvas *,void * storage) const19 SkDrawLooper::Context* createContext(SkCanvas*, void* storage) const override {
20 return SkNEW_PLACEMENT(storage, TestDrawLooperContext);
21 }
22
contextSize() const23 size_t contextSize() const override { return sizeof(TestDrawLooperContext); }
24
25 #ifndef SK_IGNORE_TO_STRING
toString(SkString * str) const26 void toString(SkString* str) const override {
27 str->append("TestLooper:");
28 }
29 #endif
30
31 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(TestLooper);
32
33 private:
34 class TestDrawLooperContext : public SkDrawLooper::Context {
35 public:
TestDrawLooperContext()36 TestDrawLooperContext() : fOnce(true) {}
~TestDrawLooperContext()37 virtual ~TestDrawLooperContext() {}
38
next(SkCanvas * canvas,SkPaint *)39 bool next(SkCanvas* canvas, SkPaint*) override {
40 if (fOnce) {
41 fOnce = false;
42 canvas->translate(SkIntToScalar(10), 0);
43 return true;
44 }
45 return false;
46 }
47 private:
48 bool fOnce;
49 };
50 };
51
CreateProc(SkReadBuffer &)52 SkFlattenable* TestLooper::CreateProc(SkReadBuffer&) { return SkNEW(TestLooper); }
53
test_drawBitmap(skiatest::Reporter * reporter)54 static void test_drawBitmap(skiatest::Reporter* reporter) {
55 SkBitmap src;
56 src.allocN32Pixels(10, 10);
57 src.eraseColor(SK_ColorWHITE);
58
59 SkBitmap dst;
60 dst.allocN32Pixels(10, 10);
61 dst.eraseColor(SK_ColorTRANSPARENT);
62
63 SkCanvas canvas(dst);
64 SkPaint paint;
65
66 // we are initially transparent
67 REPORTER_ASSERT(reporter, 0 == *dst.getAddr32(5, 5));
68
69 // we see the bitmap drawn
70 canvas.drawBitmap(src, 0, 0, &paint);
71 REPORTER_ASSERT(reporter, 0xFFFFFFFF == *dst.getAddr32(5, 5));
72
73 // reverify we are clear again
74 dst.eraseColor(SK_ColorTRANSPARENT);
75 REPORTER_ASSERT(reporter, 0 == *dst.getAddr32(5, 5));
76
77 // if the bitmap is clipped out, we don't draw it
78 canvas.drawBitmap(src, SkIntToScalar(-10), 0, &paint);
79 REPORTER_ASSERT(reporter, 0 == *dst.getAddr32(5, 5));
80
81 // now install our looper, which will draw, since it internally translates
82 // to the left. The test is to ensure that canvas' quickReject machinary
83 // allows us through, even though sans-looper we would look like we should
84 // be clipped out.
85 paint.setLooper(new TestLooper)->unref();
86 canvas.drawBitmap(src, SkIntToScalar(-10), 0, &paint);
87 REPORTER_ASSERT(reporter, 0xFFFFFFFF == *dst.getAddr32(5, 5));
88 }
89
test_layers(skiatest::Reporter * reporter)90 static void test_layers(skiatest::Reporter* reporter) {
91 SkCanvas canvas(100, 100);
92
93 SkRect r = SkRect::MakeWH(10, 10);
94 REPORTER_ASSERT(reporter, false == canvas.quickReject(r));
95
96 r.offset(300, 300);
97 REPORTER_ASSERT(reporter, true == canvas.quickReject(r));
98
99 // Test that saveLayer updates quickReject
100 SkRect bounds = SkRect::MakeLTRB(50, 50, 70, 70);
101 canvas.saveLayer(&bounds, NULL);
102 REPORTER_ASSERT(reporter, true == canvas.quickReject(SkRect::MakeWH(10, 10)));
103 REPORTER_ASSERT(reporter, false == canvas.quickReject(SkRect::MakeWH(60, 60)));
104 }
105
DEF_TEST(QuickReject,reporter)106 DEF_TEST(QuickReject, reporter) {
107 test_drawBitmap(reporter);
108 test_layers(reporter);
109 }
110