blob: 74daa1ba3a81b6401bda38364f8f3441c20eb504 [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 */
7
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "include/core/SkBitmap.h"
9#include "include/core/SkCanvas.h"
10#include "include/core/SkColorFilter.h"
11#include "include/core/SkColorPriv.h"
12#include "include/core/SkFont.h"
13#include "include/core/SkGraphics.h"
14#include "include/core/SkPath.h"
15#include "include/core/SkRegion.h"
16#include "include/core/SkShader.h"
17#include "include/core/SkTime.h"
18#include "include/core/SkTypeface.h"
19#include "include/effects/SkGradientShader.h"
20#include "include/utils/SkParsePath.h"
21#include "samplecode/Sample.h"
22#include "src/utils/SkUTF.h"
Hal Canary41248072019-07-11 16:32:53 -040023#include "tools/timer/TimeUtils.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000024
Mike Kleinc0bd9f92019-04-23 12:05:21 -050025#include "src/core/SkGeometry.h"
reed@android.coma9640282009-08-28 20:06:54 +000026
bungeman60e0fee2015-08-26 05:15:46 -070027#include <stdlib.h>
28
reed@android.coma9640282009-08-28 20:06:54 +000029// http://code.google.com/p/skia/issues/detail?id=32
30static void test_cubic() {
31 SkPoint src[4] = {
reed@google.com261b8e22011-04-14 17:53:24 +000032 { 556.25000f, 523.03003f },
33 { 556.23999f, 522.96002f },
34 { 556.21997f, 522.89001f },
35 { 556.21997f, 522.82001f }
reed@android.coma9640282009-08-28 20:06:54 +000036 };
37 SkPoint dst[11];
38 dst[10].set(42, -42); // one past the end, that we don't clobber these
39 SkScalar tval[] = { 0.33333334f, 0.99999994f };
40
41 SkChopCubicAt(src, dst, tval, 2);
42
43#if 0
44 for (int i = 0; i < 11; i++) {
45 SkDebugf("--- %d [%g %g]\n", i, dst[i].fX, dst[i].fY);
46 }
47#endif
48}
49
reed@android.comd0d0e652009-10-13 13:31:27 +000050static void test_cubic2() {
51 const char* str = "M2242 -590088L-377758 9.94099e+07L-377758 9.94099e+07L2242 -590088Z";
52 SkPath path;
53 SkParsePath::FromSVGString(str, &path);
rmistry@google.comae933ce2012-08-23 18:19:56 +000054
reed@android.comd0d0e652009-10-13 13:31:27 +000055 {
reed@android.comd0d0e652009-10-13 13:31:27 +000056 SkRect r = path.getBounds();
57 SkIRect ir;
58 r.round(&ir);
bungeman@google.comfab44db2013-10-11 18:50:45 +000059 SkDebugf("[%g %g %g %g] [%x %x %x %x]\n",
60 SkScalarToDouble(r.fLeft), SkScalarToDouble(r.fTop),
61 SkScalarToDouble(r.fRight), SkScalarToDouble(r.fBottom),
62 ir.fLeft, ir.fTop, ir.fRight, ir.fBottom);
reed@android.comd0d0e652009-10-13 13:31:27 +000063 }
rmistry@google.comae933ce2012-08-23 18:19:56 +000064
reed@android.comd0d0e652009-10-13 13:31:27 +000065 SkBitmap bitmap;
commit-bot@chromium.orga8c18312014-02-17 02:55:57 +000066 bitmap.allocN32Pixels(300, 200);
reed@android.comd0d0e652009-10-13 13:31:27 +000067
68 SkCanvas canvas(bitmap);
69 SkPaint paint;
70 paint.setAntiAlias(true);
71 canvas.drawPath(path, paint);
72}
73
Ben Wagnerb2c4ea62018-08-08 11:36:17 -040074class PathView : public Sample {
reed339cdbf2015-02-05 22:02:37 -080075 SkScalar fPrevSecs;
reed@android.com8a1c16f2008-12-17 15:59:43 +000076public:
reed339cdbf2015-02-05 22:02:37 -080077 SkScalar fDStroke, fStroke, fMinStroke, fMaxStroke;
reed@android.com8a1c16f2008-12-17 15:59:43 +000078 SkPath fPath[6];
79 bool fShowHairline;
reed@google.comc9fa63c2012-03-12 21:14:09 +000080 bool fOnce;
rmistry@google.comae933ce2012-08-23 18:19:56 +000081
82 PathView() {
reed339cdbf2015-02-05 22:02:37 -080083 fPrevSecs = 0;
reed@google.comc9fa63c2012-03-12 21:14:09 +000084 fOnce = false;
85 }
rmistry@google.comae933ce2012-08-23 18:19:56 +000086
reed@google.comc9fa63c2012-03-12 21:14:09 +000087 void init() {
88 if (fOnce) {
89 return;
90 }
91 fOnce = true;
92
reed@android.coma9640282009-08-28 20:06:54 +000093 test_cubic();
reed@android.comd0d0e652009-10-13 13:31:27 +000094 test_cubic2();
reed@android.coma9640282009-08-28 20:06:54 +000095
reed@android.com8a1c16f2008-12-17 15:59:43 +000096 fShowHairline = false;
rmistry@google.comae933ce2012-08-23 18:19:56 +000097
reed@android.com8a1c16f2008-12-17 15:59:43 +000098 fDStroke = 1;
99 fStroke = 10;
100 fMinStroke = 10;
101 fMaxStroke = 180;
102
reed339cdbf2015-02-05 22:02:37 -0800103 const SkScalar V = 85;
rmistry@google.comae933ce2012-08-23 18:19:56 +0000104
reed339cdbf2015-02-05 22:02:37 -0800105 fPath[0].moveTo(40, 70);
106 fPath[0].lineTo(70, 70 + SK_ScalarHalf);
107 fPath[0].lineTo(110, 70);
rmistry@google.comae933ce2012-08-23 18:19:56 +0000108
reed339cdbf2015-02-05 22:02:37 -0800109 fPath[1].moveTo(40, 70);
110 fPath[1].lineTo(70, 70 - SK_ScalarHalf);
111 fPath[1].lineTo(110, 70);
rmistry@google.comae933ce2012-08-23 18:19:56 +0000112
reed339cdbf2015-02-05 22:02:37 -0800113 fPath[2].moveTo(V, V);
114 fPath[2].lineTo(50, V);
115 fPath[2].lineTo(50, 50);
rmistry@google.comae933ce2012-08-23 18:19:56 +0000116
reed339cdbf2015-02-05 22:02:37 -0800117 fPath[3].moveTo(50, 50);
118 fPath[3].lineTo(50, V);
119 fPath[3].lineTo(V, V);
rmistry@google.comae933ce2012-08-23 18:19:56 +0000120
reed339cdbf2015-02-05 22:02:37 -0800121 fPath[4].moveTo(50, 50);
122 fPath[4].lineTo(50, V);
123 fPath[4].lineTo(52, 50);
rmistry@google.comae933ce2012-08-23 18:19:56 +0000124
reed339cdbf2015-02-05 22:02:37 -0800125 fPath[5].moveTo(52, 50);
126 fPath[5].lineTo(50, V);
127 fPath[5].lineTo(50, 50);
rmistry@google.comae933ce2012-08-23 18:19:56 +0000128
reed@google.com0faac1e2011-05-11 05:58:58 +0000129 this->setBGColor(0xFFDDDDDD);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000130 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000131
reed@android.com8a1c16f2008-12-17 15:59:43 +0000132protected:
Hal Canary8a027312019-07-03 10:55:44 -0400133 SkString name() override { return SkString("Paths"); }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000134
reed@google.com0faac1e2011-05-11 05:58:58 +0000135 void drawPath(SkCanvas* canvas, const SkPath& path, SkPaint::Join j) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000136 SkPaint paint;
rmistry@google.comae933ce2012-08-23 18:19:56 +0000137
reed@android.com8a1c16f2008-12-17 15:59:43 +0000138 paint.setAntiAlias(true);
139 paint.setStyle(SkPaint::kStroke_Style);
140 paint.setStrokeJoin(j);
reed339cdbf2015-02-05 22:02:37 -0800141 paint.setStrokeWidth(fStroke);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000142
reed@google.com0faac1e2011-05-11 05:58:58 +0000143 if (fShowHairline) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000144 SkPath fill;
rmistry@google.comae933ce2012-08-23 18:19:56 +0000145
146 paint.getFillPath(path, &fill);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000147 paint.setStrokeWidth(0);
148 canvas->drawPath(fill, paint);
reed@google.com0faac1e2011-05-11 05:58:58 +0000149 } else {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000150 canvas->drawPath(path, paint);
reed@google.com0faac1e2011-05-11 05:58:58 +0000151 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000152
reed@android.com8a1c16f2008-12-17 15:59:43 +0000153 paint.setColor(SK_ColorRED);
154 paint.setStrokeWidth(0);
155 canvas->drawPath(path, paint);
156 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000157
mtkleinf0599002015-07-13 06:18:39 -0700158 void onDrawContent(SkCanvas* canvas) override {
reed@google.comc9fa63c2012-03-12 21:14:09 +0000159 this->init();
reed339cdbf2015-02-05 22:02:37 -0800160 canvas->translate(50, 50);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000161
162 static const SkPaint::Join gJoins[] = {
163 SkPaint::kBevel_Join,
164 SkPaint::kMiter_Join,
165 SkPaint::kRound_Join
166 };
167
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +0000168 for (size_t i = 0; i < SK_ARRAY_COUNT(gJoins); i++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000169 canvas->save();
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +0000170 for (size_t j = 0; j < SK_ARRAY_COUNT(fPath); j++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000171 this->drawPath(canvas, fPath[j], gJoins[i]);
reed339cdbf2015-02-05 22:02:37 -0800172 canvas->translate(200, 0);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000173 }
174 canvas->restore();
rmistry@google.comae933ce2012-08-23 18:19:56 +0000175
reed339cdbf2015-02-05 22:02:37 -0800176 canvas->translate(0, 200);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000177 }
reed339cdbf2015-02-05 22:02:37 -0800178 }
mtkleinf0599002015-07-13 06:18:39 -0700179
Hal Canary41248072019-07-11 16:32:53 -0400180 bool onAnimate(double nanos) override {
181 SkScalar currSecs = TimeUtils::Scaled(1e-9 * nanos, 100);
reed339cdbf2015-02-05 22:02:37 -0800182 SkScalar delta = currSecs - fPrevSecs;
183 fPrevSecs = currSecs;
rmistry@google.comae933ce2012-08-23 18:19:56 +0000184
reed339cdbf2015-02-05 22:02:37 -0800185 fStroke += fDStroke * delta;
186 if (fStroke > fMaxStroke || fStroke < fMinStroke) {
187 fDStroke = -fDStroke;
188 }
189 return true;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000190 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000191
Hal Canaryb1f411a2019-08-29 10:39:22 -0400192 Sample::Click* onFindClickHandler(SkScalar x, SkScalar y, skui::ModifierKey modi) override {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000193 fShowHairline = !fShowHairline;
Hal Canaryfcf63592019-07-12 11:32:43 -0400194 return nullptr;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000195 }
rmistry@google.comae933ce2012-08-23 18:19:56 +0000196
reed@android.com8a1c16f2008-12-17 15:59:43 +0000197private:
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400198 typedef Sample INHERITED;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000199};
reeda7a8b102014-12-16 08:07:43 -0800200DEF_SAMPLE( return new PathView; )
reed@android.com8a1c16f2008-12-17 15:59:43 +0000201
202//////////////////////////////////////////////////////////////////////////////
203
Mike Kleinc0bd9f92019-04-23 12:05:21 -0500204#include "include/effects/SkCornerPathEffect.h"
205#include "include/utils/SkRandom.h"
reeda7a8b102014-12-16 08:07:43 -0800206
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400207class ArcToView : public Sample {
Mike Reed40e7e652017-07-22 22:12:59 -0400208 bool fDoFrame, fDoCorner, fDoConic;
209 SkPaint fPtsPaint, fSkeletonPaint, fCornerPaint;
reeda7a8b102014-12-16 08:07:43 -0800210public:
211 enum {
212 N = 4
213 };
214 SkPoint fPts[N];
reeda7a8b102014-12-16 08:07:43 -0800215
reed8b575242014-12-17 01:47:32 -0800216 ArcToView()
Mike Reed40e7e652017-07-22 22:12:59 -0400217 : fDoFrame(false), fDoCorner(false), fDoConic(false)
reed8b575242014-12-17 01:47:32 -0800218 {
reeda7a8b102014-12-16 08:07:43 -0800219 SkRandom rand;
220 for (int i = 0; i < N; ++i) {
221 fPts[i].fX = 20 + rand.nextUScalar1() * 640;
222 fPts[i].fY = 20 + rand.nextUScalar1() * 480;
223 }
mtkleinf0599002015-07-13 06:18:39 -0700224
reed8b575242014-12-17 01:47:32 -0800225 const SkScalar rad = 50;
reeda7a8b102014-12-16 08:07:43 -0800226
227 fPtsPaint.setAntiAlias(true);
228 fPtsPaint.setStrokeWidth(15);
229 fPtsPaint.setStrokeCap(SkPaint::kRound_Cap);
230
reeda7a8b102014-12-16 08:07:43 -0800231 fCornerPaint.setAntiAlias(true);
232 fCornerPaint.setStyle(SkPaint::kStroke_Style);
233 fCornerPaint.setStrokeWidth(13);
234 fCornerPaint.setColor(SK_ColorGREEN);
reeda4393342016-03-18 11:22:57 -0700235 fCornerPaint.setPathEffect(SkCornerPathEffect::Make(rad*2));
reeda7a8b102014-12-16 08:07:43 -0800236
237 fSkeletonPaint.setAntiAlias(true);
238 fSkeletonPaint.setStyle(SkPaint::kStroke_Style);
239 fSkeletonPaint.setColor(SK_ColorRED);
240 }
241
reed8b575242014-12-17 01:47:32 -0800242 void toggle(bool& value) {
243 value = !value;
reed8b575242014-12-17 01:47:32 -0800244 }
245
reeda7a8b102014-12-16 08:07:43 -0800246protected:
Hal Canary8a027312019-07-03 10:55:44 -0400247 SkString name() override { return SkString("ArcTo"); }
248
Hal Canary6cc65e12019-07-03 15:53:04 -0400249 bool onChar(SkUnichar uni) override {
reed8b575242014-12-17 01:47:32 -0800250 switch (uni) {
251 case '1': this->toggle(fDoFrame); return true;
Mike Reed40e7e652017-07-22 22:12:59 -0400252 case '2': this->toggle(fDoCorner); return true;
253 case '3': this->toggle(fDoConic); return true;
reed8b575242014-12-17 01:47:32 -0800254 default: break;
255 }
Hal Canary6cc65e12019-07-03 15:53:04 -0400256 return false;
reeda7a8b102014-12-16 08:07:43 -0800257 }
mtkleinf0599002015-07-13 06:18:39 -0700258
reed8b575242014-12-17 01:47:32 -0800259 void makePath(SkPath* path) {
260 path->moveTo(fPts[0]);
261 for (int i = 1; i < N; ++i) {
262 path->lineTo(fPts[i]);
263 }
264 if (!fDoFrame) {
265 path->close();
266 }
267 }
reeda7a8b102014-12-16 08:07:43 -0800268
mtklein36352bf2015-03-25 18:17:31 -0700269 void onDrawContent(SkCanvas* canvas) override {
reeda7a8b102014-12-16 08:07:43 -0800270 canvas->drawPoints(SkCanvas::kPoints_PointMode, N, fPts, fPtsPaint);
271
272 SkPath path;
reed8b575242014-12-17 01:47:32 -0800273 this->makePath(&path);
reeda7a8b102014-12-16 08:07:43 -0800274
reed8b575242014-12-17 01:47:32 -0800275 if (fDoCorner) {
276 canvas->drawPath(path, fCornerPaint);
reeda7a8b102014-12-16 08:07:43 -0800277 }
reeda7a8b102014-12-16 08:07:43 -0800278
reed8b575242014-12-17 01:47:32 -0800279 canvas->drawPath(path, fSkeletonPaint);
reeda7a8b102014-12-16 08:07:43 -0800280 }
281
Hal Canaryb1f411a2019-08-29 10:39:22 -0400282 Sample::Click* onFindClickHandler(SkScalar x, SkScalar y, skui::ModifierKey modi) override {
reeda7a8b102014-12-16 08:07:43 -0800283 const SkScalar tol = 4;
284 const SkRect r = SkRect::MakeXYWH(x - tol, y - tol, tol * 2, tol * 2);
285 for (int i = 0; i < N; ++i) {
286 if (r.intersects(SkRect::MakeXYWH(fPts[i].fX, fPts[i].fY, 1, 1))) {
Mike Reed0ce3e882020-04-22 11:09:30 -0400287 return new Click([this, i](Click* c) {
288 fPts[i] = c->fCurr;
289 return true;
290 });
reeda7a8b102014-12-16 08:07:43 -0800291 }
292 }
Hal Canaryfcf63592019-07-12 11:32:43 -0400293 return nullptr;
reeda7a8b102014-12-16 08:07:43 -0800294 }
295
296private:
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400297 typedef Sample INHERITED;
reeda7a8b102014-12-16 08:07:43 -0800298};
299DEF_SAMPLE( return new ArcToView; )
Mike Reeda964a292016-11-02 22:09:25 -0400300
301/////////////
302
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400303class FatStroke : public Sample {
Mike Reeda964a292016-11-02 22:09:25 -0400304 bool fClosed, fShowStroke, fShowHidden, fShowSkeleton;
305 int fJoinType, fCapType;
Mike Reed1fba7022016-12-15 10:23:42 -0500306 float fWidth = 30;
Mike Reeda964a292016-11-02 22:09:25 -0400307 SkPaint fPtsPaint, fHiddenPaint, fSkeletonPaint, fStrokePaint;
308public:
309 enum {
310 N = 4
311 };
312 SkPoint fPts[N];
Ben Wagner63fd7602017-10-09 15:45:33 -0400313
Mike Reeda964a292016-11-02 22:09:25 -0400314 FatStroke() : fClosed(false), fShowStroke(true), fShowHidden(false), fShowSkeleton(true),
315 fJoinType(0), fCapType(0)
316 {
317 SkRandom rand;
318 for (int i = 0; i < N; ++i) {
319 fPts[i].fX = 20 + rand.nextUScalar1() * 640;
320 fPts[i].fY = 20 + rand.nextUScalar1() * 480;
321 }
Ben Wagner63fd7602017-10-09 15:45:33 -0400322
Mike Reeda964a292016-11-02 22:09:25 -0400323 fPtsPaint.setAntiAlias(true);
324 fPtsPaint.setStrokeWidth(10);
325 fPtsPaint.setStrokeCap(SkPaint::kRound_Cap);
Ben Wagner63fd7602017-10-09 15:45:33 -0400326
Mike Reeda964a292016-11-02 22:09:25 -0400327 fHiddenPaint.setAntiAlias(true);
328 fHiddenPaint.setStyle(SkPaint::kStroke_Style);
329 fHiddenPaint.setColor(0xFF0000FF);
Ben Wagner63fd7602017-10-09 15:45:33 -0400330
Mike Reeda964a292016-11-02 22:09:25 -0400331 fStrokePaint.setAntiAlias(true);
332 fStrokePaint.setStyle(SkPaint::kStroke_Style);
333 fStrokePaint.setStrokeWidth(50);
334 fStrokePaint.setColor(0x8000FF00);
Ben Wagner63fd7602017-10-09 15:45:33 -0400335
Mike Reeda964a292016-11-02 22:09:25 -0400336 fSkeletonPaint.setAntiAlias(true);
337 fSkeletonPaint.setStyle(SkPaint::kStroke_Style);
338 fSkeletonPaint.setColor(SK_ColorRED);
339 }
Ben Wagner63fd7602017-10-09 15:45:33 -0400340
Mike Reeda964a292016-11-02 22:09:25 -0400341 void toggle(bool& value) {
342 value = !value;
Mike Reeda964a292016-11-02 22:09:25 -0400343 }
Ben Wagner63fd7602017-10-09 15:45:33 -0400344
Mike Reeda964a292016-11-02 22:09:25 -0400345 void toggle3(int& value) {
346 value = (value + 1) % 3;
Mike Reeda964a292016-11-02 22:09:25 -0400347 }
Ben Wagner63fd7602017-10-09 15:45:33 -0400348
Mike Reeda964a292016-11-02 22:09:25 -0400349protected:
Hal Canary8a027312019-07-03 10:55:44 -0400350 SkString name() override { return SkString("FatStroke"); }
351
Hal Canary6cc65e12019-07-03 15:53:04 -0400352 bool onChar(SkUnichar uni) override {
Mike Reeda964a292016-11-02 22:09:25 -0400353 switch (uni) {
354 case '1': this->toggle(fShowSkeleton); return true;
355 case '2': this->toggle(fShowStroke); return true;
356 case '3': this->toggle(fShowHidden); return true;
357 case '4': this->toggle3(fJoinType); return true;
358 case '5': this->toggle3(fCapType); return true;
359 case '6': this->toggle(fClosed); return true;
Brian Osmanede860e2017-11-22 16:36:07 -0500360 case '-': fWidth -= 5; return true;
361 case '=': fWidth += 5; return true;
Mike Reeda964a292016-11-02 22:09:25 -0400362 default: break;
363 }
Hal Canary6cc65e12019-07-03 15:53:04 -0400364 return false;
Mike Reeda964a292016-11-02 22:09:25 -0400365 }
Ben Wagner63fd7602017-10-09 15:45:33 -0400366
Mike Reeda964a292016-11-02 22:09:25 -0400367 void makePath(SkPath* path) {
368 path->moveTo(fPts[0]);
369 for (int i = 1; i < N; ++i) {
370 path->lineTo(fPts[i]);
371 }
372 if (fClosed) {
373 path->close();
374 }
375 }
Ben Wagner63fd7602017-10-09 15:45:33 -0400376
Mike Reeda964a292016-11-02 22:09:25 -0400377 void onDrawContent(SkCanvas* canvas) override {
378 canvas->drawColor(0xFFEEEEEE);
379
380 SkPath path;
381 this->makePath(&path);
382
Mike Reed1fba7022016-12-15 10:23:42 -0500383 fStrokePaint.setStrokeWidth(fWidth);
Mike Reeda964a292016-11-02 22:09:25 -0400384 fStrokePaint.setStrokeJoin((SkPaint::Join)fJoinType);
385 fStrokePaint.setStrokeCap((SkPaint::Cap)fCapType);
386
387 if (fShowStroke) {
388 canvas->drawPath(path, fStrokePaint);
389 }
390 if (fShowHidden) {
391 SkPath hidden;
392 fStrokePaint.getFillPath(path, &hidden);
393 canvas->drawPath(hidden, fHiddenPaint);
394 }
395 if (fShowSkeleton) {
396 canvas->drawPath(path, fSkeletonPaint);
397 }
398 canvas->drawPoints(SkCanvas::kPoints_PointMode, N, fPts, fPtsPaint);
399 }
Ben Wagner63fd7602017-10-09 15:45:33 -0400400
Hal Canaryb1f411a2019-08-29 10:39:22 -0400401 Sample::Click* onFindClickHandler(SkScalar x, SkScalar y, skui::ModifierKey modi) override {
Mike Reeda964a292016-11-02 22:09:25 -0400402 const SkScalar tol = 4;
403 const SkRect r = SkRect::MakeXYWH(x - tol, y - tol, tol * 2, tol * 2);
404 for (int i = 0; i < N; ++i) {
405 if (r.intersects(SkRect::MakeXYWH(fPts[i].fX, fPts[i].fY, 1, 1))) {
Mike Reed0ce3e882020-04-22 11:09:30 -0400406 return new Click([this, i](Click* c) {
407 fPts[i] = c->fCurr;
408 return true;
409 });
Mike Reeda964a292016-11-02 22:09:25 -0400410 }
411 }
Hal Canaryfcf63592019-07-12 11:32:43 -0400412 return nullptr;
Mike Reeda964a292016-11-02 22:09:25 -0400413 }
Ben Wagner63fd7602017-10-09 15:45:33 -0400414
Mike Reeda964a292016-11-02 22:09:25 -0400415private:
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400416 typedef Sample INHERITED;
Mike Reeda964a292016-11-02 22:09:25 -0400417};
418DEF_SAMPLE( return new FatStroke; )
Mike Reed66fa4562018-04-06 19:56:50 -0400419
420static int compute_parallel_to_base(const SkPoint pts[4], SkScalar t[2]) {
421 // F = At^3 + Bt^2 + Ct + D
422 SkVector A = pts[3] - pts[0] + (pts[1] - pts[2]) * 3.0f;
423 SkVector B = (pts[0] - pts[1] - pts[1] + pts[2]) * 3.0f;
424 SkVector C = (pts[1] - pts[0]) * 3.0f;
425 SkVector DA = pts[3] - pts[0];
426
427 // F' = 3At^2 + 2Bt + C
428 SkScalar a = 3 * A.cross(DA);
429 SkScalar b = 2 * B.cross(DA);
430 SkScalar c = C.cross(DA);
431
432 int n = SkFindUnitQuadRoots(a, b, c, t);
433 SkString str;
434 for (int i = 0; i < n; ++i) {
435 str.appendf(" %g", t[i]);
436 }
437 SkDebugf("roots %s\n", str.c_str());
438 return n;
439}
440
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400441class CubicCurve : public Sample {
Mike Reed66fa4562018-04-06 19:56:50 -0400442public:
443 enum {
444 N = 4
445 };
446 SkPoint fPts[N];
447
448 CubicCurve() {
449 SkRandom rand;
450 for (int i = 0; i < N; ++i) {
451 fPts[i].fX = 20 + rand.nextUScalar1() * 640;
452 fPts[i].fY = 20 + rand.nextUScalar1() * 480;
453 }
454 }
455
456protected:
Hal Canary8a027312019-07-03 10:55:44 -0400457 SkString name() override { return SkString("CubicCurve"); }
Mike Reed66fa4562018-04-06 19:56:50 -0400458
459 void onDrawContent(SkCanvas* canvas) override {
460 SkPaint paint;
461 paint.setAntiAlias(true);
462
463 {
464 SkPath path;
465 path.moveTo(fPts[0]);
466 path.cubicTo(fPts[1], fPts[2], fPts[3]);
467 paint.setStyle(SkPaint::kStroke_Style);
468 canvas->drawPath(path, paint);
469 }
470
471 {
472 paint.setColor(SK_ColorRED);
473 SkScalar t[2];
474 int n = compute_parallel_to_base(fPts, t);
475 SkPoint loc;
476 SkVector tan;
477 for (int i = 0; i < n; ++i) {
478 SkEvalCubicAt(fPts, t[i], &loc, &tan, nullptr);
479 tan.setLength(30);
480 canvas->drawLine(loc - tan, loc + tan, paint);
481 }
482 paint.setStrokeWidth(0.5f);
483 canvas->drawLine(fPts[0], fPts[3], paint);
484
485 paint.setColor(SK_ColorBLUE);
486 paint.setStrokeWidth(6);
487 SkEvalCubicAt(fPts, 0.5f, &loc, nullptr, nullptr);
488 canvas->drawPoint(loc, paint);
489
490 paint.setColor(0xFF008800);
491 SkEvalCubicAt(fPts, 1.0f/3, &loc, nullptr, nullptr);
492 canvas->drawPoint(loc, paint);
493 SkEvalCubicAt(fPts, 2.0f/3, &loc, nullptr, nullptr);
494 canvas->drawPoint(loc, paint);
495
496 // n = SkFindCubicInflections(fPts, t);
497 // printf("inflections %d %g %g\n", n, t[0], t[1]);
498 }
499
500 {
501 paint.setStyle(SkPaint::kFill_Style);
502 paint.setColor(SK_ColorRED);
503 for (SkPoint p : fPts) {
504 canvas->drawCircle(p.fX, p.fY, 8, paint);
505 }
506 }
507 }
508
Hal Canaryb1f411a2019-08-29 10:39:22 -0400509 Sample::Click* onFindClickHandler(SkScalar x, SkScalar y, skui::ModifierKey modi) override {
Mike Reed66fa4562018-04-06 19:56:50 -0400510 const SkScalar tol = 8;
511 const SkRect r = SkRect::MakeXYWH(x - tol, y - tol, tol * 2, tol * 2);
512 for (int i = 0; i < N; ++i) {
513 if (r.intersects(SkRect::MakeXYWH(fPts[i].fX, fPts[i].fY, 1, 1))) {
Mike Reed0ce3e882020-04-22 11:09:30 -0400514 return new Click([this, i](Click* c) {
515 fPts[i] = c->fCurr;
516 return true;
517 });
Mike Reed66fa4562018-04-06 19:56:50 -0400518 }
519 }
520 return this->INHERITED::onFindClickHandler(x, y, modi);
521 }
522
523private:
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400524 typedef Sample INHERITED;
Mike Reed66fa4562018-04-06 19:56:50 -0400525};
526DEF_SAMPLE( return new CubicCurve; )
527
528static SkPoint lerp(SkPoint a, SkPoint b, float t) {
529 return a * (1 - t) + b * t;
530}
531
Mike Reed83dc4732018-04-16 17:07:07 -0400532static int find_max_deviation_cubic(const SkPoint src[4], SkScalar ts[2]) {
533 // deviation = F' x (d - a) == 0, solve for t(s)
534 // F = At^3 + Bt^2 + Ct + D
535 // F' = 3At^2 + 2Bt + C
536 // Z = d - a
537 // F' x Z = 3(A x Z)t^2 + 2(B x Z)t + (C x Z)
538 //
539 SkVector A = src[3] + (src[1] - src[2]) * 3 - src[0];
540 SkVector B = (src[2] - src[1] - src[1] + src[0]) * 3;
541 SkVector C = (src[1] - src[0]) * 3;
542 SkVector Z = src[3] - src[0];
543 // now forumlate the quadratic coefficients we need to solve for t : F' x Z
544 return SkFindUnitQuadRoots(3 * A.cross(Z), 2 * B.cross(Z), C.cross(Z), ts);
545}
546
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400547class CubicCurve2 : public Sample {
Mike Reed66fa4562018-04-06 19:56:50 -0400548public:
549 enum {
550 N = 7
551 };
552 SkPoint fPts[N];
553 SkPoint* fQuad = fPts + 4;
554 SkScalar fT = 0.5f;
555 bool fShowSub = false;
Mike Reed48bf6ab2018-04-15 20:04:45 -0400556 bool fShowFlatness = false;
Mike Reed83dc4732018-04-16 17:07:07 -0400557 SkScalar fScale = 0.75;
Mike Reed66fa4562018-04-06 19:56:50 -0400558
559 CubicCurve2() {
560 fPts[0] = { 90, 300 };
561 fPts[1] = { 30, 60 };
562 fPts[2] = { 250, 30 };
563 fPts[3] = { 350, 200 };
564
565 fQuad[0] = fPts[0] + SkVector{ 300, 0};
566 fQuad[1] = fPts[1] + SkVector{ 300, 0};
567 fQuad[2] = fPts[2] + SkVector{ 300, 0};
568 }
569
570protected:
Hal Canary8a027312019-07-03 10:55:44 -0400571 SkString name() override { return SkString("CubicCurve2"); }
572
Hal Canary6cc65e12019-07-03 15:53:04 -0400573 bool onChar(SkUnichar uni) override {
Mike Reed66fa4562018-04-06 19:56:50 -0400574 switch (uni) {
575 case 's': fShowSub = !fShowSub; break;
Mike Reed48bf6ab2018-04-15 20:04:45 -0400576 case 'f': fShowFlatness = !fShowFlatness; break;
Mike Reed66fa4562018-04-06 19:56:50 -0400577 case '-': fT -= 1.0f / 32; break;
578 case '=': fT += 1.0f / 32; break;
Hal Canary6cc65e12019-07-03 15:53:04 -0400579 default: return false;
Mike Reed66fa4562018-04-06 19:56:50 -0400580 }
581 fT = std::min(1.0f, std::max(0.0f, fT));
582 return true;
Mike Reed66fa4562018-04-06 19:56:50 -0400583 }
584
585 void showFrame(SkCanvas* canvas, const SkPoint pts[], int count, const SkPaint& p) {
586 SkPaint paint(p);
587 SkPoint storage[3 + 2 + 1];
588 SkPoint* tmp = storage;
589 const SkPoint* prev = pts;
590 int n = count;
591 for (int n = count; n > 0; --n) {
592 for (int i = 0; i < n; ++i) {
593 canvas->drawLine(prev[i], prev[i+1], paint);
594 tmp[i] = lerp(prev[i], prev[i+1], fT);
595 }
596 prev = tmp;
597 tmp += n;
598 }
599
600 paint.setColor(SK_ColorBLUE);
601 paint.setStyle(SkPaint::kFill_Style);
602 n = tmp - storage;
603 for (int i = 0; i < n; ++i) {
604 canvas->drawCircle(storage[i].fX, storage[i].fY, 4, paint);
605 }
606 }
Hal Canaryd4998172018-07-11 11:31:34 -0400607
Mike Reed48bf6ab2018-04-15 20:04:45 -0400608 void showFlattness(SkCanvas* canvas) {
609 SkPaint paint;
610 paint.setStyle(SkPaint::kStroke_Style);
611 paint.setAntiAlias(true);
612
Mike Reeddccfa352018-04-17 17:10:51 -0400613 SkPaint paint2(paint);
614 paint2.setColor(0xFF008800);
615
Mike Reed48bf6ab2018-04-15 20:04:45 -0400616 paint.setColor(0xFF888888);
617 canvas->drawLine(fPts[0], fPts[3], paint);
618 canvas->drawLine(fQuad[0], fQuad[2], paint);
619
620 paint.setColor(0xFF0000FF);
621 SkPoint pts[2];
622 pts[0] = (fQuad[0] + fQuad[1] + fQuad[1] + fQuad[2])*0.25;
623 pts[1] = (fQuad[0] + fQuad[2]) * 0.5;
624 canvas->drawLine(pts[0], pts[1], paint);
625
Mike Reed83dc4732018-04-16 17:07:07 -0400626 // cubic
627
628 SkVector v0 = (fPts[0] - fPts[1] - fPts[1] + fPts[2]) * fScale;
629 SkVector v1 = (fPts[1] - fPts[2] - fPts[2] + fPts[3]) * fScale;
Mike Reeddccfa352018-04-17 17:10:51 -0400630 SkVector v = (v0 + v1) * 0.5f;
Mike Reed83dc4732018-04-16 17:07:07 -0400631
632 SkPoint anchor;
633 SkScalar ts[2];
634 int n = find_max_deviation_cubic(fPts, ts);
635 if (n > 0) {
636 SkEvalCubicAt(fPts, ts[0], &anchor, nullptr, nullptr);
Mike Reeddccfa352018-04-17 17:10:51 -0400637 canvas->drawLine(anchor, anchor + v, paint2);
Mike Reed83dc4732018-04-16 17:07:07 -0400638 canvas->drawLine(anchor, anchor + v0, paint);
639 if (n == 2) {
640 SkEvalCubicAt(fPts, ts[1], &anchor, nullptr, nullptr);
Mike Reeddccfa352018-04-17 17:10:51 -0400641 canvas->drawLine(anchor, anchor + v, paint2);
Mike Reed83dc4732018-04-16 17:07:07 -0400642 }
643 canvas->drawLine(anchor, anchor + v1, paint);
644 }
645 // not sure we can get here
Mike Reed48bf6ab2018-04-15 20:04:45 -0400646 }
Mike Reed66fa4562018-04-06 19:56:50 -0400647
648 void onDrawContent(SkCanvas* canvas) override {
649 SkPaint paint;
650 paint.setAntiAlias(true);
651
652 {
653 paint.setStyle(SkPaint::kStroke_Style);
654 SkPath path;
655 path.moveTo(fPts[0]);
656 path.cubicTo(fPts[1], fPts[2], fPts[3]);
657 path.moveTo(fQuad[0]);
658 path.quadTo(fQuad[1], fQuad[2]);
659 canvas->drawPath(path, paint);
660 }
661
662 if (fShowSub) {
663 paint.setColor(SK_ColorRED);
664 paint.setStrokeWidth(1.7f);
665 this->showFrame(canvas, fPts, 3, paint);
666 this->showFrame(canvas, fQuad, 2, paint);
667
Mike Reed66fa4562018-04-06 19:56:50 -0400668 paint.setColor(SK_ColorBLACK);
669 paint.setStyle(SkPaint::kFill_Style);
Hal Canary4484b8f2019-01-08 14:00:08 -0500670 SkFont font(nullptr, 20);
671 canvas->drawString(SkStringPrintf("t = %g", fT), 20, 20, font, paint);
Mike Reed66fa4562018-04-06 19:56:50 -0400672 }
673
Mike Reed48bf6ab2018-04-15 20:04:45 -0400674 if (fShowFlatness) {
675 this->showFlattness(canvas);
676 }
677
Mike Reed66fa4562018-04-06 19:56:50 -0400678 paint.setStyle(SkPaint::kFill_Style);
679 paint.setColor(SK_ColorRED);
680 for (SkPoint p : fPts) {
681 canvas->drawCircle(p.fX, p.fY, 7, paint);
682 }
Mike Reedfb65db12018-04-19 19:18:35 -0400683
684 {
685 SkScalar ts[2];
686 int n = SkFindCubicInflections(fPts, ts);
687 for (int i = 0; i < n; ++i) {
688 SkPoint p;
689 SkEvalCubicAt(fPts, ts[i], &p, nullptr, nullptr);
690 canvas->drawCircle(p.fX, p.fY, 3, paint);
691 }
692 }
693
Mike Reed66fa4562018-04-06 19:56:50 -0400694 }
695
Hal Canaryb1f411a2019-08-29 10:39:22 -0400696 Sample::Click* onFindClickHandler(SkScalar x, SkScalar y, skui::ModifierKey modi) override {
Mike Reed66fa4562018-04-06 19:56:50 -0400697 const SkScalar tol = 8;
698 const SkRect r = SkRect::MakeXYWH(x - tol, y - tol, tol * 2, tol * 2);
699 for (int i = 0; i < N; ++i) {
700 if (r.intersects(SkRect::MakeXYWH(fPts[i].fX, fPts[i].fY, 1, 1))) {
Mike Reed0ce3e882020-04-22 11:09:30 -0400701 return new Click([this, i](Click* c) {
702 fPts[i] = c->fCurr;
703 return true;
704 });
Mike Reed66fa4562018-04-06 19:56:50 -0400705 }
706 }
707 return this->INHERITED::onFindClickHandler(x, y, modi);
708 }
709
710private:
Ben Wagnerb2c4ea62018-08-08 11:36:17 -0400711 typedef Sample INHERITED;
Mike Reed66fa4562018-04-06 19:56:50 -0400712};
713DEF_SAMPLE( return new CubicCurve2; )
714