blob: 8f6556bcd6fba7c35ee3dc26fdb5a1d3eab4fc77 [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
8#include "SkCanvas.h"
9#include "SkDrawLooper.h"
msarett9da5a5a2016-08-19 08:38:36 -070010#include "SkLightingImageFilter.h"
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +000011#include "SkTypes.h"
tfarina@chromium.org8f6884a2014-01-24 20:56:26 +000012#include "Test.h"
reed@google.com3d608122011-11-21 15:16:16 +000013
14/*
15 * Subclass of looper that just draws once, with an offset in X.
16 */
17class TestLooper : public SkDrawLooper {
18public:
reed@google.com3d608122011-11-21 15:16:16 +000019
mtklein36352bf2015-03-25 18:17:31 -070020 SkDrawLooper::Context* createContext(SkCanvas*, void* storage) const override {
halcanary385fe4d2015-08-26 13:07:48 -070021 return new (storage) TestDrawLooperContext;
reed@google.com3d608122011-11-21 15:16:16 +000022 }
23
mtklein36352bf2015-03-25 18:17:31 -070024 size_t contextSize() const override { return sizeof(TestDrawLooperContext); }
reed@google.com3d608122011-11-21 15:16:16 +000025
commit-bot@chromium.org0f10f7b2014-03-13 18:02:17 +000026#ifndef SK_IGNORE_TO_STRING
mtklein36352bf2015-03-25 18:17:31 -070027 void toString(SkString* str) const override {
robertphillips@google.com4991b8f2013-01-28 20:21:59 +000028 str->append("TestLooper:");
29 }
30#endif
31
mtklein7e44bb12015-01-07 09:06:08 -080032 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(TestLooper);
33
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +000034private:
35 class TestDrawLooperContext : public SkDrawLooper::Context {
36 public:
37 TestDrawLooperContext() : fOnce(true) {}
38 virtual ~TestDrawLooperContext() {}
39
mtklein36352bf2015-03-25 18:17:31 -070040 bool next(SkCanvas* canvas, SkPaint*) override {
commit-bot@chromium.org79fbb402014-03-12 09:42:01 +000041 if (fOnce) {
42 fOnce = false;
43 canvas->translate(SkIntToScalar(10), 0);
44 return true;
45 }
46 return false;
47 }
48 private:
49 bool fOnce;
50 };
reed@google.com3d608122011-11-21 15:16:16 +000051};
52
reed60c9b582016-04-03 09:11:13 -070053sk_sp<SkFlattenable> TestLooper::CreateProc(SkReadBuffer&) { return sk_make_sp<TestLooper>(); }
mtklein7e44bb12015-01-07 09:06:08 -080054
reed@google.com3d608122011-11-21 15:16:16 +000055static void test_drawBitmap(skiatest::Reporter* reporter) {
56 SkBitmap src;
mike@reedtribe.orgdeee4962014-02-13 14:41:43 +000057 src.allocN32Pixels(10, 10);
reed@google.com3d608122011-11-21 15:16:16 +000058 src.eraseColor(SK_ColorWHITE);
59
60 SkBitmap dst;
mike@reedtribe.orgdeee4962014-02-13 14:41:43 +000061 dst.allocN32Pixels(10, 10);
junov@google.comdbfac8a2012-12-06 21:47:40 +000062 dst.eraseColor(SK_ColorTRANSPARENT);
reed@google.com3d608122011-11-21 15:16:16 +000063
64 SkCanvas canvas(dst);
65 SkPaint paint;
66
67 // we are initially transparent
68 REPORTER_ASSERT(reporter, 0 == *dst.getAddr32(5, 5));
69
70 // we see the bitmap drawn
71 canvas.drawBitmap(src, 0, 0, &paint);
72 REPORTER_ASSERT(reporter, 0xFFFFFFFF == *dst.getAddr32(5, 5));
73
74 // reverify we are clear again
junov@google.comdbfac8a2012-12-06 21:47:40 +000075 dst.eraseColor(SK_ColorTRANSPARENT);
reed@google.com3d608122011-11-21 15:16:16 +000076 REPORTER_ASSERT(reporter, 0 == *dst.getAddr32(5, 5));
77
78 // if the bitmap is clipped out, we don't draw it
79 canvas.drawBitmap(src, SkIntToScalar(-10), 0, &paint);
80 REPORTER_ASSERT(reporter, 0 == *dst.getAddr32(5, 5));
81
82 // now install our looper, which will draw, since it internally translates
83 // to the left. The test is to ensure that canvas' quickReject machinary
84 // allows us through, even though sans-looper we would look like we should
85 // be clipped out.
reed7b380d02016-03-21 13:25:16 -070086 paint.setLooper(sk_make_sp<TestLooper>());
reed@google.com3d608122011-11-21 15:16:16 +000087 canvas.drawBitmap(src, SkIntToScalar(-10), 0, &paint);
88 REPORTER_ASSERT(reporter, 0xFFFFFFFF == *dst.getAddr32(5, 5));
89}
90
reed9b3aa542015-03-11 08:47:12 -070091static void test_layers(skiatest::Reporter* reporter) {
92 SkCanvas canvas(100, 100);
93
94 SkRect r = SkRect::MakeWH(10, 10);
95 REPORTER_ASSERT(reporter, false == canvas.quickReject(r));
96
97 r.offset(300, 300);
98 REPORTER_ASSERT(reporter, true == canvas.quickReject(r));
99
100 // Test that saveLayer updates quickReject
101 SkRect bounds = SkRect::MakeLTRB(50, 50, 70, 70);
halcanary96fcdcc2015-08-27 07:41:13 -0700102 canvas.saveLayer(&bounds, nullptr);
reed9b3aa542015-03-11 08:47:12 -0700103 REPORTER_ASSERT(reporter, true == canvas.quickReject(SkRect::MakeWH(10, 10)));
104 REPORTER_ASSERT(reporter, false == canvas.quickReject(SkRect::MakeWH(60, 60)));
105}
106
msarettfbfa2582016-08-12 08:29:08 -0700107static void test_quick_reject(skiatest::Reporter* reporter) {
108 SkCanvas canvas(100, 100);
109 SkRect r0 = SkRect::MakeLTRB(-50.0f, -50.0f, 50.0f, 50.0f);
110 SkRect r1 = SkRect::MakeLTRB(-50.0f, 110.0f, 50.0f, 120.0f);
111 SkRect r2 = SkRect::MakeLTRB(110.0f, -50.0f, 120.0f, 50.0f);
112 SkRect r3 = SkRect::MakeLTRB(-120.0f, -50.0f, 120.0f, 50.0f);
113 SkRect r4 = SkRect::MakeLTRB(-50.0f, -120.0f, 50.0f, 120.0f);
114 SkRect r5 = SkRect::MakeLTRB(-120.0f, -120.0f, 120.0f, 120.0f);
115 SkRect r6 = SkRect::MakeLTRB(-120.0f, -120.0f, -110.0f, -110.0f);
116 SkRect r7 = SkRect::MakeLTRB(SK_ScalarNaN, -50.0f, 50.0f, 50.0f);
117 SkRect r8 = SkRect::MakeLTRB(-50.0f, SK_ScalarNaN, 50.0f, 50.0f);
118 SkRect r9 = SkRect::MakeLTRB(-50.0f, -50.0f, SK_ScalarNaN, 50.0f);
119 SkRect r10 = SkRect::MakeLTRB(-50.0f, -50.0f, 50.0f, SK_ScalarNaN);
120 REPORTER_ASSERT(reporter, false == canvas.quickReject(r0));
121 REPORTER_ASSERT(reporter, true == canvas.quickReject(r1));
122 REPORTER_ASSERT(reporter, true == canvas.quickReject(r2));
123 REPORTER_ASSERT(reporter, false == canvas.quickReject(r3));
124 REPORTER_ASSERT(reporter, false == canvas.quickReject(r4));
125 REPORTER_ASSERT(reporter, false == canvas.quickReject(r5));
126 REPORTER_ASSERT(reporter, true == canvas.quickReject(r6));
127 REPORTER_ASSERT(reporter, true == canvas.quickReject(r7));
128 REPORTER_ASSERT(reporter, true == canvas.quickReject(r8));
129 REPORTER_ASSERT(reporter, true == canvas.quickReject(r9));
130 REPORTER_ASSERT(reporter, true == canvas.quickReject(r10));
131
132 SkMatrix m = SkMatrix::MakeScale(2.0f);
133 m.setTranslateX(10.0f);
134 m.setTranslateY(10.0f);
135 canvas.setMatrix(m);
136 SkRect r11 = SkRect::MakeLTRB(5.0f, 5.0f, 100.0f, 100.0f);
137 SkRect r12 = SkRect::MakeLTRB(5.0f, 50.0f, 100.0f, 100.0f);
138 SkRect r13 = SkRect::MakeLTRB(50.0f, 5.0f, 100.0f, 100.0f);
139 REPORTER_ASSERT(reporter, false == canvas.quickReject(r11));
140 REPORTER_ASSERT(reporter, true == canvas.quickReject(r12));
141 REPORTER_ASSERT(reporter, true == canvas.quickReject(r13));
142}
143
tfarina@chromium.orge4fafb12013-12-12 21:11:12 +0000144DEF_TEST(QuickReject, reporter) {
reed@google.com3d608122011-11-21 15:16:16 +0000145 test_drawBitmap(reporter);
reed9b3aa542015-03-11 08:47:12 -0700146 test_layers(reporter);
msarettfbfa2582016-08-12 08:29:08 -0700147 test_quick_reject(reporter);
reed@google.com3d608122011-11-21 15:16:16 +0000148}
msarett9da5a5a2016-08-19 08:38:36 -0700149
150// Regression test to make sure that we keep fIsScaleTranslate up to date on the canvas.
151// It is possible to set a new matrix on the canvas without calling setMatrix(). This tests
152// that code path.
153DEF_TEST(QuickReject_MatrixState, reporter) {
154 SkCanvas canvas(100, 100);
155
156 SkMatrix matrix;
157 matrix.setRotate(45.0f);
158 canvas.setMatrix(matrix);
159
160 SkPaint paint;
161 sk_sp<SkImageFilter> filter = SkLightingImageFilter::MakeDistantLitDiffuse(
162 SkPoint3::Make(1.0f, 1.0f, 1.0f), 0xFF0000FF, 2.0f, 0.5f, nullptr);
163 REPORTER_ASSERT(reporter, filter);
164 paint.setImageFilter(filter);
165 SkCanvas::SaveLayerRec rec;
166 rec.fPaint = &paint;
167 canvas.saveLayer(rec);
168
169 // quickReject() will assert if the matrix is out of sync.
170 canvas.quickReject(SkRect::MakeWH(100.0f, 100.0f));
171}