blob: 6d17026fea2f5343649fd30986e0c97acde9a005 [file] [log] [blame]
caryclark88c748a2015-02-18 10:56:00 -08001/*
2 * Copyright 2012 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
caryclark88c748a2015-02-18 10:56:00 -08008#include "SampleCode.h"
Ben Wagnercb3d49c2018-03-14 15:07:43 -04009#include "SkBlendMode.h"
caryclark88c748a2015-02-18 10:56:00 -080010#include "SkCanvas.h"
Ben Wagnercb3d49c2018-03-14 15:07:43 -040011#include "SkColor.h"
caryclarkb6474dd2016-01-19 08:07:49 -080012#include "SkGeometry.h"
Ben Wagnercb3d49c2018-03-14 15:07:43 -040013#include "SkImageInfo.h"
14#include "SkMatrix.h"
15#include "SkPaint.h"
16#include "SkPath.h"
caryclark88c748a2015-02-18 10:56:00 -080017#include "SkPathMeasure.h"
Ben Wagnercb3d49c2018-03-14 15:07:43 -040018#include "SkPoint.h"
Cary Clarkdf429f32017-11-08 11:44:31 -050019#include "SkPointPriv.h"
caryclark88c748a2015-02-18 10:56:00 -080020#include "SkRRect.h"
Ben Wagnercb3d49c2018-03-14 15:07:43 -040021#include "SkRect.h"
22#include "SkRefCnt.h"
23#include "SkScalar.h"
24#include "SkShader.h"
25#include "SkString.h"
26#include "SkStroke.h"
caryclark88c748a2015-02-18 10:56:00 -080027#include "SkSurface.h"
Ben Wagnercb3d49c2018-03-14 15:07:43 -040028#include "SkTArray.h"
29#include "SkTemplates.h"
30#include "SkTypes.h"
31#include "SkView.h"
32#include "sk_tool_utils.h"
33
34#include <cfloat>
35
36class SkEvent;
caryclark88c748a2015-02-18 10:56:00 -080037
38static bool hittest(const SkPoint& target, SkScalar x, SkScalar y) {
39 const SkScalar TOL = 7;
40 return SkPoint::Distance(target, SkPoint::Make(x, y)) <= TOL;
41}
42
43static int getOnCurvePoints(const SkPath& path, SkPoint storage[]) {
44 SkPath::RawIter iter(path);
45 SkPoint pts[4];
46 SkPath::Verb verb;
47
48 int count = 0;
49 while ((verb = iter.next(pts)) != SkPath::kDone_Verb) {
50 switch (verb) {
51 case SkPath::kMove_Verb:
52 case SkPath::kLine_Verb:
53 case SkPath::kQuad_Verb:
54 case SkPath::kConic_Verb:
55 case SkPath::kCubic_Verb:
56 storage[count++] = pts[0];
57 break;
58 default:
59 break;
60 }
61 }
62 return count;
63}
64
65static void getContourCounts(const SkPath& path, SkTArray<int>* contourCounts) {
66 SkPath::RawIter iter(path);
67 SkPoint pts[4];
68 SkPath::Verb verb;
69
70 int count = 0;
71 while ((verb = iter.next(pts)) != SkPath::kDone_Verb) {
72 switch (verb) {
73 case SkPath::kMove_Verb:
74 case SkPath::kLine_Verb:
75 count += 1;
76 break;
77 case SkPath::kQuad_Verb:
78 case SkPath::kConic_Verb:
79 count += 2;
80 break;
81 case SkPath::kCubic_Verb:
82 count += 3;
83 break;
84 case SkPath::kClose_Verb:
85 contourCounts->push_back(count);
86 count = 0;
87 break;
88 default:
89 break;
90 }
91 }
92 if (count > 0) {
93 contourCounts->push_back(count);
94 }
95}
96
reede8f30622016-03-23 18:59:25 -070097static void erase(const sk_sp<SkSurface>& surface) {
caryclark88651ae2016-01-20 11:55:11 -080098 SkCanvas* canvas = surface->getCanvas();
99 if (canvas) {
100 canvas->clear(SK_ColorTRANSPARENT);
101 }
caryclark88c748a2015-02-18 10:56:00 -0800102}
103
104struct StrokeTypeButton {
105 SkRect fBounds;
106 char fLabel;
107 bool fEnabled;
108};
109
caryclark04e4d082015-02-20 06:33:57 -0800110struct CircleTypeButton : public StrokeTypeButton {
111 bool fFill;
112};
113
caryclark88c748a2015-02-18 10:56:00 -0800114class QuadStrokerView : public SampleView {
115 enum {
116 SKELETON_COLOR = 0xFF0000FF,
117 WIREFRAME_COLOR = 0x80FF0000
118 };
119
120 enum {
caryclark88651ae2016-01-20 11:55:11 -0800121 kCount = 18
caryclark88c748a2015-02-18 10:56:00 -0800122 };
123 SkPoint fPts[kCount];
caryclark04e4d082015-02-20 06:33:57 -0800124 SkRect fWeightControl;
caryclark88651ae2016-01-20 11:55:11 -0800125 SkRect fRadiusControl;
caryclark88c748a2015-02-18 10:56:00 -0800126 SkRect fErrorControl;
127 SkRect fWidthControl;
128 SkRect fBounds;
129 SkMatrix fMatrix, fInverse;
reed8a21c9f2016-03-08 18:50:00 -0800130 sk_sp<SkShader> fShader;
reede8f30622016-03-23 18:59:25 -0700131 sk_sp<SkSurface> fMinSurface;
132 sk_sp<SkSurface> fMaxSurface;
caryclark88c748a2015-02-18 10:56:00 -0800133 StrokeTypeButton fCubicButton;
caryclark04e4d082015-02-20 06:33:57 -0800134 StrokeTypeButton fConicButton;
caryclark88c748a2015-02-18 10:56:00 -0800135 StrokeTypeButton fQuadButton;
caryclark88651ae2016-01-20 11:55:11 -0800136 StrokeTypeButton fArcButton;
caryclark88c748a2015-02-18 10:56:00 -0800137 StrokeTypeButton fRRectButton;
caryclark04e4d082015-02-20 06:33:57 -0800138 CircleTypeButton fCircleButton;
caryclark88c748a2015-02-18 10:56:00 -0800139 StrokeTypeButton fTextButton;
140 SkString fText;
141 SkScalar fTextSize;
caryclark04e4d082015-02-20 06:33:57 -0800142 SkScalar fWeight;
caryclark88651ae2016-01-20 11:55:11 -0800143 SkScalar fRadius;
caryclark88c748a2015-02-18 10:56:00 -0800144 SkScalar fWidth, fDWidth;
145 SkScalar fWidthScale;
146 int fW, fH, fZoom;
147 bool fAnimate;
148 bool fDrawRibs;
149 bool fDrawTangents;
caryclarkb6474dd2016-01-19 08:07:49 -0800150 bool fDrawTDivs;
caryclarka76b7a3b2015-05-22 06:26:52 -0700151#ifdef SK_DEBUG
caryclark88c748a2015-02-18 10:56:00 -0800152 #define kStrokerErrorMin 0.001f
153 #define kStrokerErrorMax 5
154#endif
155 #define kWidthMin 1
156 #define kWidthMax 100
157public:
158 QuadStrokerView() {
159 this->setBGColor(SK_ColorLTGRAY);
160
caryclark04e4d082015-02-20 06:33:57 -0800161 fPts[0].set(50, 200); // cubic
caryclark88c748a2015-02-18 10:56:00 -0800162 fPts[1].set(50, 100);
163 fPts[2].set(150, 50);
164 fPts[3].set(300, 50);
165
caryclark04e4d082015-02-20 06:33:57 -0800166 fPts[4].set(350, 200); // conic
caryclark88c748a2015-02-18 10:56:00 -0800167 fPts[5].set(350, 100);
168 fPts[6].set(450, 50);
169
caryclark04e4d082015-02-20 06:33:57 -0800170 fPts[7].set(150, 300); // quad
171 fPts[8].set(150, 200);
172 fPts[9].set(250, 150);
caryclark88c748a2015-02-18 10:56:00 -0800173
caryclark88651ae2016-01-20 11:55:11 -0800174 fPts[10].set(250, 200); // arc
175 fPts[11].set(250, 300);
176 fPts[12].set(150, 350);
caryclark04e4d082015-02-20 06:33:57 -0800177
caryclark88651ae2016-01-20 11:55:11 -0800178 fPts[13].set(200, 200); // rrect
179 fPts[14].set(400, 400);
180
181 fPts[15].set(250, 250); // oval
182 fPts[16].set(450, 450);
caryclark04e4d082015-02-20 06:33:57 -0800183
caryclark88c748a2015-02-18 10:56:00 -0800184 fText = "a";
185 fTextSize = 12;
186 fWidth = 50;
187 fDWidth = 0.25f;
caryclark04e4d082015-02-20 06:33:57 -0800188 fWeight = 1;
caryclark88651ae2016-01-20 11:55:11 -0800189 fRadius = 150;
caryclark88c748a2015-02-18 10:56:00 -0800190
191 fCubicButton.fLabel = 'C';
192 fCubicButton.fEnabled = false;
caryclark04e4d082015-02-20 06:33:57 -0800193 fConicButton.fLabel = 'K';
caryclark88651ae2016-01-20 11:55:11 -0800194 fConicButton.fEnabled = false;
caryclark88c748a2015-02-18 10:56:00 -0800195 fQuadButton.fLabel = 'Q';
196 fQuadButton.fEnabled = false;
caryclark88651ae2016-01-20 11:55:11 -0800197 fArcButton.fLabel = 'A';
198 fArcButton.fEnabled = true;
caryclark88c748a2015-02-18 10:56:00 -0800199 fRRectButton.fLabel = 'R';
200 fRRectButton.fEnabled = false;
caryclark04e4d082015-02-20 06:33:57 -0800201 fCircleButton.fLabel = 'O';
caryclark88651ae2016-01-20 11:55:11 -0800202 fCircleButton.fEnabled = true;
203 fCircleButton.fFill = true;
caryclark88c748a2015-02-18 10:56:00 -0800204 fTextButton.fLabel = 'T';
caryclark04e4d082015-02-20 06:33:57 -0800205 fTextButton.fEnabled = false;
caryclark88651ae2016-01-20 11:55:11 -0800206 fAnimate = false;
caryclark88c748a2015-02-18 10:56:00 -0800207 setAsNeeded();
208 }
209
210protected:
mtklein36352bf2015-03-25 18:17:31 -0700211 bool onQuery(SkEvent* evt) override {
caryclark88c748a2015-02-18 10:56:00 -0800212 if (SampleCode::TitleQ(*evt)) {
213 SampleCode::TitleR(evt, "QuadStroker");
214 return true;
215 }
216 SkUnichar uni;
217 if (fTextButton.fEnabled && SampleCode::CharQ(*evt, &uni)) {
218 switch (uni) {
219 case ' ':
220 fText = "";
221 break;
222 case '-':
223 fTextSize = SkTMax(1.0f, fTextSize - 1);
224 break;
225 case '+':
226 case '=':
227 fTextSize += 1;
228 break;
229 default:
230 fText.appendUnichar(uni);
231 }
caryclark88c748a2015-02-18 10:56:00 -0800232 return true;
233 }
234 return this->INHERITED::onQuery(evt);
235 }
236
mtklein36352bf2015-03-25 18:17:31 -0700237 void onSizeChange() override {
caryclark88651ae2016-01-20 11:55:11 -0800238 fRadiusControl.setXYWH(this->width() - 200, 30, 30, 400);
caryclark04e4d082015-02-20 06:33:57 -0800239 fWeightControl.setXYWH(this->width() - 150, 30, 30, 400);
caryclark88c748a2015-02-18 10:56:00 -0800240 fErrorControl.setXYWH(this->width() - 100, 30, 30, 400);
241 fWidthControl.setXYWH(this->width() - 50, 30, 30, 400);
caryclark04e4d082015-02-20 06:33:57 -0800242 int buttonOffset = 450;
243 fCubicButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
244 buttonOffset += 50;
245 fConicButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
246 buttonOffset += 50;
247 fQuadButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
248 buttonOffset += 50;
caryclark88651ae2016-01-20 11:55:11 -0800249 fArcButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
250 buttonOffset += 50;
caryclark04e4d082015-02-20 06:33:57 -0800251 fRRectButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
252 buttonOffset += 50;
253 fCircleButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
254 buttonOffset += 50;
255 fTextButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
caryclark88c748a2015-02-18 10:56:00 -0800256 this->INHERITED::onSizeChange();
257 }
258
259 void copyMinToMax() {
260 erase(fMaxSurface);
261 SkCanvas* canvas = fMaxSurface->getCanvas();
262 canvas->save();
263 canvas->concat(fMatrix);
halcanary96fcdcc2015-08-27 07:41:13 -0700264 fMinSurface->draw(canvas, 0, 0, nullptr);
caryclark88c748a2015-02-18 10:56:00 -0800265 canvas->restore();
266
267 SkPaint paint;
reed374772b2016-10-05 17:33:02 -0700268 paint.setBlendMode(SkBlendMode::kClear);
caryclark88c748a2015-02-18 10:56:00 -0800269 for (int iy = 1; iy < fH; ++iy) {
270 SkScalar y = SkIntToScalar(iy * fZoom);
271 canvas->drawLine(0, y - SK_ScalarHalf, 999, y - SK_ScalarHalf, paint);
272 }
273 for (int ix = 1; ix < fW; ++ix) {
274 SkScalar x = SkIntToScalar(ix * fZoom);
275 canvas->drawLine(x - SK_ScalarHalf, 0, x - SK_ScalarHalf, 999, paint);
276 }
277 }
278
279 void setWHZ(int width, int height, int zoom) {
280 fZoom = zoom;
281 fBounds.set(0, 0, SkIntToScalar(width * zoom), SkIntToScalar(height * zoom));
282 fMatrix.setScale(SkIntToScalar(zoom), SkIntToScalar(zoom));
283 fInverse.setScale(SK_Scalar1 / zoom, SK_Scalar1 / zoom);
reed8a21c9f2016-03-08 18:50:00 -0800284 fShader = sk_tool_utils::create_checkerboard_shader(0xFFCCCCCC, 0xFFFFFFFF, zoom);
caryclark88c748a2015-02-18 10:56:00 -0800285
286 SkImageInfo info = SkImageInfo::MakeN32Premul(width, height);
reede8f30622016-03-23 18:59:25 -0700287 fMinSurface = SkSurface::MakeRaster(info);
caryclark88c748a2015-02-18 10:56:00 -0800288 info = info.makeWH(width * zoom, height * zoom);
reede8f30622016-03-23 18:59:25 -0700289 fMaxSurface = SkSurface::MakeRaster(info);
caryclark88c748a2015-02-18 10:56:00 -0800290 }
291
292 void draw_points(SkCanvas* canvas, const SkPath& path, SkColor color,
293 bool show_lines) {
294 SkPaint paint;
295 paint.setColor(color);
296 paint.setAlpha(0x80);
297 paint.setAntiAlias(true);
298 int n = path.countPoints();
299 SkAutoSTArray<32, SkPoint> pts(n);
300 if (show_lines && fDrawTangents) {
301 SkTArray<int> contourCounts;
302 getContourCounts(path, &contourCounts);
303 SkPoint* ptPtr = pts.get();
304 for (int i = 0; i < contourCounts.count(); ++i) {
305 int count = contourCounts[i];
306 path.getPoints(ptPtr, count);
307 canvas->drawPoints(SkCanvas::kPolygon_PointMode, count, ptPtr, paint);
308 ptPtr += count;
309 }
310 } else {
311 n = getOnCurvePoints(path, pts.get());
312 }
313 paint.setStrokeWidth(5);
314 canvas->drawPoints(SkCanvas::kPoints_PointMode, n, pts.get(), paint);
315 }
316
317 void draw_ribs(SkCanvas* canvas, const SkPath& path, SkScalar width,
318 SkColor color) {
319 const SkScalar radius = width / 2;
320
321 SkPathMeasure meas(path, false);
322 SkScalar total = meas.getLength();
323
324 SkScalar delta = 8;
caryclarkb6474dd2016-01-19 08:07:49 -0800325 SkPaint paint, labelP;
caryclark88c748a2015-02-18 10:56:00 -0800326 paint.setColor(color);
caryclarkb6474dd2016-01-19 08:07:49 -0800327 labelP.setColor(color & 0xff5f9f5f);
caryclark88c748a2015-02-18 10:56:00 -0800328 SkPoint pos, tan;
caryclarkb6474dd2016-01-19 08:07:49 -0800329 int index = 0;
caryclark88c748a2015-02-18 10:56:00 -0800330 for (SkScalar dist = 0; dist <= total; dist += delta) {
331 if (meas.getPosTan(dist, &pos, &tan)) {
332 tan.scale(radius);
Cary Clarkdf429f32017-11-08 11:44:31 -0500333 SkPointPriv::RotateCCW(&tan);
caryclark88c748a2015-02-18 10:56:00 -0800334 canvas->drawLine(pos.x() + tan.x(), pos.y() + tan.y(),
335 pos.x() - tan.x(), pos.y() - tan.y(), paint);
caryclarkb6474dd2016-01-19 08:07:49 -0800336 if (0 == index % 10) {
337 SkString label;
338 label.appendS32(index);
339 SkRect dot = SkRect::MakeXYWH(pos.x() - 2, pos.y() - 2, 4, 4);
340 canvas->drawRect(dot, labelP);
Cary Clark2a475ea2017-04-28 15:35:12 -0400341 canvas->drawString(label,
caryclarkb6474dd2016-01-19 08:07:49 -0800342 pos.x() - tan.x() * 1.25f, pos.y() - tan.y() * 1.25f, labelP);
343 }
344 }
345 ++index;
346 }
347 }
348
349 void draw_t_divs(SkCanvas* canvas, const SkPath& path, SkScalar width, SkColor color) {
350 const SkScalar radius = width / 2;
351 SkPaint paint;
352 paint.setColor(color);
353 SkPathMeasure meas(path, false);
354 SkScalar total = meas.getLength();
355 SkScalar delta = 8;
356 int ribs = 0;
357 for (SkScalar dist = 0; dist <= total; dist += delta) {
358 ++ribs;
359 }
360 SkPath::RawIter iter(path);
361 SkPoint pts[4];
362 if (SkPath::kMove_Verb != iter.next(pts)) {
363 SkASSERT(0);
364 return;
365 }
366 SkPath::Verb verb = iter.next(pts);
367 SkASSERT(SkPath::kLine_Verb <= verb && verb <= SkPath::kCubic_Verb);
368 SkPoint pos, tan;
369 for (int index = 0; index < ribs; ++index) {
370 SkScalar t = (SkScalar) index / ribs;
371 switch (verb) {
372 case SkPath::kLine_Verb:
373 tan = pts[1] - pts[0];
374 pos = pts[0];
375 pos.fX += tan.fX * t;
376 pos.fY += tan.fY * t;
377 break;
378 case SkPath::kQuad_Verb:
379 pos = SkEvalQuadAt(pts, t);
380 tan = SkEvalQuadTangentAt(pts, t);
381 break;
382 case SkPath::kConic_Verb: {
383 SkConic conic(pts, iter.conicWeight());
384 pos = conic.evalAt(t);
385 tan = conic.evalTangentAt(t);
386 } break;
387 case SkPath::kCubic_Verb:
388 SkEvalCubicAt(pts, t, &pos, &tan, nullptr);
389 break;
390 default:
391 SkASSERT(0);
392 return;
393 }
394 tan.setLength(radius);
Cary Clarkdf429f32017-11-08 11:44:31 -0500395 SkPointPriv::RotateCCW(&tan);
caryclarkb6474dd2016-01-19 08:07:49 -0800396 canvas->drawLine(pos.x() + tan.x(), pos.y() + tan.y(),
397 pos.x() - tan.x(), pos.y() - tan.y(), paint);
398 if (0 == index % 10) {
399 SkString label;
400 label.appendS32(index);
Cary Clark2a475ea2017-04-28 15:35:12 -0400401 canvas->drawString(label,
caryclarkb6474dd2016-01-19 08:07:49 -0800402 pos.x() + tan.x() * 1.25f, pos.y() + tan.y() * 1.25f, paint);
caryclark88c748a2015-02-18 10:56:00 -0800403 }
404 }
405 }
406
caryclark04e4d082015-02-20 06:33:57 -0800407 void draw_stroke(SkCanvas* canvas, const SkPath& path, SkScalar width, SkScalar scale,
408 bool drawText) {
caryclark612f70d2015-05-19 11:05:37 -0700409 if (path.isEmpty()) {
caryclark88c748a2015-02-18 10:56:00 -0800410 return;
411 }
caryclark612f70d2015-05-19 11:05:37 -0700412 SkRect bounds = path.getBounds();
halcanary9d524f22016-03-29 09:03:52 -0700413 this->setWHZ(SkScalarCeilToInt(bounds.right()), drawText
caryclark04e4d082015-02-20 06:33:57 -0800414 ? SkScalarRoundToInt(scale * 3 / 2) : SkScalarRoundToInt(scale),
415 SkScalarRoundToInt(950.0f / scale));
caryclark88c748a2015-02-18 10:56:00 -0800416 erase(fMinSurface);
417 SkPaint paint;
418 paint.setColor(0x1f1f0f0f);
caryclark88c748a2015-02-18 10:56:00 -0800419 paint.setStyle(SkPaint::kStroke_Style);
caryclark04e4d082015-02-20 06:33:57 -0800420 paint.setStrokeWidth(width * scale * scale);
caryclark88c748a2015-02-18 10:56:00 -0800421 paint.setColor(0x3f0f1f3f);
caryclark04e4d082015-02-20 06:33:57 -0800422 if (drawText) {
423 fMinSurface->getCanvas()->drawPath(path, paint);
424 this->copyMinToMax();
halcanary96fcdcc2015-08-27 07:41:13 -0700425 fMaxSurface->draw(canvas, 0, 0, nullptr);
caryclark04e4d082015-02-20 06:33:57 -0800426 }
caryclark88c748a2015-02-18 10:56:00 -0800427 paint.setAntiAlias(true);
428 paint.setStyle(SkPaint::kStroke_Style);
429 paint.setStrokeWidth(1);
430
431 paint.setColor(SKELETON_COLOR);
432 SkPath scaled;
433 SkMatrix matrix;
434 matrix.reset();
caryclark04e4d082015-02-20 06:33:57 -0800435 matrix.setScale(950 / scale, 950 / scale);
caryclark88c748a2015-02-18 10:56:00 -0800436 if (drawText) {
437 path.transform(matrix, &scaled);
438 } else {
439 scaled = path;
440 }
441 canvas->drawPath(scaled, paint);
442 draw_points(canvas, scaled, SKELETON_COLOR, true);
443
444 if (fDrawRibs) {
445 draw_ribs(canvas, scaled, width, 0xFF00FF00);
446 }
447
caryclarkb6474dd2016-01-19 08:07:49 -0800448 if (fDrawTDivs) {
449 draw_t_divs(canvas, scaled, width, 0xFF3F3F00);
450 }
451
caryclark88c748a2015-02-18 10:56:00 -0800452 SkPath fill;
453
454 SkPaint p;
455 p.setStyle(SkPaint::kStroke_Style);
caryclark04e4d082015-02-20 06:33:57 -0800456 if (drawText) {
457 p.setStrokeWidth(width * scale * scale);
458 } else {
459 p.setStrokeWidth(width);
460 }
caryclark88c748a2015-02-18 10:56:00 -0800461 p.getFillPath(path, &fill);
462 SkPath scaledFill;
463 if (drawText) {
464 fill.transform(matrix, &scaledFill);
465 } else {
466 scaledFill = fill;
467 }
468 paint.setColor(WIREFRAME_COLOR);
469 canvas->drawPath(scaledFill, paint);
470 draw_points(canvas, scaledFill, WIREFRAME_COLOR, false);
471 }
472
caryclark04e4d082015-02-20 06:33:57 -0800473 void draw_fill(SkCanvas* canvas, const SkRect& rect, SkScalar width) {
474 if (rect.isEmpty()) {
475 return;
476 }
477 SkPaint paint;
478 paint.setColor(0x1f1f0f0f);
479 paint.setStyle(SkPaint::kStroke_Style);
480 paint.setStrokeWidth(width);
481 SkPath path;
482 SkScalar maxSide = SkTMax(rect.width(), rect.height()) / 2;
483 SkPoint center = { rect.fLeft + maxSide, rect.fTop + maxSide };
484 path.addCircle(center.fX, center.fY, maxSide);
485 canvas->drawPath(path, paint);
486 paint.setStyle(SkPaint::kFill_Style);
487 path.reset();
488 path.addCircle(center.fX, center.fY, maxSide - width / 2);
489 paint.setColor(0x3f0f1f3f);
490 canvas->drawPath(path, paint);
491 path.reset();
492 path.setFillType(SkPath::kEvenOdd_FillType);
493 path.addCircle(center.fX, center.fY, maxSide + width / 2);
halcanary9d524f22016-03-29 09:03:52 -0700494 SkRect outside = SkRect::MakeXYWH(center.fX - maxSide - width, center.fY - maxSide - width,
caryclark04e4d082015-02-20 06:33:57 -0800495 (maxSide + width) * 2, (maxSide + width) * 2);
496 path.addRect(outside);
497 canvas->drawPath(path, paint);
498 }
499
caryclark88c748a2015-02-18 10:56:00 -0800500 void draw_button(SkCanvas* canvas, const StrokeTypeButton& button) {
501 SkPaint paint;
502 paint.setAntiAlias(true);
503 paint.setStyle(SkPaint::kStroke_Style);
504 paint.setColor(button.fEnabled ? 0xFF3F0000 : 0x6F3F0000);
505 canvas->drawRect(button.fBounds, paint);
506 paint.setTextSize(25.0f);
507 paint.setColor(button.fEnabled ? 0xFF3F0000 : 0x6F3F0000);
508 paint.setTextAlign(SkPaint::kCenter_Align);
509 paint.setStyle(SkPaint::kFill_Style);
510 canvas->drawText(&button.fLabel, 1, button.fBounds.centerX(), button.fBounds.fBottom - 5,
511 paint);
512 }
513
514 void draw_control(SkCanvas* canvas, const SkRect& bounds, SkScalar value,
515 SkScalar min, SkScalar max, const char* name) {
516 SkPaint paint;
517 paint.setAntiAlias(true);
518 paint.setStyle(SkPaint::kStroke_Style);
519 canvas->drawRect(bounds, paint);
520 SkScalar scale = max - min;
521 SkScalar yPos = bounds.fTop + (value - min) * bounds.height() / scale;
522 paint.setColor(0xFFFF0000);
523 canvas->drawLine(bounds.fLeft - 5, yPos, bounds.fRight + 5, yPos, paint);
524 SkString label;
525 label.printf("%0.3g", value);
526 paint.setColor(0xFF000000);
527 paint.setTextSize(11.0f);
528 paint.setStyle(SkPaint::kFill_Style);
Cary Clark2a475ea2017-04-28 15:35:12 -0400529 canvas->drawString(label, bounds.fLeft + 5, yPos - 5, paint);
caryclark88c748a2015-02-18 10:56:00 -0800530 paint.setTextSize(13.0f);
Cary Clark2a475ea2017-04-28 15:35:12 -0400531 canvas->drawString(name, bounds.fLeft, bounds.bottom() + 11, paint);
caryclark88c748a2015-02-18 10:56:00 -0800532 }
533
534 void setForGeometry() {
535 fDrawRibs = true;
536 fDrawTangents = true;
caryclarkb6474dd2016-01-19 08:07:49 -0800537 fDrawTDivs = false;
caryclark88c748a2015-02-18 10:56:00 -0800538 fWidthScale = 1;
539 }
540
541 void setForText() {
caryclarkb6474dd2016-01-19 08:07:49 -0800542 fDrawRibs = fDrawTangents = fDrawTDivs = false;
caryclark88c748a2015-02-18 10:56:00 -0800543 fWidthScale = 0.002f;
544 }
545
caryclarkb6474dd2016-01-19 08:07:49 -0800546 void setForSingles() {
547 setForGeometry();
548 fDrawTDivs = true;
549 }
550
caryclark88c748a2015-02-18 10:56:00 -0800551 void setAsNeeded() {
caryclarkb6474dd2016-01-19 08:07:49 -0800552 if (fConicButton.fEnabled || fCubicButton.fEnabled || fQuadButton.fEnabled) {
553 setForSingles();
caryclark88651ae2016-01-20 11:55:11 -0800554 } else if (fRRectButton.fEnabled || fCircleButton.fEnabled || fArcButton.fEnabled) {
caryclark88c748a2015-02-18 10:56:00 -0800555 setForGeometry();
556 } else {
557 setForText();
558 }
559 }
560
caryclark88651ae2016-01-20 11:55:11 -0800561 bool arcCenter(SkPoint* center) {
562 SkPath path;
563 path.moveTo(fPts[10]);
564 path.arcTo(fPts[11], fPts[12], fRadius);
565 SkPath::Iter iter(path, false);
566 SkPoint pts[4];
567 iter.next(pts);
568 if (SkPath::kLine_Verb == iter.next(pts)) {
569 iter.next(pts);
570 }
571 SkVector before = pts[0] - pts[1];
572 SkVector after = pts[1] - pts[2];
573 before.setLength(fRadius);
574 after.setLength(fRadius);
575 SkVector beforeCCW, afterCCW;
Cary Clarkdf429f32017-11-08 11:44:31 -0500576 SkPointPriv::RotateCCW(before, &beforeCCW);
577 SkPointPriv::RotateCCW(after, &afterCCW);
caryclark88651ae2016-01-20 11:55:11 -0800578 beforeCCW += pts[0];
579 afterCCW += pts[2];
580 *center = beforeCCW;
581 if (SkScalarNearlyEqual(beforeCCW.fX, afterCCW.fX)
582 && SkScalarNearlyEqual(beforeCCW.fY, afterCCW.fY)) {
583 return true;
584 }
585 SkVector beforeCW, afterCW;
Cary Clarkdf429f32017-11-08 11:44:31 -0500586 SkPointPriv::RotateCW(before, &beforeCW);
587 SkPointPriv::RotateCW(after, &afterCW);
caryclark88651ae2016-01-20 11:55:11 -0800588 beforeCW += pts[0];
589 afterCW += pts[2];
590 *center = beforeCW;
591 return SkScalarNearlyEqual(beforeCW.fX, afterCW.fX)
592 && SkScalarNearlyEqual(beforeCCW.fY, afterCW.fY);
593 }
594
mtklein36352bf2015-03-25 18:17:31 -0700595 void onDrawContent(SkCanvas* canvas) override {
caryclark88c748a2015-02-18 10:56:00 -0800596 SkPath path;
597 SkScalar width = fWidth;
598
599 if (fCubicButton.fEnabled) {
600 path.moveTo(fPts[0]);
601 path.cubicTo(fPts[1], fPts[2], fPts[3]);
caryclarkb6474dd2016-01-19 08:07:49 -0800602 setForSingles();
caryclark04e4d082015-02-20 06:33:57 -0800603 draw_stroke(canvas, path, width, 950, false);
604 }
605
606 if (fConicButton.fEnabled) {
caryclarkb6474dd2016-01-19 08:07:49 -0800607 path.reset();
caryclark04e4d082015-02-20 06:33:57 -0800608 path.moveTo(fPts[4]);
609 path.conicTo(fPts[5], fPts[6], fWeight);
caryclarkb6474dd2016-01-19 08:07:49 -0800610 setForSingles();
caryclark04e4d082015-02-20 06:33:57 -0800611 draw_stroke(canvas, path, width, 950, false);
caryclark88c748a2015-02-18 10:56:00 -0800612 }
613
614 if (fQuadButton.fEnabled) {
615 path.reset();
caryclark04e4d082015-02-20 06:33:57 -0800616 path.moveTo(fPts[7]);
617 path.quadTo(fPts[8], fPts[9]);
caryclarkb6474dd2016-01-19 08:07:49 -0800618 setForSingles();
caryclark04e4d082015-02-20 06:33:57 -0800619 draw_stroke(canvas, path, width, 950, false);
caryclark88c748a2015-02-18 10:56:00 -0800620 }
621
caryclark88651ae2016-01-20 11:55:11 -0800622 if (fArcButton.fEnabled) {
623 path.reset();
624 path.moveTo(fPts[10]);
625 path.arcTo(fPts[11], fPts[12], fRadius);
626 setForGeometry();
627 draw_stroke(canvas, path, width, 950, false);
628 SkPath pathPts;
629 pathPts.moveTo(fPts[10]);
630 pathPts.lineTo(fPts[11]);
631 pathPts.lineTo(fPts[12]);
632 draw_points(canvas, pathPts, SK_ColorDKGRAY, true);
633 }
634
caryclark88c748a2015-02-18 10:56:00 -0800635 if (fRRectButton.fEnabled) {
636 SkScalar rad = 32;
637 SkRect r;
caryclark88651ae2016-01-20 11:55:11 -0800638 r.set(&fPts[13], 2);
caryclark88c748a2015-02-18 10:56:00 -0800639 path.reset();
640 SkRRect rr;
641 rr.setRectXY(r, rad, rad);
642 path.addRRect(rr);
643 setForGeometry();
caryclark04e4d082015-02-20 06:33:57 -0800644 draw_stroke(canvas, path, width, 950, false);
caryclark88c748a2015-02-18 10:56:00 -0800645
646 path.reset();
647 SkRRect rr2;
648 rr.inset(width/2, width/2, &rr2);
649 path.addRRect(rr2, SkPath::kCCW_Direction);
650 rr.inset(-width/2, -width/2, &rr2);
651 path.addRRect(rr2, SkPath::kCW_Direction);
652 SkPaint paint;
653 paint.setAntiAlias(true);
654 paint.setColor(0x40FF8844);
655 canvas->drawPath(path, paint);
656 }
657
caryclark04e4d082015-02-20 06:33:57 -0800658 if (fCircleButton.fEnabled) {
659 path.reset();
660 SkRect r;
caryclark88651ae2016-01-20 11:55:11 -0800661 r.set(&fPts[15], 2);
caryclark04e4d082015-02-20 06:33:57 -0800662 path.addOval(r);
663 setForGeometry();
664 if (fCircleButton.fFill) {
caryclark88651ae2016-01-20 11:55:11 -0800665 if (fArcButton.fEnabled) {
666 SkPoint center;
667 if (arcCenter(&center)) {
halcanary9d524f22016-03-29 09:03:52 -0700668 r.set(center.fX - fRadius, center.fY - fRadius, center.fX + fRadius,
caryclark88651ae2016-01-20 11:55:11 -0800669 center.fY + fRadius);
670 }
671 }
caryclark04e4d082015-02-20 06:33:57 -0800672 draw_fill(canvas, r, width);
673 } else {
674 draw_stroke(canvas, path, width, 950, false);
675 }
676 }
677
caryclark88c748a2015-02-18 10:56:00 -0800678 if (fTextButton.fEnabled) {
679 path.reset();
680 SkPaint paint;
681 paint.setAntiAlias(true);
682 paint.setTextSize(fTextSize);
683 paint.getTextPath(fText.c_str(), fText.size(), 0, fTextSize, &path);
684 setForText();
caryclark04e4d082015-02-20 06:33:57 -0800685 draw_stroke(canvas, path, width * fWidthScale / fTextSize, fTextSize, true);
caryclark88c748a2015-02-18 10:56:00 -0800686 }
687
688 if (fAnimate) {
689 fWidth += fDWidth;
690 if (fDWidth > 0 && fWidth > kWidthMax) {
691 fDWidth = -fDWidth;
692 } else if (fDWidth < 0 && fWidth < kWidthMin) {
693 fDWidth = -fDWidth;
694 }
695 }
696 setAsNeeded();
caryclark04e4d082015-02-20 06:33:57 -0800697 if (fConicButton.fEnabled) {
698 draw_control(canvas, fWeightControl, fWeight, 0, 5, "weight");
699 }
caryclark88651ae2016-01-20 11:55:11 -0800700 if (fArcButton.fEnabled) {
701 draw_control(canvas, fRadiusControl, fRadius, 0, 500, "radius");
702 }
caryclarka76b7a3b2015-05-22 06:26:52 -0700703#ifdef SK_DEBUG
caryclark88c748a2015-02-18 10:56:00 -0800704 draw_control(canvas, fErrorControl, gDebugStrokerError, kStrokerErrorMin, kStrokerErrorMax,
705 "error");
706#endif
707 draw_control(canvas, fWidthControl, fWidth * fWidthScale, kWidthMin * fWidthScale,
708 kWidthMax * fWidthScale, "width");
709 draw_button(canvas, fQuadButton);
710 draw_button(canvas, fCubicButton);
caryclark04e4d082015-02-20 06:33:57 -0800711 draw_button(canvas, fConicButton);
caryclark88651ae2016-01-20 11:55:11 -0800712 draw_button(canvas, fArcButton);
caryclark88c748a2015-02-18 10:56:00 -0800713 draw_button(canvas, fRRectButton);
caryclark04e4d082015-02-20 06:33:57 -0800714 draw_button(canvas, fCircleButton);
caryclark88c748a2015-02-18 10:56:00 -0800715 draw_button(canvas, fTextButton);
caryclark88c748a2015-02-18 10:56:00 -0800716 }
717
718 class MyClick : public Click {
719 public:
720 int fIndex;
721 MyClick(SkView* target, int index) : Click(target), fIndex(index) {}
722 };
723
724 virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y,
mtklein36352bf2015-03-25 18:17:31 -0700725 unsigned modi) override {
caryclark88c748a2015-02-18 10:56:00 -0800726 for (size_t i = 0; i < SK_ARRAY_COUNT(fPts); ++i) {
727 if (hittest(fPts[i], x, y)) {
728 return new MyClick(this, (int)i);
729 }
730 }
731 const SkRect& rectPt = SkRect::MakeXYWH(x, y, 1, 1);
caryclark04e4d082015-02-20 06:33:57 -0800732 if (fWeightControl.contains(rectPt)) {
733 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 1);
734 }
caryclark88651ae2016-01-20 11:55:11 -0800735 if (fRadiusControl.contains(rectPt)) {
736 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 2);
737 }
caryclarka76b7a3b2015-05-22 06:26:52 -0700738#ifdef SK_DEBUG
caryclark88c748a2015-02-18 10:56:00 -0800739 if (fErrorControl.contains(rectPt)) {
caryclark88651ae2016-01-20 11:55:11 -0800740 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 3);
caryclark88c748a2015-02-18 10:56:00 -0800741 }
742#endif
743 if (fWidthControl.contains(rectPt)) {
caryclark88651ae2016-01-20 11:55:11 -0800744 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 4);
caryclark88c748a2015-02-18 10:56:00 -0800745 }
746 if (fCubicButton.fBounds.contains(rectPt)) {
747 fCubicButton.fEnabled ^= true;
caryclark88651ae2016-01-20 11:55:11 -0800748 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 5);
caryclark88c748a2015-02-18 10:56:00 -0800749 }
caryclark04e4d082015-02-20 06:33:57 -0800750 if (fConicButton.fBounds.contains(rectPt)) {
751 fConicButton.fEnabled ^= true;
caryclark88651ae2016-01-20 11:55:11 -0800752 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 6);
caryclark04e4d082015-02-20 06:33:57 -0800753 }
caryclark88c748a2015-02-18 10:56:00 -0800754 if (fQuadButton.fBounds.contains(rectPt)) {
755 fQuadButton.fEnabled ^= true;
caryclark88651ae2016-01-20 11:55:11 -0800756 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 7);
757 }
758 if (fArcButton.fBounds.contains(rectPt)) {
759 fArcButton.fEnabled ^= true;
760 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 8);
caryclark88c748a2015-02-18 10:56:00 -0800761 }
762 if (fRRectButton.fBounds.contains(rectPt)) {
763 fRRectButton.fEnabled ^= true;
caryclark88651ae2016-01-20 11:55:11 -0800764 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 9);
caryclark04e4d082015-02-20 06:33:57 -0800765 }
766 if (fCircleButton.fBounds.contains(rectPt)) {
767 bool wasEnabled = fCircleButton.fEnabled;
768 fCircleButton.fEnabled = !fCircleButton.fFill;
769 fCircleButton.fFill = wasEnabled && !fCircleButton.fFill;
caryclark88651ae2016-01-20 11:55:11 -0800770 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 10);
caryclark88c748a2015-02-18 10:56:00 -0800771 }
772 if (fTextButton.fBounds.contains(rectPt)) {
773 fTextButton.fEnabled ^= true;
caryclark88651ae2016-01-20 11:55:11 -0800774 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 11);
caryclark88c748a2015-02-18 10:56:00 -0800775 }
776 return this->INHERITED::onFindClickHandler(x, y, modi);
777 }
778
779 static SkScalar MapScreenYtoValue(int y, const SkRect& control, SkScalar min,
780 SkScalar max) {
781 return (SkIntToScalar(y) - control.fTop) / control.height() * (max - min) + min;
782 }
783
mtklein36352bf2015-03-25 18:17:31 -0700784 bool onClick(Click* click) override {
caryclark88c748a2015-02-18 10:56:00 -0800785 int index = ((MyClick*)click)->fIndex;
786 if (index < (int) SK_ARRAY_COUNT(fPts)) {
787 fPts[index].offset(SkIntToScalar(click->fICurr.fX - click->fIPrev.fX),
788 SkIntToScalar(click->fICurr.fY - click->fIPrev.fY));
caryclark04e4d082015-02-20 06:33:57 -0800789 } else if (index == (int) SK_ARRAY_COUNT(fPts) + 1) {
790 fWeight = MapScreenYtoValue(click->fICurr.fY, fWeightControl, 0, 5);
caryclark88651ae2016-01-20 11:55:11 -0800791 } else if (index == (int) SK_ARRAY_COUNT(fPts) + 2) {
792 fRadius = MapScreenYtoValue(click->fICurr.fY, fRadiusControl, 0, 500);
caryclark88c748a2015-02-18 10:56:00 -0800793 }
caryclarka76b7a3b2015-05-22 06:26:52 -0700794#ifdef SK_DEBUG
caryclark88651ae2016-01-20 11:55:11 -0800795 else if (index == (int) SK_ARRAY_COUNT(fPts) + 3) {
caryclark88c748a2015-02-18 10:56:00 -0800796 gDebugStrokerError = SkTMax(FLT_EPSILON, MapScreenYtoValue(click->fICurr.fY,
797 fErrorControl, kStrokerErrorMin, kStrokerErrorMax));
798 gDebugStrokerErrorSet = true;
799 }
800#endif
caryclark88651ae2016-01-20 11:55:11 -0800801 else if (index == (int) SK_ARRAY_COUNT(fPts) + 4) {
caryclark88c748a2015-02-18 10:56:00 -0800802 fWidth = SkTMax(FLT_EPSILON, MapScreenYtoValue(click->fICurr.fY, fWidthControl,
803 kWidthMin, kWidthMax));
804 fAnimate = fWidth <= kWidthMin;
805 }
806 return true;
807 }
808
809private:
810 typedef SkView INHERITED;
811};
812
813///////////////////////////////////////////////////////////////////////////////
814
815static SkView* F2() { return new QuadStrokerView; }
816static SkViewRegister gR2(F2);