blob: 7c441a40feb4c761e3145e9dca7f3a2a39f61e9f [file] [log] [blame]
djsollen@google.com5587ac02013-08-29 20:20:40 +00001/*
2 * Copyright 2013 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
djsollen@google.com5587ac02013-08-29 20:20:40 +00008#include "SkBitmapDevice.h"
9#include "SkCanvas.h"
10#include "SkCanvasStateUtils.h"
11#include "SkDrawFilter.h"
commit-bot@chromium.org07f6cf32013-09-18 20:15:12 +000012#include "SkError.h"
djsollen@google.com5587ac02013-08-29 20:20:40 +000013#include "SkPaint.h"
djsollen@google.com5587ac02013-08-29 20:20:40 +000014#include "SkRRect.h"
tfarina@chromium.org8f6884a2014-01-24 20:56:26 +000015#include "SkRect.h"
16#include "Test.h"
djsollen@google.com5587ac02013-08-29 20:20:40 +000017
18static void test_complex_layers(skiatest::Reporter* reporter) {
djsollen@google.com5587ac02013-08-29 20:20:40 +000019 const int WIDTH = 400;
20 const int HEIGHT = 400;
21 const int SPACER = 10;
22
djsollen@google.com20146b32013-08-29 20:36:22 +000023 SkRect rect = SkRect::MakeXYWH(SkIntToScalar(SPACER), SkIntToScalar(SPACER),
24 SkIntToScalar(WIDTH-(2*SPACER)),
25 SkIntToScalar((HEIGHT-(2*SPACER)) / 7));
djsollen@google.com5587ac02013-08-29 20:20:40 +000026
commit-bot@chromium.orgfa9e5fa2014-02-13 22:00:04 +000027 const SkColorType colorTypes[] = {
28 kRGB_565_SkColorType, kPMColor_SkColorType
djsollen@google.com5587ac02013-08-29 20:20:40 +000029 };
commit-bot@chromium.orgfa9e5fa2014-02-13 22:00:04 +000030 const int configCount = sizeof(colorTypes) / sizeof(SkBitmap::Config);
djsollen@google.com5587ac02013-08-29 20:20:40 +000031
32 const int layerAlpha[] = { 255, 255, 0 };
33 const SkCanvas::SaveFlags flags[] = { SkCanvas::kARGB_NoClipLayer_SaveFlag,
34 SkCanvas::kARGB_ClipLayer_SaveFlag,
35 SkCanvas::kARGB_NoClipLayer_SaveFlag
36 };
37 REPORTER_ASSERT(reporter, sizeof(layerAlpha) == sizeof(flags));
38 const int layerCombinations = sizeof(layerAlpha) / sizeof(int);
39
40 for (int i = 0; i < configCount; ++i) {
41 SkBitmap bitmaps[2];
42 for (int j = 0; j < 2; ++j) {
commit-bot@chromium.orgfa9e5fa2014-02-13 22:00:04 +000043 bitmaps[j].allocPixels(SkImageInfo::Make(WIDTH, HEIGHT,
44 colorTypes[i],
45 kPremul_SkAlphaType));
djsollen@google.com5587ac02013-08-29 20:20:40 +000046
47 SkCanvas canvas(bitmaps[j]);
48
49 canvas.drawColor(SK_ColorRED);
50
51 for (int k = 0; k < layerCombinations; ++k) {
52 // draw a rect within the layer's bounds and again outside the layer's bounds
53 canvas.saveLayerAlpha(&rect, layerAlpha[k], flags[k]);
54
55 SkCanvasState* state = NULL;
56 SkCanvas* tmpCanvas = NULL;
57 if (j) {
58 state = SkCanvasStateUtils::CaptureCanvasState(&canvas);
59 REPORTER_ASSERT(reporter, state);
60 tmpCanvas = SkCanvasStateUtils::CreateFromCanvasState(state);
61 REPORTER_ASSERT(reporter, tmpCanvas);
62 } else {
63 tmpCanvas = SkRef(&canvas);
64 }
65
66 SkPaint bluePaint;
67 bluePaint.setColor(SK_ColorBLUE);
68 bluePaint.setStyle(SkPaint::kFill_Style);
69
70 tmpCanvas->drawRect(rect, bluePaint);
71 tmpCanvas->translate(0, rect.height() + SPACER);
72 tmpCanvas->drawRect(rect, bluePaint);
73
74 tmpCanvas->unref();
75 SkCanvasStateUtils::ReleaseCanvasState(state);
76
77 canvas.restore();
78
79 // translate the canvas for the next iteration
80 canvas.translate(0, 2*(rect.height() + SPACER));
81 }
82 }
83
84 // now we memcmp the two bitmaps
85 REPORTER_ASSERT(reporter, bitmaps[0].getSize() == bitmaps[1].getSize());
86 REPORTER_ASSERT(reporter, !memcmp(bitmaps[0].getPixels(),
87 bitmaps[1].getPixels(),
88 bitmaps[0].getSize()));
89 }
90}
91
92////////////////////////////////////////////////////////////////////////////////
93
djsollen@google.com339e79f2013-09-04 17:16:00 +000094static void test_complex_clips(skiatest::Reporter* reporter) {
95
96 const int WIDTH = 400;
97 const int HEIGHT = 400;
djsollen@google.com1037c7b2013-09-04 18:20:30 +000098 const int SPACER = 10;
djsollen@google.com339e79f2013-09-04 17:16:00 +000099
djsollen@google.com1037c7b2013-09-04 18:20:30 +0000100 SkIRect layerRect = SkIRect::MakeWH(WIDTH, HEIGHT / 4);
djsollen@google.com339e79f2013-09-04 17:16:00 +0000101 layerRect.inset(2*SPACER, 2*SPACER);
102
djsollen@google.com1037c7b2013-09-04 18:20:30 +0000103 SkIRect clipRect = layerRect;
djsollen@google.com339e79f2013-09-04 17:16:00 +0000104 clipRect.fRight = clipRect.fLeft + (clipRect.width() / 2) - (2*SPACER);
105 clipRect.outset(SPACER, SPACER);
106
djsollen@google.com1037c7b2013-09-04 18:20:30 +0000107 SkIRect regionBounds = clipRect;
djsollen@google.com339e79f2013-09-04 17:16:00 +0000108 regionBounds.offset(clipRect.width() + (2*SPACER), 0);
109
110 SkIRect regionInterior = regionBounds;
111 regionInterior.inset(SPACER*3, SPACER*3);
112
113 SkRegion clipRegion;
114 clipRegion.setRect(regionBounds);
115 clipRegion.op(regionInterior, SkRegion::kDifference_Op);
116
117
118 const SkRegion::Op clipOps[] = { SkRegion::kIntersect_Op,
119 SkRegion::kIntersect_Op,
120 SkRegion::kReplace_Op,
121 };
122 const SkCanvas::SaveFlags flags[] = { SkCanvas::kARGB_NoClipLayer_SaveFlag,
123 SkCanvas::kARGB_ClipLayer_SaveFlag,
124 SkCanvas::kARGB_NoClipLayer_SaveFlag,
125 };
126 REPORTER_ASSERT(reporter, sizeof(clipOps) == sizeof(flags));
127 const int layerCombinations = sizeof(flags) / sizeof(SkCanvas::SaveFlags);
128
129 SkBitmap bitmaps[2];
130 for (int i = 0; i < 2; ++i) {
commit-bot@chromium.orgfa9e5fa2014-02-13 22:00:04 +0000131 bitmaps[i].allocN32Pixels(WIDTH, HEIGHT);
djsollen@google.com339e79f2013-09-04 17:16:00 +0000132
133 SkCanvas canvas(bitmaps[i]);
134
135 canvas.drawColor(SK_ColorRED);
136
137 SkRegion localRegion = clipRegion;
138
139 for (int j = 0; j < layerCombinations; ++j) {
djsollen@google.com1037c7b2013-09-04 18:20:30 +0000140 SkRect layerBounds = SkRect::Make(layerRect);
141 canvas.saveLayerAlpha(&layerBounds, 128, flags[j]);
djsollen@google.com339e79f2013-09-04 17:16:00 +0000142
143 SkCanvasState* state = NULL;
144 SkCanvas* tmpCanvas = NULL;
145 if (i) {
146 state = SkCanvasStateUtils::CaptureCanvasState(&canvas);
147 REPORTER_ASSERT(reporter, state);
148 tmpCanvas = SkCanvasStateUtils::CreateFromCanvasState(state);
149 REPORTER_ASSERT(reporter, tmpCanvas);
150 } else {
151 tmpCanvas = SkRef(&canvas);
152 }
153
154 tmpCanvas->save();
djsollen@google.com1037c7b2013-09-04 18:20:30 +0000155 tmpCanvas->clipRect(SkRect::Make(clipRect), clipOps[j]);
djsollen@google.com339e79f2013-09-04 17:16:00 +0000156 tmpCanvas->drawColor(SK_ColorBLUE);
157 tmpCanvas->restore();
158
159 tmpCanvas->clipRegion(localRegion, clipOps[j]);
160 tmpCanvas->drawColor(SK_ColorBLUE);
161
162 tmpCanvas->unref();
163 SkCanvasStateUtils::ReleaseCanvasState(state);
164
165 canvas.restore();
166
167 // translate the canvas and region for the next iteration
djsollen@google.com1037c7b2013-09-04 18:20:30 +0000168 canvas.translate(0, SkIntToScalar(2*(layerRect.height() + (SPACER))));
djsollen@google.com339e79f2013-09-04 17:16:00 +0000169 localRegion.translate(0, 2*(layerRect.height() + SPACER));
170 }
171 }
172
173 // now we memcmp the two bitmaps
174 REPORTER_ASSERT(reporter, bitmaps[0].getSize() == bitmaps[1].getSize());
175 REPORTER_ASSERT(reporter, !memcmp(bitmaps[0].getPixels(),
176 bitmaps[1].getPixels(),
177 bitmaps[0].getSize()));
178}
179
180////////////////////////////////////////////////////////////////////////////////
181
djsollen@google.com5587ac02013-08-29 20:20:40 +0000182class TestDrawFilter : public SkDrawFilter {
183public:
184 virtual bool filter(SkPaint*, Type) SK_OVERRIDE { return true; }
185};
186
187static void test_draw_filters(skiatest::Reporter* reporter) {
188 TestDrawFilter drawFilter;
189 SkBitmapDevice device(SkBitmap::kARGB_8888_Config, 10, 10);
190 SkCanvas canvas(&device);
191
192 canvas.setDrawFilter(&drawFilter);
193
194 SkCanvasState* state = SkCanvasStateUtils::CaptureCanvasState(&canvas);
195 REPORTER_ASSERT(reporter, state);
196 SkCanvas* tmpCanvas = SkCanvasStateUtils::CreateFromCanvasState(state);
197 REPORTER_ASSERT(reporter, tmpCanvas);
198
199 REPORTER_ASSERT(reporter, NULL != canvas.getDrawFilter());
200 REPORTER_ASSERT(reporter, NULL == tmpCanvas->getDrawFilter());
201
202 tmpCanvas->unref();
203 SkCanvasStateUtils::ReleaseCanvasState(state);
204}
205
206////////////////////////////////////////////////////////////////////////////////
207
commit-bot@chromium.org07f6cf32013-09-18 20:15:12 +0000208// we need this function to prevent SkError from printing to stdout
209static void error_callback(SkError code, void* ctx) {}
210
djsollen@google.com5587ac02013-08-29 20:20:40 +0000211static void test_soft_clips(skiatest::Reporter* reporter) {
212 SkBitmapDevice device(SkBitmap::kARGB_8888_Config, 10, 10);
213 SkCanvas canvas(&device);
214
215 SkRRect roundRect;
216 roundRect.setOval(SkRect::MakeWH(5, 5));
217
218 canvas.clipRRect(roundRect, SkRegion::kIntersect_Op, true);
219
commit-bot@chromium.org07f6cf32013-09-18 20:15:12 +0000220 SkSetErrorCallback(error_callback, NULL);
221
djsollen@google.com5587ac02013-08-29 20:20:40 +0000222 SkCanvasState* state = SkCanvasStateUtils::CaptureCanvasState(&canvas);
223 REPORTER_ASSERT(reporter, !state);
commit-bot@chromium.org07f6cf32013-09-18 20:15:12 +0000224
225 REPORTER_ASSERT(reporter, kInvalidOperation_SkError == SkGetLastError());
226 SkClearLastError();
djsollen@google.com5587ac02013-08-29 20:20:40 +0000227}
228
tfarina@chromium.orge4fafb12013-12-12 21:11:12 +0000229DEF_TEST(CanvasState, reporter) {
djsollen@google.com5587ac02013-08-29 20:20:40 +0000230 test_complex_layers(reporter);
djsollen@google.com339e79f2013-09-04 17:16:00 +0000231 test_complex_clips(reporter);
djsollen@google.com5587ac02013-08-29 20:20:40 +0000232 test_draw_filters(reporter);
233 test_soft_clips(reporter);
234}