blob: d673ceb84424108d32bd91f838833ac62ba2d006 [file] [log] [blame]
reed@google.com3d608122011-11-21 15:16:16 +00001/*
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
Herb Derby73fe7b02017-02-08 15:12:19 -05008#include "SkArenaAlloc.h"
Florin Malita39e08552017-07-06 14:16:18 -04009#include "SkBitmap.h"
reed@google.com3d608122011-11-21 15:16:16 +000010#include "SkCanvas.h"
Matt Sarettcdc651d2017-03-30 12:41:48 -040011#include "SkColorSpaceXformer.h"
reed@google.com3d608122011-11-21 15:16:16 +000012#include "SkDrawLooper.h"
Cary Clark4dc5a452018-05-21 11:56:57 -040013#include "SkFlattenablePriv.h"
msarett9da5a5a2016-08-19 08:38:36 -070014#include "SkLightingImageFilter.h"
Mike Reed54518ac2017-07-22 08:29:48 -040015#include "SkPoint3.h"
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +000016#include "SkTypes.h"
tfarina@chromium.org8f6884a2014-01-24 20:56:26 +000017#include "Test.h"
reed@google.com3d608122011-11-21 15:16:16 +000018
19/*
20 * Subclass of looper that just draws once, with an offset in X.
21 */
22class TestLooper : public SkDrawLooper {
23public:
reed@google.com3d608122011-11-21 15:16:16 +000024
Herb Derby73fe7b02017-02-08 15:12:19 -050025 SkDrawLooper::Context* makeContext(SkCanvas*, SkArenaAlloc* alloc) const override {
26 return alloc->make<TestDrawLooperContext>();
reed@google.com3d608122011-11-21 15:16:16 +000027 }
28
Matt Sarettcdc651d2017-03-30 12:41:48 -040029 sk_sp<SkDrawLooper> onMakeColorSpace(SkColorSpaceXformer*) const override {
30 return nullptr;
31 }
32
Cary Clark32a49102018-05-20 23:15:43 +000033 void toString(SkString* str) const override {
34 str->append("TestLooper:");
35 }
36
Mike Kleinfc6c37b2016-09-27 09:34:10 -040037 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(TestLooper)
mtklein7e44bb12015-01-07 09:06:08 -080038
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +000039private:
40 class TestDrawLooperContext : public SkDrawLooper::Context {
41 public:
42 TestDrawLooperContext() : fOnce(true) {}
Brian Salomond3b65972017-03-22 12:05:03 -040043 ~TestDrawLooperContext() override {}
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +000044
mtklein36352bf2015-03-25 18:17:31 -070045 bool next(SkCanvas* canvas, SkPaint*) override {
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +000046 if (fOnce) {
47 fOnce = false;
48 canvas->translate(SkIntToScalar(10), 0);
49 return true;
50 }
51 return false;
52 }
Matt Sarettcdc651d2017-03-30 12:41:48 -040053
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +000054 private:
55 bool fOnce;
56 };
reed@google.com3d608122011-11-21 15:16:16 +000057};
58
reed60c9b582016-04-03 09:11:13 -070059sk_sp<SkFlattenable> TestLooper::CreateProc(SkReadBuffer&) { return sk_make_sp<TestLooper>(); }
mtklein7e44bb12015-01-07 09:06:08 -080060
reed@google.com3d608122011-11-21 15:16:16 +000061static void test_drawBitmap(skiatest::Reporter* reporter) {
62 SkBitmap src;
mike@reedtribe.orgdeee4962014-02-13 14:41:43 +000063 src.allocN32Pixels(10, 10);
reed@google.com3d608122011-11-21 15:16:16 +000064 src.eraseColor(SK_ColorWHITE);
65
66 SkBitmap dst;
mike@reedtribe.orgdeee4962014-02-13 14:41:43 +000067 dst.allocN32Pixels(10, 10);
junov@google.comdbfac8a2012-12-06 21:47:40 +000068 dst.eraseColor(SK_ColorTRANSPARENT);
reed@google.com3d608122011-11-21 15:16:16 +000069
70 SkCanvas canvas(dst);
71 SkPaint paint;
72
73 // we are initially transparent
74 REPORTER_ASSERT(reporter, 0 == *dst.getAddr32(5, 5));
75
76 // we see the bitmap drawn
77 canvas.drawBitmap(src, 0, 0, &paint);
78 REPORTER_ASSERT(reporter, 0xFFFFFFFF == *dst.getAddr32(5, 5));
79
80 // reverify we are clear again
junov@google.comdbfac8a2012-12-06 21:47:40 +000081 dst.eraseColor(SK_ColorTRANSPARENT);
reed@google.com3d608122011-11-21 15:16:16 +000082 REPORTER_ASSERT(reporter, 0 == *dst.getAddr32(5, 5));
83
84 // if the bitmap is clipped out, we don't draw it
85 canvas.drawBitmap(src, SkIntToScalar(-10), 0, &paint);
86 REPORTER_ASSERT(reporter, 0 == *dst.getAddr32(5, 5));
87
88 // now install our looper, which will draw, since it internally translates
89 // to the left. The test is to ensure that canvas' quickReject machinary
90 // allows us through, even though sans-looper we would look like we should
91 // be clipped out.
reed7b380d02016-03-21 13:25:16 -070092 paint.setLooper(sk_make_sp<TestLooper>());
reed@google.com3d608122011-11-21 15:16:16 +000093 canvas.drawBitmap(src, SkIntToScalar(-10), 0, &paint);
94 REPORTER_ASSERT(reporter, 0xFFFFFFFF == *dst.getAddr32(5, 5));
95}
96
reed9b3aa542015-03-11 08:47:12 -070097static void test_layers(skiatest::Reporter* reporter) {
98 SkCanvas canvas(100, 100);
99
100 SkRect r = SkRect::MakeWH(10, 10);
101 REPORTER_ASSERT(reporter, false == canvas.quickReject(r));
102
103 r.offset(300, 300);
104 REPORTER_ASSERT(reporter, true == canvas.quickReject(r));
105
106 // Test that saveLayer updates quickReject
107 SkRect bounds = SkRect::MakeLTRB(50, 50, 70, 70);
halcanary96fcdcc2015-08-27 07:41:13 -0700108 canvas.saveLayer(&bounds, nullptr);
reed9b3aa542015-03-11 08:47:12 -0700109 REPORTER_ASSERT(reporter, true == canvas.quickReject(SkRect::MakeWH(10, 10)));
110 REPORTER_ASSERT(reporter, false == canvas.quickReject(SkRect::MakeWH(60, 60)));
111}
112
msarettfbfa2582016-08-12 08:29:08 -0700113static void test_quick_reject(skiatest::Reporter* reporter) {
114 SkCanvas canvas(100, 100);
115 SkRect r0 = SkRect::MakeLTRB(-50.0f, -50.0f, 50.0f, 50.0f);
116 SkRect r1 = SkRect::MakeLTRB(-50.0f, 110.0f, 50.0f, 120.0f);
117 SkRect r2 = SkRect::MakeLTRB(110.0f, -50.0f, 120.0f, 50.0f);
118 SkRect r3 = SkRect::MakeLTRB(-120.0f, -50.0f, 120.0f, 50.0f);
119 SkRect r4 = SkRect::MakeLTRB(-50.0f, -120.0f, 50.0f, 120.0f);
120 SkRect r5 = SkRect::MakeLTRB(-120.0f, -120.0f, 120.0f, 120.0f);
121 SkRect r6 = SkRect::MakeLTRB(-120.0f, -120.0f, -110.0f, -110.0f);
122 SkRect r7 = SkRect::MakeLTRB(SK_ScalarNaN, -50.0f, 50.0f, 50.0f);
123 SkRect r8 = SkRect::MakeLTRB(-50.0f, SK_ScalarNaN, 50.0f, 50.0f);
124 SkRect r9 = SkRect::MakeLTRB(-50.0f, -50.0f, SK_ScalarNaN, 50.0f);
125 SkRect r10 = SkRect::MakeLTRB(-50.0f, -50.0f, 50.0f, SK_ScalarNaN);
126 REPORTER_ASSERT(reporter, false == canvas.quickReject(r0));
127 REPORTER_ASSERT(reporter, true == canvas.quickReject(r1));
128 REPORTER_ASSERT(reporter, true == canvas.quickReject(r2));
129 REPORTER_ASSERT(reporter, false == canvas.quickReject(r3));
130 REPORTER_ASSERT(reporter, false == canvas.quickReject(r4));
131 REPORTER_ASSERT(reporter, false == canvas.quickReject(r5));
132 REPORTER_ASSERT(reporter, true == canvas.quickReject(r6));
133 REPORTER_ASSERT(reporter, true == canvas.quickReject(r7));
134 REPORTER_ASSERT(reporter, true == canvas.quickReject(r8));
135 REPORTER_ASSERT(reporter, true == canvas.quickReject(r9));
136 REPORTER_ASSERT(reporter, true == canvas.quickReject(r10));
137
138 SkMatrix m = SkMatrix::MakeScale(2.0f);
139 m.setTranslateX(10.0f);
140 m.setTranslateY(10.0f);
141 canvas.setMatrix(m);
142 SkRect r11 = SkRect::MakeLTRB(5.0f, 5.0f, 100.0f, 100.0f);
143 SkRect r12 = SkRect::MakeLTRB(5.0f, 50.0f, 100.0f, 100.0f);
144 SkRect r13 = SkRect::MakeLTRB(50.0f, 5.0f, 100.0f, 100.0f);
145 REPORTER_ASSERT(reporter, false == canvas.quickReject(r11));
146 REPORTER_ASSERT(reporter, true == canvas.quickReject(r12));
147 REPORTER_ASSERT(reporter, true == canvas.quickReject(r13));
148}
149
tfarina@chromium.orge4fafb12013-12-12 21:11:12 +0000150DEF_TEST(QuickReject, reporter) {
reed@google.com3d608122011-11-21 15:16:16 +0000151 test_drawBitmap(reporter);
reed9b3aa542015-03-11 08:47:12 -0700152 test_layers(reporter);
msarettfbfa2582016-08-12 08:29:08 -0700153 test_quick_reject(reporter);
reed@google.com3d608122011-11-21 15:16:16 +0000154}
msarett9da5a5a2016-08-19 08:38:36 -0700155
156// Regression test to make sure that we keep fIsScaleTranslate up to date on the canvas.
157// It is possible to set a new matrix on the canvas without calling setMatrix(). This tests
158// that code path.
159DEF_TEST(QuickReject_MatrixState, reporter) {
160 SkCanvas canvas(100, 100);
161
162 SkMatrix matrix;
163 matrix.setRotate(45.0f);
164 canvas.setMatrix(matrix);
165
166 SkPaint paint;
167 sk_sp<SkImageFilter> filter = SkLightingImageFilter::MakeDistantLitDiffuse(
168 SkPoint3::Make(1.0f, 1.0f, 1.0f), 0xFF0000FF, 2.0f, 0.5f, nullptr);
169 REPORTER_ASSERT(reporter, filter);
170 paint.setImageFilter(filter);
171 SkCanvas::SaveLayerRec rec;
172 rec.fPaint = &paint;
173 canvas.saveLayer(rec);
174
175 // quickReject() will assert if the matrix is out of sync.
176 canvas.quickReject(SkRect::MakeWH(100.0f, 100.0f));
177}
Mike Reed59af19f2018-04-12 17:26:40 -0400178
179#include "SkLayerDrawLooper.h"
180#include "SkSurface.h"
181DEF_TEST(looper_nothingtodraw, reporter) {
182 auto surf = SkSurface::MakeRasterN32Premul(20, 20);
183
184 SkPaint paint;
185 paint.setColor(0);
186 REPORTER_ASSERT(reporter, paint.nothingToDraw());
187
188 SkLayerDrawLooper::Builder builder;
189 builder.addLayer();
190 paint.setDrawLooper(builder.detach());
191 // the presence of the looper fools this predicate, so we think it might draw
192 REPORTER_ASSERT(reporter, !paint.nothingToDraw());
193
194 // Before fixing, this would assert in ~AutoDrawLooper() in SkCanvas.cpp as it checked for
195 // a balance in the save/restore count after handling the looper. Before the fix, this
196 // code would call nothingToDraw() and since it now clears the looper, that predicate will
197 // return true, aborting the sequence prematurely, and not finishing the iterator on the looper
198 // which handles the final "restore". This was a bug -- we *must* call the looper's iterator
199 // until it returns done to keep the canvas balanced. The fix was to remove this early-exit
200 // in the autodrawlooper. Now this call won't assert.
201 // See https://skia-review.googlesource.com/c/skia/+/121220
202 surf->getCanvas()->drawRect({1, 1, 10, 10}, paint);
203}