blob: 5fdaeedb55be04bb993d908233f83586ca2bfc0a [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +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 */
reed@android.comc07d23a2009-02-06 13:30:58 +00007#include "SampleCode.h"
8#include "SkView.h"
9#include "SkCanvas.h"
10#include "SkGradientShader.h"
11#include "SkGraphics.h"
reed@android.comc07d23a2009-02-06 13:30:58 +000012#include "SkPath.h"
reed@android.comc07d23a2009-02-06 13:30:58 +000013#include "SkRegion.h"
14#include "SkShader.h"
15#include "SkUtils.h"
reed@android.comc07d23a2009-02-06 13:30:58 +000016#include "SkColorPriv.h"
17#include "SkColorFilter.h"
18#include "SkTime.h"
19#include "SkTypeface.h"
20
reed@google.com81e3d7f2011-06-01 12:42:36 +000021class PathClipView : public SampleView {
reed@android.comc07d23a2009-02-06 13:30:58 +000022public:
23 SkRect fOval;
24 SkPoint fCenter;
25
reed67c65132015-09-28 06:16:07 -070026 PathClipView() : fOval(SkRect::MakeWH(200, 50)), fCenter(SkPoint::Make(250, 250)) {}
rmistry@google.comae933ce2012-08-23 18:19:56 +000027
reed@android.comc07d23a2009-02-06 13:30:58 +000028protected:
mtkleinf0599002015-07-13 06:18:39 -070029 bool onQuery(SkEvent* evt) override {
reed@android.comc07d23a2009-02-06 13:30:58 +000030 if (SampleCode::TitleQ(*evt)) {
31 SampleCode::TitleR(evt, "PathClip");
32 return true;
33 }
34 return this->INHERITED::onQuery(evt);
35 }
rmistry@google.comae933ce2012-08-23 18:19:56 +000036
mtkleinf0599002015-07-13 06:18:39 -070037 void onDrawContent(SkCanvas* canvas) override {
reed67c65132015-09-28 06:16:07 -070038 const SkRect oval = fOval.makeOffset(fCenter.fX - fOval.centerX(),
39 fCenter.fY - fOval.centerY());
rmistry@google.comae933ce2012-08-23 18:19:56 +000040
reed@android.comc07d23a2009-02-06 13:30:58 +000041 SkPaint p;
42 p.setAntiAlias(true);
rmistry@google.comae933ce2012-08-23 18:19:56 +000043
reed@android.comc07d23a2009-02-06 13:30:58 +000044 p.setStyle(SkPaint::kStroke_Style);
45 canvas->drawOval(oval, p);
46
reed67c65132015-09-28 06:16:07 -070047 const SkRect r = SkRect::MakeLTRB(200, 200, 300, 300);
reed@android.comc07d23a2009-02-06 13:30:58 +000048 canvas->clipRect(r);
rmistry@google.comae933ce2012-08-23 18:19:56 +000049
reed@android.comc07d23a2009-02-06 13:30:58 +000050 p.setStyle(SkPaint::kFill_Style);
51 p.setColor(SK_ColorRED);
52 canvas->drawRect(r, p);
rmistry@google.comae933ce2012-08-23 18:19:56 +000053
reed@android.comc07d23a2009-02-06 13:30:58 +000054 p.setColor(0x800000FF);
reed@android.comc07d23a2009-02-06 13:30:58 +000055 canvas->drawOval(oval, p);
56 }
reed@android.come72fee52009-11-16 14:52:01 +000057
mtklein36352bf2015-03-25 18:17:31 -070058 SkView::Click* onFindClickHandler(SkScalar x, SkScalar y, unsigned) override {
reed@android.come72fee52009-11-16 14:52:01 +000059 return new Click(this);
60 }
rmistry@google.comae933ce2012-08-23 18:19:56 +000061
mtkleinf0599002015-07-13 06:18:39 -070062 bool onClick(Click* click) override {
reed@android.come72fee52009-11-16 14:52:01 +000063 fCenter.set(click->fCurr.fX, click->fCurr.fY);
halcanary96fcdcc2015-08-27 07:41:13 -070064 this->inval(nullptr);
caryclark@google.com02939ce2012-06-06 12:09:51 +000065 return false;
reed@android.comc07d23a2009-02-06 13:30:58 +000066 }
rmistry@google.comae933ce2012-08-23 18:19:56 +000067
reed@android.comc07d23a2009-02-06 13:30:58 +000068private:
reed@google.com81e3d7f2011-06-01 12:42:36 +000069 typedef SampleView INHERITED;
reed@android.comc07d23a2009-02-06 13:30:58 +000070};
reed67c65132015-09-28 06:16:07 -070071DEF_SAMPLE( return new PathClipView; )
reed@android.comc07d23a2009-02-06 13:30:58 +000072
73//////////////////////////////////////////////////////////////////////////////
74
reed67c65132015-09-28 06:16:07 -070075static int clip_line(const SkRect& bounds, SkPoint p0, SkPoint p1, SkPoint edges[]) {
76 SkPoint* edgesStart = edges;
77
78 if (p0.fY == p1.fY) {
79 return 0;
80 }
halcanary9d524f22016-03-29 09:03:52 -070081
reed67c65132015-09-28 06:16:07 -070082 if (p0.fY > p1.fY) {
83 SkTSwap(p0, p1);
84 }
85 // now we're monotonic in Y: p0 <= p1
86 if (p1.fY <= bounds.top() || p0.fY >= bounds.bottom()) {
87 return 0;
88 }
halcanary9d524f22016-03-29 09:03:52 -070089
reed67c65132015-09-28 06:16:07 -070090 double dxdy = (double)(p1.fX - p0.fX) / (p1.fY - p0.fY);
91 if (p0.fY < bounds.top()) {
92 p0.fX = SkDoubleToScalar(p0.fX + dxdy * (bounds.top() - p0.fY));
93 p0.fY = bounds.top();
94 }
95 if (p1.fY > bounds.bottom()) {
96 p1.fX = SkDoubleToScalar(p1.fX + dxdy * (bounds.bottom() - p1.fY));
97 p1.fY = bounds.bottom();
98 }
halcanary9d524f22016-03-29 09:03:52 -070099
reed67c65132015-09-28 06:16:07 -0700100 // Now p0...p1 is strictly inside bounds vertically, so we just need to clip horizontally
halcanary9d524f22016-03-29 09:03:52 -0700101
reed67c65132015-09-28 06:16:07 -0700102 if (p0.fX > p1.fX) {
103 SkTSwap(p0, p1);
104 }
105 // now we're left-to-right: p0 .. p1
halcanary9d524f22016-03-29 09:03:52 -0700106
reed67c65132015-09-28 06:16:07 -0700107 if (p1.fX <= bounds.left()) { // entirely to the left
108 p0.fX = p1.fX = bounds.left();
109 *edges++ = p0;
110 *edges++ = p1;
111 return 2;
112 }
113 if (p0.fX >= bounds.right()) { // entirely to the right
114 p0.fX = p1.fX = bounds.right();
115 *edges++ = p0;
116 *edges++ = p1;
117 return 2;
118 }
halcanary9d524f22016-03-29 09:03:52 -0700119
reed67c65132015-09-28 06:16:07 -0700120 if (p0.fX < bounds.left()) {
121 float y = SkDoubleToScalar(p0.fY + (bounds.left() - p0.fX) / dxdy);
122 *edges++ = SkPoint::Make(bounds.left(), p0.fY);
123 *edges++ = SkPoint::Make(bounds.left(), y);
124 p0.set(bounds.left(), y);
125 }
126 if (p1.fX > bounds.right()) {
127 float y = SkDoubleToScalar(p0.fY + (bounds.right() - p0.fX) / dxdy);
128 *edges++ = p0;
129 *edges++ = SkPoint::Make(bounds.right(), y);
130 *edges++ = SkPoint::Make(bounds.right(), p1.fY);
131 } else {
132 *edges++ = p0;
133 *edges++ = p1;
134 }
135 return SkToInt(edges - edgesStart);
136}
137
138static void draw_clipped_line(SkCanvas* canvas, const SkRect& bounds,
139 SkPoint p0, SkPoint p1, const SkPaint& paint) {
140 SkPoint verts[6];
141 int count = clip_line(bounds, p0, p1, verts);
142
143 SkPath path;
144 path.addPoly(verts, count, false);
145 canvas->drawPath(path, paint);
146}
147
148// Demonstrate edge-clipping that is used in the scan converter
149//
150class EdgeClipView : public SampleView {
151 enum {
152 N = 3
153 };
154public:
155 SkPoint fPoly[N];
156 SkRect fClip;
157 SkColor fEdgeColor[N];
halcanary9d524f22016-03-29 09:03:52 -0700158
reed67c65132015-09-28 06:16:07 -0700159 EdgeClipView() : fClip(SkRect::MakeLTRB(150, 150, 550, 450)) {
160 fPoly[0].set(300, 40);
161 fPoly[1].set(550, 250);
162 fPoly[2].set(40, 450);
163
164 fEdgeColor[0] = 0xFFFF0000;
165 fEdgeColor[1] = 0xFF00FF00;
166 fEdgeColor[2] = 0xFF0000FF;
167 }
halcanary9d524f22016-03-29 09:03:52 -0700168
reed67c65132015-09-28 06:16:07 -0700169protected:
170 bool onQuery(SkEvent* evt) override {
171 if (SampleCode::TitleQ(*evt)) {
172 SampleCode::TitleR(evt, "EdgeClip");
173 return true;
174 }
175 return this->INHERITED::onQuery(evt);
176 }
177
178 static SkScalar snap(SkScalar x) {
179 return SkScalarRoundToScalar(x * 0.5f) * 2;
180 }
181 static SkPoint snap(const SkPoint& pt) {
182 return SkPoint::Make(snap(pt.x()), snap(pt.y()));
183 }
184 static void snap(SkPoint dst[], const SkPoint src[], int count) {
185 for (int i = 0; i < count; ++i) {
186 dst[i] = snap(src[i]);
187 }
188 }
189
190 void onDrawContent(SkCanvas* canvas) override {
191 SkPath path;
192 path.addPoly(fPoly, N, true);
193
194 // Draw the full triangle, stroked and filled
195 SkPaint p;
196 p.setAntiAlias(true);
197 p.setColor(0xFFE0E0E0);
198 canvas->drawPath(path, p);
199 p.setStyle(SkPaint::kStroke_Style);
200 p.setStrokeWidth(2);
201 for (int i = 0; i < N; ++i) {
202 const int j = (i + 1) % N;
203 p.setColor(fEdgeColor[i]);
204 p.setAlpha(0x88);
Hal Canary23e474c2017-05-15 13:35:35 -0400205 canvas->drawLine(fPoly[i], fPoly[j], p);
reed67c65132015-09-28 06:16:07 -0700206 }
207 p.setStyle(SkPaint::kFill_Style);
208
209 // Draw the clip itself
210 p.setColor(0xFF8888CC);
211 canvas->drawRect(fClip, p);
212
213 // Draw the filled triangle through the clip
214 p.setColor(0xFF88CC88);
215 canvas->save();
216 canvas->clipRect(fClip);
217 canvas->drawPath(path, p);
218 canvas->restore();
219
220 p.setStyle(SkPaint::kStroke_Style);
221 p.setStrokeWidth(6);
222
223 // Draw each of the "Edges" that survived the clipping
224 // We use a layer, so we can PLUS the different edge-colors, showing where two edges
225 // canceled each other out.
226 canvas->saveLayer(nullptr, nullptr);
reed374772b2016-10-05 17:33:02 -0700227 p.setBlendMode(SkBlendMode::kPlus);
reed67c65132015-09-28 06:16:07 -0700228 for (int i = 0; i < N; ++i) {
229 const int j = (i + 1) % N;
230 p.setColor(fEdgeColor[i]);
231 draw_clipped_line(canvas, fClip, fPoly[i], fPoly[j], p);
232 }
233 canvas->restore();
234 }
235
236 class MyClick : public Click {
237 public:
238 MyClick(SkView* view) : Click(view) {}
239 virtual void handleMove() = 0;
240 };
halcanary9d524f22016-03-29 09:03:52 -0700241
reed67c65132015-09-28 06:16:07 -0700242 class VertClick : public MyClick {
243 SkPoint* fPt;
244 public:
245 VertClick(SkView* view, SkPoint* pt) : MyClick(view), fPt(pt) {}
246 void handleMove() override { *fPt = snap(fCurr); }
247 };
halcanary9d524f22016-03-29 09:03:52 -0700248
reed67c65132015-09-28 06:16:07 -0700249 class DragRectClick : public MyClick {
250 SkRect* fRect;
251 public:
252 DragRectClick(SkView* view, SkRect* rect) : MyClick(view), fRect(rect) {}
253 void handleMove() override { fRect->offset(fCurr.x() - fPrev.x(), fCurr.y() - fPrev.y()); }
254 };
halcanary9d524f22016-03-29 09:03:52 -0700255
reed67c65132015-09-28 06:16:07 -0700256 class DragPolyClick : public MyClick {
257 SkPoint fSrc[100];
258 SkPoint* fPoly;
259 int fCount;
260 public:
261 DragPolyClick(SkView* view, SkPoint poly[], int count)
262 : MyClick(view), fPoly(poly), fCount(count)
263 {
264 SkASSERT((size_t)count <= SK_ARRAY_COUNT(fSrc));
265 memcpy(fSrc, poly, count * sizeof(SkPoint));
266 }
267 void handleMove() override {
268 const SkScalar dx = fCurr.x() - fOrig.x();
269 const SkScalar dy = fCurr.y() - fOrig.y();
270 for (int i = 0; i < fCount; ++i) {
271 fPoly[i].set(snap(fSrc[i].x() + dx), snap(fSrc[i].y() + dy));
272 }
273 }
274 };
275
276 class DoNothingClick : public MyClick {
277 public:
278 DoNothingClick(SkView* view) : MyClick(view) {}
279 void handleMove() override {}
280 };
281
282 static bool hit_test(const SkPoint& pt, SkScalar x, SkScalar y) {
283 const SkScalar rad = 8;
284 const SkScalar dx = pt.x() - x;
285 const SkScalar dy = pt.y() - y;
286 return dx*dx + dy*dy <= rad*rad;
287 }
288
289 SkView::Click* onFindClickHandler(SkScalar x, SkScalar y, unsigned) override {
290 for (int i = 0; i < N; ++i) {
291 if (hit_test(fPoly[i], x, y)) {
292 return new VertClick(this, &fPoly[i]);
293 }
294 }
halcanary9d524f22016-03-29 09:03:52 -0700295
reed67c65132015-09-28 06:16:07 -0700296 SkPath path;
297 path.addPoly(fPoly, N, true);
298 if (path.contains(x, y)) {
299 return new DragPolyClick(this, fPoly, N);
300 }
301
302 if (fClip.intersects(SkRect::MakeLTRB(x - 1, y - 1, x + 1, y + 1))) {
303 return new DragRectClick(this, &fClip);
304 }
305 return new DoNothingClick(this);
306 }
halcanary9d524f22016-03-29 09:03:52 -0700307
reed67c65132015-09-28 06:16:07 -0700308 bool onClick(Click* click) override {
309 ((MyClick*)click)->handleMove();
310 this->inval(nullptr);
311 return false;
312 }
halcanary9d524f22016-03-29 09:03:52 -0700313
reed67c65132015-09-28 06:16:07 -0700314private:
315 typedef SampleView INHERITED;
316};
317DEF_SAMPLE( return new EdgeClipView; )