blob: fa45d6eb0dba63c87abe8018d873f4f26c616987 [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
8#include "sk_tool_utils.h"
9#include "SampleCode.h"
10#include "SkView.h"
11#include "SkCanvas.h"
caryclarkb6474dd2016-01-19 08:07:49 -080012#include "SkGeometry.h"
caryclark88c748a2015-02-18 10:56:00 -080013#include "SkPathMeasure.h"
Cary Clarkdf429f32017-11-08 11:44:31 -050014#include "SkPointPriv.h"
caryclark88c748a2015-02-18 10:56:00 -080015#include "SkRandom.h"
16#include "SkRRect.h"
17#include "SkColorPriv.h"
18#include "SkStrokerPriv.h"
19#include "SkSurface.h"
20
21static bool hittest(const SkPoint& target, SkScalar x, SkScalar y) {
22 const SkScalar TOL = 7;
23 return SkPoint::Distance(target, SkPoint::Make(x, y)) <= TOL;
24}
25
26static int getOnCurvePoints(const SkPath& path, SkPoint storage[]) {
27 SkPath::RawIter iter(path);
28 SkPoint pts[4];
29 SkPath::Verb verb;
30
31 int count = 0;
32 while ((verb = iter.next(pts)) != SkPath::kDone_Verb) {
33 switch (verb) {
34 case SkPath::kMove_Verb:
35 case SkPath::kLine_Verb:
36 case SkPath::kQuad_Verb:
37 case SkPath::kConic_Verb:
38 case SkPath::kCubic_Verb:
39 storage[count++] = pts[0];
40 break;
41 default:
42 break;
43 }
44 }
45 return count;
46}
47
48static void getContourCounts(const SkPath& path, SkTArray<int>* contourCounts) {
49 SkPath::RawIter iter(path);
50 SkPoint pts[4];
51 SkPath::Verb verb;
52
53 int count = 0;
54 while ((verb = iter.next(pts)) != SkPath::kDone_Verb) {
55 switch (verb) {
56 case SkPath::kMove_Verb:
57 case SkPath::kLine_Verb:
58 count += 1;
59 break;
60 case SkPath::kQuad_Verb:
61 case SkPath::kConic_Verb:
62 count += 2;
63 break;
64 case SkPath::kCubic_Verb:
65 count += 3;
66 break;
67 case SkPath::kClose_Verb:
68 contourCounts->push_back(count);
69 count = 0;
70 break;
71 default:
72 break;
73 }
74 }
75 if (count > 0) {
76 contourCounts->push_back(count);
77 }
78}
79
reede8f30622016-03-23 18:59:25 -070080static void erase(const sk_sp<SkSurface>& surface) {
caryclark88651ae2016-01-20 11:55:11 -080081 SkCanvas* canvas = surface->getCanvas();
82 if (canvas) {
83 canvas->clear(SK_ColorTRANSPARENT);
84 }
caryclark88c748a2015-02-18 10:56:00 -080085}
86
87struct StrokeTypeButton {
88 SkRect fBounds;
89 char fLabel;
90 bool fEnabled;
91};
92
caryclark04e4d082015-02-20 06:33:57 -080093struct CircleTypeButton : public StrokeTypeButton {
94 bool fFill;
95};
96
caryclark88c748a2015-02-18 10:56:00 -080097class QuadStrokerView : public SampleView {
98 enum {
99 SKELETON_COLOR = 0xFF0000FF,
100 WIREFRAME_COLOR = 0x80FF0000
101 };
102
103 enum {
caryclark88651ae2016-01-20 11:55:11 -0800104 kCount = 18
caryclark88c748a2015-02-18 10:56:00 -0800105 };
106 SkPoint fPts[kCount];
caryclark04e4d082015-02-20 06:33:57 -0800107 SkRect fWeightControl;
caryclark88651ae2016-01-20 11:55:11 -0800108 SkRect fRadiusControl;
caryclark88c748a2015-02-18 10:56:00 -0800109 SkRect fErrorControl;
110 SkRect fWidthControl;
111 SkRect fBounds;
112 SkMatrix fMatrix, fInverse;
reed8a21c9f2016-03-08 18:50:00 -0800113 sk_sp<SkShader> fShader;
reede8f30622016-03-23 18:59:25 -0700114 sk_sp<SkSurface> fMinSurface;
115 sk_sp<SkSurface> fMaxSurface;
caryclark88c748a2015-02-18 10:56:00 -0800116 StrokeTypeButton fCubicButton;
caryclark04e4d082015-02-20 06:33:57 -0800117 StrokeTypeButton fConicButton;
caryclark88c748a2015-02-18 10:56:00 -0800118 StrokeTypeButton fQuadButton;
caryclark88651ae2016-01-20 11:55:11 -0800119 StrokeTypeButton fArcButton;
caryclark88c748a2015-02-18 10:56:00 -0800120 StrokeTypeButton fRRectButton;
caryclark04e4d082015-02-20 06:33:57 -0800121 CircleTypeButton fCircleButton;
caryclark88c748a2015-02-18 10:56:00 -0800122 StrokeTypeButton fTextButton;
123 SkString fText;
124 SkScalar fTextSize;
caryclark04e4d082015-02-20 06:33:57 -0800125 SkScalar fWeight;
caryclark88651ae2016-01-20 11:55:11 -0800126 SkScalar fRadius;
caryclark88c748a2015-02-18 10:56:00 -0800127 SkScalar fWidth, fDWidth;
128 SkScalar fWidthScale;
129 int fW, fH, fZoom;
130 bool fAnimate;
131 bool fDrawRibs;
132 bool fDrawTangents;
caryclarkb6474dd2016-01-19 08:07:49 -0800133 bool fDrawTDivs;
caryclarka76b7a3b2015-05-22 06:26:52 -0700134#ifdef SK_DEBUG
caryclark88c748a2015-02-18 10:56:00 -0800135 #define kStrokerErrorMin 0.001f
136 #define kStrokerErrorMax 5
137#endif
138 #define kWidthMin 1
139 #define kWidthMax 100
140public:
141 QuadStrokerView() {
142 this->setBGColor(SK_ColorLTGRAY);
143
caryclark04e4d082015-02-20 06:33:57 -0800144 fPts[0].set(50, 200); // cubic
caryclark88c748a2015-02-18 10:56:00 -0800145 fPts[1].set(50, 100);
146 fPts[2].set(150, 50);
147 fPts[3].set(300, 50);
148
caryclark04e4d082015-02-20 06:33:57 -0800149 fPts[4].set(350, 200); // conic
caryclark88c748a2015-02-18 10:56:00 -0800150 fPts[5].set(350, 100);
151 fPts[6].set(450, 50);
152
caryclark04e4d082015-02-20 06:33:57 -0800153 fPts[7].set(150, 300); // quad
154 fPts[8].set(150, 200);
155 fPts[9].set(250, 150);
caryclark88c748a2015-02-18 10:56:00 -0800156
caryclark88651ae2016-01-20 11:55:11 -0800157 fPts[10].set(250, 200); // arc
158 fPts[11].set(250, 300);
159 fPts[12].set(150, 350);
caryclark04e4d082015-02-20 06:33:57 -0800160
caryclark88651ae2016-01-20 11:55:11 -0800161 fPts[13].set(200, 200); // rrect
162 fPts[14].set(400, 400);
163
164 fPts[15].set(250, 250); // oval
165 fPts[16].set(450, 450);
caryclark04e4d082015-02-20 06:33:57 -0800166
caryclark88c748a2015-02-18 10:56:00 -0800167 fText = "a";
168 fTextSize = 12;
169 fWidth = 50;
170 fDWidth = 0.25f;
caryclark04e4d082015-02-20 06:33:57 -0800171 fWeight = 1;
caryclark88651ae2016-01-20 11:55:11 -0800172 fRadius = 150;
caryclark88c748a2015-02-18 10:56:00 -0800173
174 fCubicButton.fLabel = 'C';
175 fCubicButton.fEnabled = false;
caryclark04e4d082015-02-20 06:33:57 -0800176 fConicButton.fLabel = 'K';
caryclark88651ae2016-01-20 11:55:11 -0800177 fConicButton.fEnabled = false;
caryclark88c748a2015-02-18 10:56:00 -0800178 fQuadButton.fLabel = 'Q';
179 fQuadButton.fEnabled = false;
caryclark88651ae2016-01-20 11:55:11 -0800180 fArcButton.fLabel = 'A';
181 fArcButton.fEnabled = true;
caryclark88c748a2015-02-18 10:56:00 -0800182 fRRectButton.fLabel = 'R';
183 fRRectButton.fEnabled = false;
caryclark04e4d082015-02-20 06:33:57 -0800184 fCircleButton.fLabel = 'O';
caryclark88651ae2016-01-20 11:55:11 -0800185 fCircleButton.fEnabled = true;
186 fCircleButton.fFill = true;
caryclark88c748a2015-02-18 10:56:00 -0800187 fTextButton.fLabel = 'T';
caryclark04e4d082015-02-20 06:33:57 -0800188 fTextButton.fEnabled = false;
caryclark88651ae2016-01-20 11:55:11 -0800189 fAnimate = false;
caryclark88c748a2015-02-18 10:56:00 -0800190 setAsNeeded();
191 }
192
193protected:
mtklein36352bf2015-03-25 18:17:31 -0700194 bool onQuery(SkEvent* evt) override {
caryclark88c748a2015-02-18 10:56:00 -0800195 if (SampleCode::TitleQ(*evt)) {
196 SampleCode::TitleR(evt, "QuadStroker");
197 return true;
198 }
199 SkUnichar uni;
200 if (fTextButton.fEnabled && SampleCode::CharQ(*evt, &uni)) {
201 switch (uni) {
202 case ' ':
203 fText = "";
204 break;
205 case '-':
206 fTextSize = SkTMax(1.0f, fTextSize - 1);
207 break;
208 case '+':
209 case '=':
210 fTextSize += 1;
211 break;
212 default:
213 fText.appendUnichar(uni);
214 }
halcanary96fcdcc2015-08-27 07:41:13 -0700215 this->inval(nullptr);
caryclark88c748a2015-02-18 10:56:00 -0800216 return true;
217 }
218 return this->INHERITED::onQuery(evt);
219 }
220
mtklein36352bf2015-03-25 18:17:31 -0700221 void onSizeChange() override {
caryclark88651ae2016-01-20 11:55:11 -0800222 fRadiusControl.setXYWH(this->width() - 200, 30, 30, 400);
caryclark04e4d082015-02-20 06:33:57 -0800223 fWeightControl.setXYWH(this->width() - 150, 30, 30, 400);
caryclark88c748a2015-02-18 10:56:00 -0800224 fErrorControl.setXYWH(this->width() - 100, 30, 30, 400);
225 fWidthControl.setXYWH(this->width() - 50, 30, 30, 400);
caryclark04e4d082015-02-20 06:33:57 -0800226 int buttonOffset = 450;
227 fCubicButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
228 buttonOffset += 50;
229 fConicButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
230 buttonOffset += 50;
231 fQuadButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
232 buttonOffset += 50;
caryclark88651ae2016-01-20 11:55:11 -0800233 fArcButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
234 buttonOffset += 50;
caryclark04e4d082015-02-20 06:33:57 -0800235 fRRectButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
236 buttonOffset += 50;
237 fCircleButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
238 buttonOffset += 50;
239 fTextButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
caryclark88c748a2015-02-18 10:56:00 -0800240 this->INHERITED::onSizeChange();
241 }
242
243 void copyMinToMax() {
244 erase(fMaxSurface);
245 SkCanvas* canvas = fMaxSurface->getCanvas();
246 canvas->save();
247 canvas->concat(fMatrix);
halcanary96fcdcc2015-08-27 07:41:13 -0700248 fMinSurface->draw(canvas, 0, 0, nullptr);
caryclark88c748a2015-02-18 10:56:00 -0800249 canvas->restore();
250
251 SkPaint paint;
reed374772b2016-10-05 17:33:02 -0700252 paint.setBlendMode(SkBlendMode::kClear);
caryclark88c748a2015-02-18 10:56:00 -0800253 for (int iy = 1; iy < fH; ++iy) {
254 SkScalar y = SkIntToScalar(iy * fZoom);
255 canvas->drawLine(0, y - SK_ScalarHalf, 999, y - SK_ScalarHalf, paint);
256 }
257 for (int ix = 1; ix < fW; ++ix) {
258 SkScalar x = SkIntToScalar(ix * fZoom);
259 canvas->drawLine(x - SK_ScalarHalf, 0, x - SK_ScalarHalf, 999, paint);
260 }
261 }
262
263 void setWHZ(int width, int height, int zoom) {
264 fZoom = zoom;
265 fBounds.set(0, 0, SkIntToScalar(width * zoom), SkIntToScalar(height * zoom));
266 fMatrix.setScale(SkIntToScalar(zoom), SkIntToScalar(zoom));
267 fInverse.setScale(SK_Scalar1 / zoom, SK_Scalar1 / zoom);
reed8a21c9f2016-03-08 18:50:00 -0800268 fShader = sk_tool_utils::create_checkerboard_shader(0xFFCCCCCC, 0xFFFFFFFF, zoom);
caryclark88c748a2015-02-18 10:56:00 -0800269
270 SkImageInfo info = SkImageInfo::MakeN32Premul(width, height);
reede8f30622016-03-23 18:59:25 -0700271 fMinSurface = SkSurface::MakeRaster(info);
caryclark88c748a2015-02-18 10:56:00 -0800272 info = info.makeWH(width * zoom, height * zoom);
reede8f30622016-03-23 18:59:25 -0700273 fMaxSurface = SkSurface::MakeRaster(info);
caryclark88c748a2015-02-18 10:56:00 -0800274 }
275
276 void draw_points(SkCanvas* canvas, const SkPath& path, SkColor color,
277 bool show_lines) {
278 SkPaint paint;
279 paint.setColor(color);
280 paint.setAlpha(0x80);
281 paint.setAntiAlias(true);
282 int n = path.countPoints();
283 SkAutoSTArray<32, SkPoint> pts(n);
284 if (show_lines && fDrawTangents) {
285 SkTArray<int> contourCounts;
286 getContourCounts(path, &contourCounts);
287 SkPoint* ptPtr = pts.get();
288 for (int i = 0; i < contourCounts.count(); ++i) {
289 int count = contourCounts[i];
290 path.getPoints(ptPtr, count);
291 canvas->drawPoints(SkCanvas::kPolygon_PointMode, count, ptPtr, paint);
292 ptPtr += count;
293 }
294 } else {
295 n = getOnCurvePoints(path, pts.get());
296 }
297 paint.setStrokeWidth(5);
298 canvas->drawPoints(SkCanvas::kPoints_PointMode, n, pts.get(), paint);
299 }
300
301 void draw_ribs(SkCanvas* canvas, const SkPath& path, SkScalar width,
302 SkColor color) {
303 const SkScalar radius = width / 2;
304
305 SkPathMeasure meas(path, false);
306 SkScalar total = meas.getLength();
307
308 SkScalar delta = 8;
caryclarkb6474dd2016-01-19 08:07:49 -0800309 SkPaint paint, labelP;
caryclark88c748a2015-02-18 10:56:00 -0800310 paint.setColor(color);
caryclarkb6474dd2016-01-19 08:07:49 -0800311 labelP.setColor(color & 0xff5f9f5f);
caryclark88c748a2015-02-18 10:56:00 -0800312 SkPoint pos, tan;
caryclarkb6474dd2016-01-19 08:07:49 -0800313 int index = 0;
caryclark88c748a2015-02-18 10:56:00 -0800314 for (SkScalar dist = 0; dist <= total; dist += delta) {
315 if (meas.getPosTan(dist, &pos, &tan)) {
316 tan.scale(radius);
Cary Clarkdf429f32017-11-08 11:44:31 -0500317 SkPointPriv::RotateCCW(&tan);
caryclark88c748a2015-02-18 10:56:00 -0800318 canvas->drawLine(pos.x() + tan.x(), pos.y() + tan.y(),
319 pos.x() - tan.x(), pos.y() - tan.y(), paint);
caryclarkb6474dd2016-01-19 08:07:49 -0800320 if (0 == index % 10) {
321 SkString label;
322 label.appendS32(index);
323 SkRect dot = SkRect::MakeXYWH(pos.x() - 2, pos.y() - 2, 4, 4);
324 canvas->drawRect(dot, labelP);
Cary Clark2a475ea2017-04-28 15:35:12 -0400325 canvas->drawString(label,
caryclarkb6474dd2016-01-19 08:07:49 -0800326 pos.x() - tan.x() * 1.25f, pos.y() - tan.y() * 1.25f, labelP);
327 }
328 }
329 ++index;
330 }
331 }
332
333 void draw_t_divs(SkCanvas* canvas, const SkPath& path, SkScalar width, SkColor color) {
334 const SkScalar radius = width / 2;
335 SkPaint paint;
336 paint.setColor(color);
337 SkPathMeasure meas(path, false);
338 SkScalar total = meas.getLength();
339 SkScalar delta = 8;
340 int ribs = 0;
341 for (SkScalar dist = 0; dist <= total; dist += delta) {
342 ++ribs;
343 }
344 SkPath::RawIter iter(path);
345 SkPoint pts[4];
346 if (SkPath::kMove_Verb != iter.next(pts)) {
347 SkASSERT(0);
348 return;
349 }
350 SkPath::Verb verb = iter.next(pts);
351 SkASSERT(SkPath::kLine_Verb <= verb && verb <= SkPath::kCubic_Verb);
352 SkPoint pos, tan;
353 for (int index = 0; index < ribs; ++index) {
354 SkScalar t = (SkScalar) index / ribs;
355 switch (verb) {
356 case SkPath::kLine_Verb:
357 tan = pts[1] - pts[0];
358 pos = pts[0];
359 pos.fX += tan.fX * t;
360 pos.fY += tan.fY * t;
361 break;
362 case SkPath::kQuad_Verb:
363 pos = SkEvalQuadAt(pts, t);
364 tan = SkEvalQuadTangentAt(pts, t);
365 break;
366 case SkPath::kConic_Verb: {
367 SkConic conic(pts, iter.conicWeight());
368 pos = conic.evalAt(t);
369 tan = conic.evalTangentAt(t);
370 } break;
371 case SkPath::kCubic_Verb:
372 SkEvalCubicAt(pts, t, &pos, &tan, nullptr);
373 break;
374 default:
375 SkASSERT(0);
376 return;
377 }
378 tan.setLength(radius);
Cary Clarkdf429f32017-11-08 11:44:31 -0500379 SkPointPriv::RotateCCW(&tan);
caryclarkb6474dd2016-01-19 08:07:49 -0800380 canvas->drawLine(pos.x() + tan.x(), pos.y() + tan.y(),
381 pos.x() - tan.x(), pos.y() - tan.y(), paint);
382 if (0 == index % 10) {
383 SkString label;
384 label.appendS32(index);
Cary Clark2a475ea2017-04-28 15:35:12 -0400385 canvas->drawString(label,
caryclarkb6474dd2016-01-19 08:07:49 -0800386 pos.x() + tan.x() * 1.25f, pos.y() + tan.y() * 1.25f, paint);
caryclark88c748a2015-02-18 10:56:00 -0800387 }
388 }
389 }
390
caryclark04e4d082015-02-20 06:33:57 -0800391 void draw_stroke(SkCanvas* canvas, const SkPath& path, SkScalar width, SkScalar scale,
392 bool drawText) {
caryclark612f70d2015-05-19 11:05:37 -0700393 if (path.isEmpty()) {
caryclark88c748a2015-02-18 10:56:00 -0800394 return;
395 }
caryclark612f70d2015-05-19 11:05:37 -0700396 SkRect bounds = path.getBounds();
halcanary9d524f22016-03-29 09:03:52 -0700397 this->setWHZ(SkScalarCeilToInt(bounds.right()), drawText
caryclark04e4d082015-02-20 06:33:57 -0800398 ? SkScalarRoundToInt(scale * 3 / 2) : SkScalarRoundToInt(scale),
399 SkScalarRoundToInt(950.0f / scale));
caryclark88c748a2015-02-18 10:56:00 -0800400 erase(fMinSurface);
401 SkPaint paint;
402 paint.setColor(0x1f1f0f0f);
caryclark88c748a2015-02-18 10:56:00 -0800403 paint.setStyle(SkPaint::kStroke_Style);
caryclark04e4d082015-02-20 06:33:57 -0800404 paint.setStrokeWidth(width * scale * scale);
caryclark88c748a2015-02-18 10:56:00 -0800405 paint.setColor(0x3f0f1f3f);
caryclark04e4d082015-02-20 06:33:57 -0800406 if (drawText) {
407 fMinSurface->getCanvas()->drawPath(path, paint);
408 this->copyMinToMax();
halcanary96fcdcc2015-08-27 07:41:13 -0700409 fMaxSurface->draw(canvas, 0, 0, nullptr);
caryclark04e4d082015-02-20 06:33:57 -0800410 }
caryclark88c748a2015-02-18 10:56:00 -0800411 paint.setAntiAlias(true);
412 paint.setStyle(SkPaint::kStroke_Style);
413 paint.setStrokeWidth(1);
414
415 paint.setColor(SKELETON_COLOR);
416 SkPath scaled;
417 SkMatrix matrix;
418 matrix.reset();
caryclark04e4d082015-02-20 06:33:57 -0800419 matrix.setScale(950 / scale, 950 / scale);
caryclark88c748a2015-02-18 10:56:00 -0800420 if (drawText) {
421 path.transform(matrix, &scaled);
422 } else {
423 scaled = path;
424 }
425 canvas->drawPath(scaled, paint);
426 draw_points(canvas, scaled, SKELETON_COLOR, true);
427
428 if (fDrawRibs) {
429 draw_ribs(canvas, scaled, width, 0xFF00FF00);
430 }
431
caryclarkb6474dd2016-01-19 08:07:49 -0800432 if (fDrawTDivs) {
433 draw_t_divs(canvas, scaled, width, 0xFF3F3F00);
434 }
435
caryclark88c748a2015-02-18 10:56:00 -0800436 SkPath fill;
437
438 SkPaint p;
439 p.setStyle(SkPaint::kStroke_Style);
caryclark04e4d082015-02-20 06:33:57 -0800440 if (drawText) {
441 p.setStrokeWidth(width * scale * scale);
442 } else {
443 p.setStrokeWidth(width);
444 }
caryclark88c748a2015-02-18 10:56:00 -0800445 p.getFillPath(path, &fill);
446 SkPath scaledFill;
447 if (drawText) {
448 fill.transform(matrix, &scaledFill);
449 } else {
450 scaledFill = fill;
451 }
452 paint.setColor(WIREFRAME_COLOR);
453 canvas->drawPath(scaledFill, paint);
454 draw_points(canvas, scaledFill, WIREFRAME_COLOR, false);
455 }
456
caryclark04e4d082015-02-20 06:33:57 -0800457 void draw_fill(SkCanvas* canvas, const SkRect& rect, SkScalar width) {
458 if (rect.isEmpty()) {
459 return;
460 }
461 SkPaint paint;
462 paint.setColor(0x1f1f0f0f);
463 paint.setStyle(SkPaint::kStroke_Style);
464 paint.setStrokeWidth(width);
465 SkPath path;
466 SkScalar maxSide = SkTMax(rect.width(), rect.height()) / 2;
467 SkPoint center = { rect.fLeft + maxSide, rect.fTop + maxSide };
468 path.addCircle(center.fX, center.fY, maxSide);
469 canvas->drawPath(path, paint);
470 paint.setStyle(SkPaint::kFill_Style);
471 path.reset();
472 path.addCircle(center.fX, center.fY, maxSide - width / 2);
473 paint.setColor(0x3f0f1f3f);
474 canvas->drawPath(path, paint);
475 path.reset();
476 path.setFillType(SkPath::kEvenOdd_FillType);
477 path.addCircle(center.fX, center.fY, maxSide + width / 2);
halcanary9d524f22016-03-29 09:03:52 -0700478 SkRect outside = SkRect::MakeXYWH(center.fX - maxSide - width, center.fY - maxSide - width,
caryclark04e4d082015-02-20 06:33:57 -0800479 (maxSide + width) * 2, (maxSide + width) * 2);
480 path.addRect(outside);
481 canvas->drawPath(path, paint);
482 }
483
caryclark88c748a2015-02-18 10:56:00 -0800484 void draw_button(SkCanvas* canvas, const StrokeTypeButton& button) {
485 SkPaint paint;
486 paint.setAntiAlias(true);
487 paint.setStyle(SkPaint::kStroke_Style);
488 paint.setColor(button.fEnabled ? 0xFF3F0000 : 0x6F3F0000);
489 canvas->drawRect(button.fBounds, paint);
490 paint.setTextSize(25.0f);
491 paint.setColor(button.fEnabled ? 0xFF3F0000 : 0x6F3F0000);
492 paint.setTextAlign(SkPaint::kCenter_Align);
493 paint.setStyle(SkPaint::kFill_Style);
494 canvas->drawText(&button.fLabel, 1, button.fBounds.centerX(), button.fBounds.fBottom - 5,
495 paint);
496 }
497
498 void draw_control(SkCanvas* canvas, const SkRect& bounds, SkScalar value,
499 SkScalar min, SkScalar max, const char* name) {
500 SkPaint paint;
501 paint.setAntiAlias(true);
502 paint.setStyle(SkPaint::kStroke_Style);
503 canvas->drawRect(bounds, paint);
504 SkScalar scale = max - min;
505 SkScalar yPos = bounds.fTop + (value - min) * bounds.height() / scale;
506 paint.setColor(0xFFFF0000);
507 canvas->drawLine(bounds.fLeft - 5, yPos, bounds.fRight + 5, yPos, paint);
508 SkString label;
509 label.printf("%0.3g", value);
510 paint.setColor(0xFF000000);
511 paint.setTextSize(11.0f);
512 paint.setStyle(SkPaint::kFill_Style);
Cary Clark2a475ea2017-04-28 15:35:12 -0400513 canvas->drawString(label, bounds.fLeft + 5, yPos - 5, paint);
caryclark88c748a2015-02-18 10:56:00 -0800514 paint.setTextSize(13.0f);
Cary Clark2a475ea2017-04-28 15:35:12 -0400515 canvas->drawString(name, bounds.fLeft, bounds.bottom() + 11, paint);
caryclark88c748a2015-02-18 10:56:00 -0800516 }
517
518 void setForGeometry() {
519 fDrawRibs = true;
520 fDrawTangents = true;
caryclarkb6474dd2016-01-19 08:07:49 -0800521 fDrawTDivs = false;
caryclark88c748a2015-02-18 10:56:00 -0800522 fWidthScale = 1;
523 }
524
525 void setForText() {
caryclarkb6474dd2016-01-19 08:07:49 -0800526 fDrawRibs = fDrawTangents = fDrawTDivs = false;
caryclark88c748a2015-02-18 10:56:00 -0800527 fWidthScale = 0.002f;
528 }
529
caryclarkb6474dd2016-01-19 08:07:49 -0800530 void setForSingles() {
531 setForGeometry();
532 fDrawTDivs = true;
533 }
534
caryclark88c748a2015-02-18 10:56:00 -0800535 void setAsNeeded() {
caryclarkb6474dd2016-01-19 08:07:49 -0800536 if (fConicButton.fEnabled || fCubicButton.fEnabled || fQuadButton.fEnabled) {
537 setForSingles();
caryclark88651ae2016-01-20 11:55:11 -0800538 } else if (fRRectButton.fEnabled || fCircleButton.fEnabled || fArcButton.fEnabled) {
caryclark88c748a2015-02-18 10:56:00 -0800539 setForGeometry();
540 } else {
541 setForText();
542 }
543 }
544
caryclark88651ae2016-01-20 11:55:11 -0800545 bool arcCenter(SkPoint* center) {
546 SkPath path;
547 path.moveTo(fPts[10]);
548 path.arcTo(fPts[11], fPts[12], fRadius);
549 SkPath::Iter iter(path, false);
550 SkPoint pts[4];
551 iter.next(pts);
552 if (SkPath::kLine_Verb == iter.next(pts)) {
553 iter.next(pts);
554 }
555 SkVector before = pts[0] - pts[1];
556 SkVector after = pts[1] - pts[2];
557 before.setLength(fRadius);
558 after.setLength(fRadius);
559 SkVector beforeCCW, afterCCW;
Cary Clarkdf429f32017-11-08 11:44:31 -0500560 SkPointPriv::RotateCCW(before, &beforeCCW);
561 SkPointPriv::RotateCCW(after, &afterCCW);
caryclark88651ae2016-01-20 11:55:11 -0800562 beforeCCW += pts[0];
563 afterCCW += pts[2];
564 *center = beforeCCW;
565 if (SkScalarNearlyEqual(beforeCCW.fX, afterCCW.fX)
566 && SkScalarNearlyEqual(beforeCCW.fY, afterCCW.fY)) {
567 return true;
568 }
569 SkVector beforeCW, afterCW;
Cary Clarkdf429f32017-11-08 11:44:31 -0500570 SkPointPriv::RotateCW(before, &beforeCW);
571 SkPointPriv::RotateCW(after, &afterCW);
caryclark88651ae2016-01-20 11:55:11 -0800572 beforeCW += pts[0];
573 afterCW += pts[2];
574 *center = beforeCW;
575 return SkScalarNearlyEqual(beforeCW.fX, afterCW.fX)
576 && SkScalarNearlyEqual(beforeCCW.fY, afterCW.fY);
577 }
578
mtklein36352bf2015-03-25 18:17:31 -0700579 void onDrawContent(SkCanvas* canvas) override {
caryclark88c748a2015-02-18 10:56:00 -0800580 SkPath path;
581 SkScalar width = fWidth;
582
583 if (fCubicButton.fEnabled) {
584 path.moveTo(fPts[0]);
585 path.cubicTo(fPts[1], fPts[2], fPts[3]);
caryclarkb6474dd2016-01-19 08:07:49 -0800586 setForSingles();
caryclark04e4d082015-02-20 06:33:57 -0800587 draw_stroke(canvas, path, width, 950, false);
588 }
589
590 if (fConicButton.fEnabled) {
caryclarkb6474dd2016-01-19 08:07:49 -0800591 path.reset();
caryclark04e4d082015-02-20 06:33:57 -0800592 path.moveTo(fPts[4]);
593 path.conicTo(fPts[5], fPts[6], fWeight);
caryclarkb6474dd2016-01-19 08:07:49 -0800594 setForSingles();
caryclark04e4d082015-02-20 06:33:57 -0800595 draw_stroke(canvas, path, width, 950, false);
caryclark88c748a2015-02-18 10:56:00 -0800596 }
597
598 if (fQuadButton.fEnabled) {
599 path.reset();
caryclark04e4d082015-02-20 06:33:57 -0800600 path.moveTo(fPts[7]);
601 path.quadTo(fPts[8], fPts[9]);
caryclarkb6474dd2016-01-19 08:07:49 -0800602 setForSingles();
caryclark04e4d082015-02-20 06:33:57 -0800603 draw_stroke(canvas, path, width, 950, false);
caryclark88c748a2015-02-18 10:56:00 -0800604 }
605
caryclark88651ae2016-01-20 11:55:11 -0800606 if (fArcButton.fEnabled) {
607 path.reset();
608 path.moveTo(fPts[10]);
609 path.arcTo(fPts[11], fPts[12], fRadius);
610 setForGeometry();
611 draw_stroke(canvas, path, width, 950, false);
612 SkPath pathPts;
613 pathPts.moveTo(fPts[10]);
614 pathPts.lineTo(fPts[11]);
615 pathPts.lineTo(fPts[12]);
616 draw_points(canvas, pathPts, SK_ColorDKGRAY, true);
617 }
618
caryclark88c748a2015-02-18 10:56:00 -0800619 if (fRRectButton.fEnabled) {
620 SkScalar rad = 32;
621 SkRect r;
caryclark88651ae2016-01-20 11:55:11 -0800622 r.set(&fPts[13], 2);
caryclark88c748a2015-02-18 10:56:00 -0800623 path.reset();
624 SkRRect rr;
625 rr.setRectXY(r, rad, rad);
626 path.addRRect(rr);
627 setForGeometry();
caryclark04e4d082015-02-20 06:33:57 -0800628 draw_stroke(canvas, path, width, 950, false);
caryclark88c748a2015-02-18 10:56:00 -0800629
630 path.reset();
631 SkRRect rr2;
632 rr.inset(width/2, width/2, &rr2);
633 path.addRRect(rr2, SkPath::kCCW_Direction);
634 rr.inset(-width/2, -width/2, &rr2);
635 path.addRRect(rr2, SkPath::kCW_Direction);
636 SkPaint paint;
637 paint.setAntiAlias(true);
638 paint.setColor(0x40FF8844);
639 canvas->drawPath(path, paint);
640 }
641
caryclark04e4d082015-02-20 06:33:57 -0800642 if (fCircleButton.fEnabled) {
643 path.reset();
644 SkRect r;
caryclark88651ae2016-01-20 11:55:11 -0800645 r.set(&fPts[15], 2);
caryclark04e4d082015-02-20 06:33:57 -0800646 path.addOval(r);
647 setForGeometry();
648 if (fCircleButton.fFill) {
caryclark88651ae2016-01-20 11:55:11 -0800649 if (fArcButton.fEnabled) {
650 SkPoint center;
651 if (arcCenter(&center)) {
halcanary9d524f22016-03-29 09:03:52 -0700652 r.set(center.fX - fRadius, center.fY - fRadius, center.fX + fRadius,
caryclark88651ae2016-01-20 11:55:11 -0800653 center.fY + fRadius);
654 }
655 }
caryclark04e4d082015-02-20 06:33:57 -0800656 draw_fill(canvas, r, width);
657 } else {
658 draw_stroke(canvas, path, width, 950, false);
659 }
660 }
661
caryclark88c748a2015-02-18 10:56:00 -0800662 if (fTextButton.fEnabled) {
663 path.reset();
664 SkPaint paint;
665 paint.setAntiAlias(true);
666 paint.setTextSize(fTextSize);
667 paint.getTextPath(fText.c_str(), fText.size(), 0, fTextSize, &path);
668 setForText();
caryclark04e4d082015-02-20 06:33:57 -0800669 draw_stroke(canvas, path, width * fWidthScale / fTextSize, fTextSize, true);
caryclark88c748a2015-02-18 10:56:00 -0800670 }
671
672 if (fAnimate) {
673 fWidth += fDWidth;
674 if (fDWidth > 0 && fWidth > kWidthMax) {
675 fDWidth = -fDWidth;
676 } else if (fDWidth < 0 && fWidth < kWidthMin) {
677 fDWidth = -fDWidth;
678 }
679 }
680 setAsNeeded();
caryclark04e4d082015-02-20 06:33:57 -0800681 if (fConicButton.fEnabled) {
682 draw_control(canvas, fWeightControl, fWeight, 0, 5, "weight");
683 }
caryclark88651ae2016-01-20 11:55:11 -0800684 if (fArcButton.fEnabled) {
685 draw_control(canvas, fRadiusControl, fRadius, 0, 500, "radius");
686 }
caryclarka76b7a3b2015-05-22 06:26:52 -0700687#ifdef SK_DEBUG
caryclark88c748a2015-02-18 10:56:00 -0800688 draw_control(canvas, fErrorControl, gDebugStrokerError, kStrokerErrorMin, kStrokerErrorMax,
689 "error");
690#endif
691 draw_control(canvas, fWidthControl, fWidth * fWidthScale, kWidthMin * fWidthScale,
692 kWidthMax * fWidthScale, "width");
693 draw_button(canvas, fQuadButton);
694 draw_button(canvas, fCubicButton);
caryclark04e4d082015-02-20 06:33:57 -0800695 draw_button(canvas, fConicButton);
caryclark88651ae2016-01-20 11:55:11 -0800696 draw_button(canvas, fArcButton);
caryclark88c748a2015-02-18 10:56:00 -0800697 draw_button(canvas, fRRectButton);
caryclark04e4d082015-02-20 06:33:57 -0800698 draw_button(canvas, fCircleButton);
caryclark88c748a2015-02-18 10:56:00 -0800699 draw_button(canvas, fTextButton);
halcanary96fcdcc2015-08-27 07:41:13 -0700700 this->inval(nullptr);
caryclark88c748a2015-02-18 10:56:00 -0800701 }
702
703 class MyClick : public Click {
704 public:
705 int fIndex;
706 MyClick(SkView* target, int index) : Click(target), fIndex(index) {}
707 };
708
709 virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y,
mtklein36352bf2015-03-25 18:17:31 -0700710 unsigned modi) override {
caryclark88c748a2015-02-18 10:56:00 -0800711 for (size_t i = 0; i < SK_ARRAY_COUNT(fPts); ++i) {
712 if (hittest(fPts[i], x, y)) {
713 return new MyClick(this, (int)i);
714 }
715 }
716 const SkRect& rectPt = SkRect::MakeXYWH(x, y, 1, 1);
caryclark04e4d082015-02-20 06:33:57 -0800717 if (fWeightControl.contains(rectPt)) {
718 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 1);
719 }
caryclark88651ae2016-01-20 11:55:11 -0800720 if (fRadiusControl.contains(rectPt)) {
721 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 2);
722 }
caryclarka76b7a3b2015-05-22 06:26:52 -0700723#ifdef SK_DEBUG
caryclark88c748a2015-02-18 10:56:00 -0800724 if (fErrorControl.contains(rectPt)) {
caryclark88651ae2016-01-20 11:55:11 -0800725 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 3);
caryclark88c748a2015-02-18 10:56:00 -0800726 }
727#endif
728 if (fWidthControl.contains(rectPt)) {
caryclark88651ae2016-01-20 11:55:11 -0800729 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 4);
caryclark88c748a2015-02-18 10:56:00 -0800730 }
731 if (fCubicButton.fBounds.contains(rectPt)) {
732 fCubicButton.fEnabled ^= true;
caryclark88651ae2016-01-20 11:55:11 -0800733 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 5);
caryclark88c748a2015-02-18 10:56:00 -0800734 }
caryclark04e4d082015-02-20 06:33:57 -0800735 if (fConicButton.fBounds.contains(rectPt)) {
736 fConicButton.fEnabled ^= true;
caryclark88651ae2016-01-20 11:55:11 -0800737 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 6);
caryclark04e4d082015-02-20 06:33:57 -0800738 }
caryclark88c748a2015-02-18 10:56:00 -0800739 if (fQuadButton.fBounds.contains(rectPt)) {
740 fQuadButton.fEnabled ^= true;
caryclark88651ae2016-01-20 11:55:11 -0800741 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 7);
742 }
743 if (fArcButton.fBounds.contains(rectPt)) {
744 fArcButton.fEnabled ^= true;
745 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 8);
caryclark88c748a2015-02-18 10:56:00 -0800746 }
747 if (fRRectButton.fBounds.contains(rectPt)) {
748 fRRectButton.fEnabled ^= true;
caryclark88651ae2016-01-20 11:55:11 -0800749 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 9);
caryclark04e4d082015-02-20 06:33:57 -0800750 }
751 if (fCircleButton.fBounds.contains(rectPt)) {
752 bool wasEnabled = fCircleButton.fEnabled;
753 fCircleButton.fEnabled = !fCircleButton.fFill;
754 fCircleButton.fFill = wasEnabled && !fCircleButton.fFill;
caryclark88651ae2016-01-20 11:55:11 -0800755 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 10);
caryclark88c748a2015-02-18 10:56:00 -0800756 }
757 if (fTextButton.fBounds.contains(rectPt)) {
758 fTextButton.fEnabled ^= true;
caryclark88651ae2016-01-20 11:55:11 -0800759 return new MyClick(this, (int) SK_ARRAY_COUNT(fPts) + 11);
caryclark88c748a2015-02-18 10:56:00 -0800760 }
761 return this->INHERITED::onFindClickHandler(x, y, modi);
762 }
763
764 static SkScalar MapScreenYtoValue(int y, const SkRect& control, SkScalar min,
765 SkScalar max) {
766 return (SkIntToScalar(y) - control.fTop) / control.height() * (max - min) + min;
767 }
768
mtklein36352bf2015-03-25 18:17:31 -0700769 bool onClick(Click* click) override {
caryclark88c748a2015-02-18 10:56:00 -0800770 int index = ((MyClick*)click)->fIndex;
771 if (index < (int) SK_ARRAY_COUNT(fPts)) {
772 fPts[index].offset(SkIntToScalar(click->fICurr.fX - click->fIPrev.fX),
773 SkIntToScalar(click->fICurr.fY - click->fIPrev.fY));
halcanary96fcdcc2015-08-27 07:41:13 -0700774 this->inval(nullptr);
caryclark04e4d082015-02-20 06:33:57 -0800775 } else if (index == (int) SK_ARRAY_COUNT(fPts) + 1) {
776 fWeight = MapScreenYtoValue(click->fICurr.fY, fWeightControl, 0, 5);
caryclark88651ae2016-01-20 11:55:11 -0800777 } else if (index == (int) SK_ARRAY_COUNT(fPts) + 2) {
778 fRadius = MapScreenYtoValue(click->fICurr.fY, fRadiusControl, 0, 500);
caryclark88c748a2015-02-18 10:56:00 -0800779 }
caryclarka76b7a3b2015-05-22 06:26:52 -0700780#ifdef SK_DEBUG
caryclark88651ae2016-01-20 11:55:11 -0800781 else if (index == (int) SK_ARRAY_COUNT(fPts) + 3) {
caryclark88c748a2015-02-18 10:56:00 -0800782 gDebugStrokerError = SkTMax(FLT_EPSILON, MapScreenYtoValue(click->fICurr.fY,
783 fErrorControl, kStrokerErrorMin, kStrokerErrorMax));
784 gDebugStrokerErrorSet = true;
785 }
786#endif
caryclark88651ae2016-01-20 11:55:11 -0800787 else if (index == (int) SK_ARRAY_COUNT(fPts) + 4) {
caryclark88c748a2015-02-18 10:56:00 -0800788 fWidth = SkTMax(FLT_EPSILON, MapScreenYtoValue(click->fICurr.fY, fWidthControl,
789 kWidthMin, kWidthMax));
790 fAnimate = fWidth <= kWidthMin;
791 }
792 return true;
793 }
794
795private:
796 typedef SkView INHERITED;
797};
798
799///////////////////////////////////////////////////////////////////////////////
800
801static SkView* F2() { return new QuadStrokerView; }
802static SkViewRegister gR2(F2);