73e6a66146
Change-Id: Ibfa2c0efa3e4addf21aa400e3d323675ac0185ef Reviewed-on: https://skia-review.googlesource.com/c/skia/+/551886 Reviewed-by: John Stiles <johnstiles@google.com> Commit-Queue: Herb Derby <herb@google.com>
808 lines
27 KiB
C++
808 lines
27 KiB
C++
/*
|
|
* Copyright 2012 Google Inc.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
* found in the LICENSE file.
|
|
*/
|
|
|
|
#include "include/core/SkBlendMode.h"
|
|
#include "include/core/SkCanvas.h"
|
|
#include "include/core/SkColor.h"
|
|
#include "include/core/SkFont.h"
|
|
#include "include/core/SkImageInfo.h"
|
|
#include "include/core/SkMatrix.h"
|
|
#include "include/core/SkPaint.h"
|
|
#include "include/core/SkPath.h"
|
|
#include "include/core/SkPathMeasure.h"
|
|
#include "include/core/SkPoint.h"
|
|
#include "include/core/SkRRect.h"
|
|
#include "include/core/SkRect.h"
|
|
#include "include/core/SkRefCnt.h"
|
|
#include "include/core/SkScalar.h"
|
|
#include "include/core/SkShader.h"
|
|
#include "include/core/SkString.h"
|
|
#include "include/core/SkSurface.h"
|
|
#include "include/core/SkTypes.h"
|
|
#include "include/private/SkTArray.h"
|
|
#include "include/private/SkTemplates.h"
|
|
#include "include/utils/SkTextUtils.h"
|
|
#include "samplecode/Sample.h"
|
|
#include "src/core/SkGeometry.h"
|
|
#include "src/core/SkPathPriv.h"
|
|
#include "src/core/SkPointPriv.h"
|
|
#include "src/core/SkStroke.h"
|
|
#include "tools/ToolUtils.h"
|
|
|
|
#include <cfloat>
|
|
|
|
class SkEvent;
|
|
|
|
static bool hittest(const SkPoint& target, SkScalar x, SkScalar y) {
|
|
const SkScalar TOL = 7;
|
|
return SkPoint::Distance(target, SkPoint::Make(x, y)) <= TOL;
|
|
}
|
|
|
|
static int getOnCurvePoints(const SkPath& path, SkPoint storage[]) {
|
|
int count = 0;
|
|
for (auto [verb, pts, w] : SkPathPriv::Iterate(path)) {
|
|
switch (verb) {
|
|
case SkPathVerb::kMove:
|
|
case SkPathVerb::kLine:
|
|
case SkPathVerb::kQuad:
|
|
case SkPathVerb::kConic:
|
|
case SkPathVerb::kCubic:
|
|
storage[count++] = pts[0];
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
return count;
|
|
}
|
|
|
|
static void getContourCounts(const SkPath& path, SkTArray<int>* contourCounts) {
|
|
int count = 0;
|
|
for (auto [verb, pts, w] : SkPathPriv::Iterate(path)) {
|
|
switch (verb) {
|
|
case SkPathVerb::kMove:
|
|
case SkPathVerb::kLine:
|
|
count += 1;
|
|
break;
|
|
case SkPathVerb::kQuad:
|
|
case SkPathVerb::kConic:
|
|
count += 2;
|
|
break;
|
|
case SkPathVerb::kCubic:
|
|
count += 3;
|
|
break;
|
|
case SkPathVerb::kClose:
|
|
contourCounts->push_back(count);
|
|
count = 0;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
if (count > 0) {
|
|
contourCounts->push_back(count);
|
|
}
|
|
}
|
|
|
|
static void erase(const sk_sp<SkSurface>& surface) {
|
|
SkCanvas* canvas = surface->getCanvas();
|
|
if (canvas) {
|
|
canvas->clear(SK_ColorTRANSPARENT);
|
|
}
|
|
}
|
|
|
|
struct StrokeTypeButton {
|
|
SkRect fBounds;
|
|
char fLabel;
|
|
bool fEnabled;
|
|
};
|
|
|
|
struct CircleTypeButton : public StrokeTypeButton {
|
|
bool fFill;
|
|
};
|
|
|
|
class QuadStrokerView : public Sample {
|
|
enum {
|
|
SKELETON_COLOR = 0xFF0000FF,
|
|
WIREFRAME_COLOR = 0x80FF0000
|
|
};
|
|
|
|
enum {
|
|
kCount = 18
|
|
};
|
|
SkPoint fPts[kCount];
|
|
SkRect fWeightControl;
|
|
SkRect fRadiusControl;
|
|
SkRect fErrorControl;
|
|
SkRect fWidthControl;
|
|
SkRect fBounds;
|
|
SkMatrix fMatrix, fInverse;
|
|
sk_sp<SkShader> fShader;
|
|
sk_sp<SkSurface> fMinSurface;
|
|
sk_sp<SkSurface> fMaxSurface;
|
|
StrokeTypeButton fCubicButton;
|
|
StrokeTypeButton fConicButton;
|
|
StrokeTypeButton fQuadButton;
|
|
StrokeTypeButton fArcButton;
|
|
StrokeTypeButton fRRectButton;
|
|
CircleTypeButton fCircleButton;
|
|
StrokeTypeButton fTextButton;
|
|
SkString fText;
|
|
SkScalar fTextSize;
|
|
SkScalar fWeight;
|
|
SkScalar fRadius;
|
|
SkScalar fWidth, fDWidth;
|
|
SkScalar fWidthScale;
|
|
int fW, fH, fZoom;
|
|
bool fAnimate;
|
|
bool fDrawRibs;
|
|
bool fDrawTangents;
|
|
bool fDrawTDivs;
|
|
#ifdef SK_DEBUG
|
|
#define kStrokerErrorMin 0.001f
|
|
#define kStrokerErrorMax 5
|
|
#endif
|
|
#define kWidthMin 1
|
|
#define kWidthMax 100
|
|
public:
|
|
QuadStrokerView() {
|
|
this->setBGColor(SK_ColorLTGRAY);
|
|
|
|
fPts[0].set(50, 200); // cubic
|
|
fPts[1].set(50, 100);
|
|
fPts[2].set(150, 50);
|
|
fPts[3].set(300, 50);
|
|
|
|
fPts[4].set(350, 200); // conic
|
|
fPts[5].set(350, 100);
|
|
fPts[6].set(450, 50);
|
|
|
|
fPts[7].set(150, 300); // quad
|
|
fPts[8].set(150, 200);
|
|
fPts[9].set(250, 150);
|
|
|
|
fPts[10].set(250, 200); // arc
|
|
fPts[11].set(250, 300);
|
|
fPts[12].set(150, 350);
|
|
|
|
fPts[13].set(200, 200); // rrect
|
|
fPts[14].set(400, 400);
|
|
|
|
fPts[15].set(250, 250); // oval
|
|
fPts[16].set(450, 450);
|
|
|
|
fText = "a";
|
|
fTextSize = 12;
|
|
fWidth = 50;
|
|
fDWidth = 0.25f;
|
|
fWeight = 1;
|
|
fRadius = 150;
|
|
|
|
fCubicButton.fLabel = 'C';
|
|
fCubicButton.fEnabled = false;
|
|
fConicButton.fLabel = 'K';
|
|
fConicButton.fEnabled = false;
|
|
fQuadButton.fLabel = 'Q';
|
|
fQuadButton.fEnabled = false;
|
|
fArcButton.fLabel = 'A';
|
|
fArcButton.fEnabled = true;
|
|
fRRectButton.fLabel = 'R';
|
|
fRRectButton.fEnabled = false;
|
|
fCircleButton.fLabel = 'O';
|
|
fCircleButton.fEnabled = true;
|
|
fCircleButton.fFill = true;
|
|
fTextButton.fLabel = 'T';
|
|
fTextButton.fEnabled = false;
|
|
fAnimate = false;
|
|
setAsNeeded();
|
|
}
|
|
|
|
protected:
|
|
SkString name() override { return SkString("QuadStroker"); }
|
|
|
|
bool onChar(SkUnichar uni) override {
|
|
if (fTextButton.fEnabled) {
|
|
switch (uni) {
|
|
case ' ':
|
|
fText = "";
|
|
break;
|
|
case '-':
|
|
fTextSize = std::max(1.0f, fTextSize - 1);
|
|
break;
|
|
case '+':
|
|
case '=':
|
|
fTextSize += 1;
|
|
break;
|
|
default:
|
|
fText.appendUnichar(uni);
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void onSizeChange() override {
|
|
fRadiusControl.setXYWH(this->width() - 200, 30, 30, 400);
|
|
fWeightControl.setXYWH(this->width() - 150, 30, 30, 400);
|
|
fErrorControl.setXYWH(this->width() - 100, 30, 30, 400);
|
|
fWidthControl.setXYWH(this->width() - 50, 30, 30, 400);
|
|
int buttonOffset = 450;
|
|
fCubicButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
|
|
buttonOffset += 50;
|
|
fConicButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
|
|
buttonOffset += 50;
|
|
fQuadButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
|
|
buttonOffset += 50;
|
|
fArcButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
|
|
buttonOffset += 50;
|
|
fRRectButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
|
|
buttonOffset += 50;
|
|
fCircleButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
|
|
buttonOffset += 50;
|
|
fTextButton.fBounds.setXYWH(this->width() - 50, SkIntToScalar(buttonOffset), 30, 30);
|
|
this->INHERITED::onSizeChange();
|
|
}
|
|
|
|
void copyMinToMax() {
|
|
erase(fMaxSurface);
|
|
SkCanvas* canvas = fMaxSurface->getCanvas();
|
|
canvas->save();
|
|
canvas->concat(fMatrix);
|
|
fMinSurface->draw(canvas, 0, 0);
|
|
canvas->restore();
|
|
|
|
SkPaint paint;
|
|
paint.setBlendMode(SkBlendMode::kClear);
|
|
for (int iy = 1; iy < fH; ++iy) {
|
|
SkScalar y = SkIntToScalar(iy * fZoom);
|
|
canvas->drawLine(0, y - SK_ScalarHalf, 999, y - SK_ScalarHalf, paint);
|
|
}
|
|
for (int ix = 1; ix < fW; ++ix) {
|
|
SkScalar x = SkIntToScalar(ix * fZoom);
|
|
canvas->drawLine(x - SK_ScalarHalf, 0, x - SK_ScalarHalf, 999, paint);
|
|
}
|
|
}
|
|
|
|
void setWHZ(int width, int height, int zoom) {
|
|
fZoom = zoom;
|
|
fBounds.setIWH(width * zoom, height * zoom);
|
|
fMatrix.setScale(SkIntToScalar(zoom), SkIntToScalar(zoom));
|
|
fInverse.setScale(SK_Scalar1 / zoom, SK_Scalar1 / zoom);
|
|
fShader = ToolUtils::create_checkerboard_shader(0xFFCCCCCC, 0xFFFFFFFF, zoom);
|
|
|
|
SkImageInfo info = SkImageInfo::MakeN32Premul(width, height);
|
|
fMinSurface = SkSurface::MakeRaster(info);
|
|
info = info.makeWH(width * zoom, height * zoom);
|
|
fMaxSurface = SkSurface::MakeRaster(info);
|
|
}
|
|
|
|
void draw_points(SkCanvas* canvas, const SkPath& path, SkColor color,
|
|
bool show_lines) {
|
|
SkPaint paint;
|
|
paint.setColor(color);
|
|
paint.setAlpha(0x80);
|
|
paint.setAntiAlias(true);
|
|
int n = path.countPoints();
|
|
SkAutoSTArray<32, SkPoint> pts(n);
|
|
if (show_lines && fDrawTangents) {
|
|
SkTArray<int> contourCounts;
|
|
getContourCounts(path, &contourCounts);
|
|
SkPoint* ptPtr = pts.get();
|
|
for (int i = 0; i < contourCounts.count(); ++i) {
|
|
int count = contourCounts[i];
|
|
path.getPoints(ptPtr, count);
|
|
canvas->drawPoints(SkCanvas::kPolygon_PointMode, count, ptPtr, paint);
|
|
ptPtr += count;
|
|
}
|
|
} else {
|
|
n = getOnCurvePoints(path, pts.get());
|
|
}
|
|
paint.setStrokeWidth(5);
|
|
canvas->drawPoints(SkCanvas::kPoints_PointMode, n, pts.get(), paint);
|
|
}
|
|
|
|
void draw_ribs(SkCanvas* canvas, const SkPath& path, SkScalar width,
|
|
SkColor color) {
|
|
const SkScalar radius = width / 2;
|
|
|
|
SkPathMeasure meas(path, false);
|
|
SkScalar total = meas.getLength();
|
|
|
|
SkScalar delta = 8;
|
|
SkPaint paint, labelP;
|
|
paint.setColor(color);
|
|
labelP.setColor(color & 0xff5f9f5f);
|
|
SkFont font;
|
|
SkPoint pos, tan;
|
|
int index = 0;
|
|
for (SkScalar dist = 0; dist <= total; dist += delta) {
|
|
if (meas.getPosTan(dist, &pos, &tan)) {
|
|
tan.scale(radius);
|
|
SkPointPriv::RotateCCW(&tan);
|
|
canvas->drawLine(pos.x() + tan.x(), pos.y() + tan.y(),
|
|
pos.x() - tan.x(), pos.y() - tan.y(), paint);
|
|
if (0 == index % 10) {
|
|
SkString label;
|
|
label.appendS32(index);
|
|
SkRect dot = SkRect::MakeXYWH(pos.x() - 2, pos.y() - 2, 4, 4);
|
|
canvas->drawRect(dot, labelP);
|
|
canvas->drawString(label,
|
|
pos.x() - tan.x() * 1.25f, pos.y() - tan.y() * 1.25f, font, labelP);
|
|
}
|
|
}
|
|
++index;
|
|
}
|
|
}
|
|
|
|
void draw_t_divs(SkCanvas* canvas, const SkPath& path, SkScalar width, SkColor color) {
|
|
const SkScalar radius = width / 2;
|
|
SkPaint paint;
|
|
paint.setColor(color);
|
|
SkPathMeasure meas(path, false);
|
|
SkScalar total = meas.getLength();
|
|
SkScalar delta = 8;
|
|
int ribs = 0;
|
|
for (SkScalar dist = 0; dist <= total; dist += delta) {
|
|
++ribs;
|
|
}
|
|
const uint8_t* verbs = SkPathPriv::VerbData(path);
|
|
if (path.countVerbs() < 2 || SkPath::kMove_Verb != verbs[0]) {
|
|
SkASSERT(0);
|
|
return;
|
|
}
|
|
auto verb = static_cast<SkPath::Verb>(verbs[1]);
|
|
SkASSERT(SkPath::kLine_Verb <= verb && verb <= SkPath::kCubic_Verb);
|
|
const SkPoint* pts = SkPathPriv::PointData(path);
|
|
SkPoint pos, tan;
|
|
for (int index = 0; index < ribs; ++index) {
|
|
SkScalar t = (SkScalar) index / ribs;
|
|
switch (verb) {
|
|
case SkPath::kLine_Verb:
|
|
tan = pts[1] - pts[0];
|
|
pos = pts[0];
|
|
pos.fX += tan.fX * t;
|
|
pos.fY += tan.fY * t;
|
|
break;
|
|
case SkPath::kQuad_Verb:
|
|
pos = SkEvalQuadAt(pts, t);
|
|
tan = SkEvalQuadTangentAt(pts, t);
|
|
break;
|
|
case SkPath::kConic_Verb: {
|
|
SkConic conic(pts, SkPathPriv::ConicWeightData(path)[0]);
|
|
pos = conic.evalAt(t);
|
|
tan = conic.evalTangentAt(t);
|
|
} break;
|
|
case SkPath::kCubic_Verb:
|
|
SkEvalCubicAt(pts, t, &pos, &tan, nullptr);
|
|
break;
|
|
default:
|
|
SkASSERT(0);
|
|
return;
|
|
}
|
|
tan.setLength(radius);
|
|
SkPointPriv::RotateCCW(&tan);
|
|
canvas->drawLine(pos.x() + tan.x(), pos.y() + tan.y(),
|
|
pos.x() - tan.x(), pos.y() - tan.y(), paint);
|
|
if (0 == index % 10) {
|
|
SkString label;
|
|
label.appendS32(index);
|
|
canvas->drawString(label,
|
|
pos.x() + tan.x() * 1.25f, pos.y() + tan.y() * 1.25f, SkFont(), paint);
|
|
}
|
|
}
|
|
}
|
|
|
|
void draw_stroke(SkCanvas* canvas, const SkPath& path, SkScalar width, SkScalar scale,
|
|
bool drawText) {
|
|
if (path.isEmpty()) {
|
|
return;
|
|
}
|
|
SkRect bounds = path.getBounds();
|
|
this->setWHZ(SkScalarCeilToInt(bounds.right()), drawText
|
|
? SkScalarRoundToInt(scale * 3 / 2) : SkScalarRoundToInt(scale),
|
|
SkScalarRoundToInt(950.0f / scale));
|
|
erase(fMinSurface);
|
|
SkPaint paint;
|
|
paint.setColor(0x1f1f0f0f);
|
|
paint.setStyle(SkPaint::kStroke_Style);
|
|
paint.setStrokeWidth(width * scale * scale);
|
|
paint.setColor(0x3f0f1f3f);
|
|
if (drawText) {
|
|
fMinSurface->getCanvas()->drawPath(path, paint);
|
|
this->copyMinToMax();
|
|
fMaxSurface->draw(canvas, 0, 0);
|
|
}
|
|
paint.setAntiAlias(true);
|
|
paint.setStyle(SkPaint::kStroke_Style);
|
|
paint.setStrokeWidth(1);
|
|
|
|
paint.setColor(SKELETON_COLOR);
|
|
SkPath scaled;
|
|
SkMatrix matrix;
|
|
matrix.reset();
|
|
matrix.setScale(950 / scale, 950 / scale);
|
|
if (drawText) {
|
|
path.transform(matrix, &scaled);
|
|
} else {
|
|
scaled = path;
|
|
}
|
|
canvas->drawPath(scaled, paint);
|
|
draw_points(canvas, scaled, SKELETON_COLOR, true);
|
|
|
|
if (fDrawRibs) {
|
|
draw_ribs(canvas, scaled, width, 0xFF00FF00);
|
|
}
|
|
|
|
if (fDrawTDivs) {
|
|
draw_t_divs(canvas, scaled, width, 0xFF3F3F00);
|
|
}
|
|
|
|
SkPath fill;
|
|
|
|
SkPaint p;
|
|
p.setStyle(SkPaint::kStroke_Style);
|
|
if (drawText) {
|
|
p.setStrokeWidth(width * scale * scale);
|
|
} else {
|
|
p.setStrokeWidth(width);
|
|
}
|
|
p.getFillPath(path, &fill);
|
|
SkPath scaledFill;
|
|
if (drawText) {
|
|
fill.transform(matrix, &scaledFill);
|
|
} else {
|
|
scaledFill = fill;
|
|
}
|
|
paint.setColor(WIREFRAME_COLOR);
|
|
canvas->drawPath(scaledFill, paint);
|
|
draw_points(canvas, scaledFill, WIREFRAME_COLOR, false);
|
|
}
|
|
|
|
void draw_fill(SkCanvas* canvas, const SkRect& rect, SkScalar width) {
|
|
if (rect.isEmpty()) {
|
|
return;
|
|
}
|
|
SkPaint paint;
|
|
paint.setColor(0x1f1f0f0f);
|
|
paint.setStyle(SkPaint::kStroke_Style);
|
|
paint.setStrokeWidth(width);
|
|
SkPath path;
|
|
SkScalar maxSide = std::max(rect.width(), rect.height()) / 2;
|
|
SkPoint center = { rect.fLeft + maxSide, rect.fTop + maxSide };
|
|
path.addCircle(center.fX, center.fY, maxSide);
|
|
canvas->drawPath(path, paint);
|
|
paint.setStyle(SkPaint::kFill_Style);
|
|
path.reset();
|
|
path.addCircle(center.fX, center.fY, maxSide - width / 2);
|
|
paint.setColor(0x3f0f1f3f);
|
|
canvas->drawPath(path, paint);
|
|
path.reset();
|
|
path.setFillType(SkPathFillType::kEvenOdd);
|
|
path.addCircle(center.fX, center.fY, maxSide + width / 2);
|
|
SkRect outside = SkRect::MakeXYWH(center.fX - maxSide - width, center.fY - maxSide - width,
|
|
(maxSide + width) * 2, (maxSide + width) * 2);
|
|
path.addRect(outside);
|
|
canvas->drawPath(path, paint);
|
|
}
|
|
|
|
void draw_button(SkCanvas* canvas, const StrokeTypeButton& button) {
|
|
SkPaint paint;
|
|
paint.setAntiAlias(true);
|
|
paint.setStyle(SkPaint::kStroke_Style);
|
|
paint.setColor(button.fEnabled ? 0xFF3F0000 : 0x6F3F0000);
|
|
canvas->drawRect(button.fBounds, paint);
|
|
paint.setColor(button.fEnabled ? 0xFF3F0000 : 0x6F3F0000);
|
|
paint.setStyle(SkPaint::kFill_Style);
|
|
SkFont font;
|
|
font.setSize(25.0f);
|
|
SkTextUtils::Draw(canvas, &button.fLabel, 1, SkTextEncoding::kUTF8,
|
|
button.fBounds.centerX(), button.fBounds.fBottom - 5,
|
|
font, paint, SkTextUtils::kCenter_Align);
|
|
}
|
|
|
|
void draw_control(SkCanvas* canvas, const SkRect& bounds, SkScalar value,
|
|
SkScalar min, SkScalar max, const char* name) {
|
|
SkPaint paint;
|
|
paint.setAntiAlias(true);
|
|
paint.setStyle(SkPaint::kStroke_Style);
|
|
canvas->drawRect(bounds, paint);
|
|
SkScalar scale = max - min;
|
|
SkScalar yPos = bounds.fTop + (value - min) * bounds.height() / scale;
|
|
paint.setColor(0xFFFF0000);
|
|
canvas->drawLine(bounds.fLeft - 5, yPos, bounds.fRight + 5, yPos, paint);
|
|
SkString label;
|
|
label.printf("%0.3g", value);
|
|
paint.setColor(0xFF000000);
|
|
paint.setStyle(SkPaint::kFill_Style);
|
|
SkFont font(nullptr, 11.0f);
|
|
canvas->drawString(label, bounds.fLeft + 5, yPos - 5, font, paint);
|
|
font.setSize(13.0f);
|
|
canvas->drawString(name, bounds.fLeft, bounds.bottom() + 11, font, paint);
|
|
}
|
|
|
|
void setForGeometry() {
|
|
fDrawRibs = true;
|
|
fDrawTangents = true;
|
|
fDrawTDivs = false;
|
|
fWidthScale = 1;
|
|
}
|
|
|
|
void setForText() {
|
|
fDrawRibs = fDrawTangents = fDrawTDivs = false;
|
|
fWidthScale = 0.002f;
|
|
}
|
|
|
|
void setForSingles() {
|
|
setForGeometry();
|
|
fDrawTDivs = true;
|
|
}
|
|
|
|
void setAsNeeded() {
|
|
if (fConicButton.fEnabled || fCubicButton.fEnabled || fQuadButton.fEnabled) {
|
|
setForSingles();
|
|
} else if (fRRectButton.fEnabled || fCircleButton.fEnabled || fArcButton.fEnabled) {
|
|
setForGeometry();
|
|
} else {
|
|
setForText();
|
|
}
|
|
}
|
|
|
|
bool arcCenter(SkPoint* center) {
|
|
SkPath path;
|
|
path.moveTo(fPts[10]);
|
|
path.arcTo(fPts[11], fPts[12], fRadius);
|
|
SkPath::Iter iter(path, false);
|
|
SkPoint pts[4];
|
|
iter.next(pts);
|
|
if (SkPath::kLine_Verb == iter.next(pts)) {
|
|
iter.next(pts);
|
|
}
|
|
SkVector before = pts[0] - pts[1];
|
|
SkVector after = pts[1] - pts[2];
|
|
before.setLength(fRadius);
|
|
after.setLength(fRadius);
|
|
SkVector beforeCCW, afterCCW;
|
|
SkPointPriv::RotateCCW(before, &beforeCCW);
|
|
SkPointPriv::RotateCCW(after, &afterCCW);
|
|
beforeCCW += pts[0];
|
|
afterCCW += pts[2];
|
|
*center = beforeCCW;
|
|
if (SkScalarNearlyEqual(beforeCCW.fX, afterCCW.fX)
|
|
&& SkScalarNearlyEqual(beforeCCW.fY, afterCCW.fY)) {
|
|
return true;
|
|
}
|
|
SkVector beforeCW, afterCW;
|
|
SkPointPriv::RotateCW(before, &beforeCW);
|
|
SkPointPriv::RotateCW(after, &afterCW);
|
|
beforeCW += pts[0];
|
|
afterCW += pts[2];
|
|
*center = beforeCW;
|
|
return SkScalarNearlyEqual(beforeCW.fX, afterCW.fX)
|
|
&& SkScalarNearlyEqual(beforeCCW.fY, afterCW.fY);
|
|
}
|
|
|
|
void onDrawContent(SkCanvas* canvas) override {
|
|
SkPath path;
|
|
SkScalar width = fWidth;
|
|
|
|
if (fCubicButton.fEnabled) {
|
|
path.moveTo(fPts[0]);
|
|
path.cubicTo(fPts[1], fPts[2], fPts[3]);
|
|
setForSingles();
|
|
draw_stroke(canvas, path, width, 950, false);
|
|
}
|
|
|
|
if (fConicButton.fEnabled) {
|
|
path.reset();
|
|
path.moveTo(fPts[4]);
|
|
path.conicTo(fPts[5], fPts[6], fWeight);
|
|
setForSingles();
|
|
draw_stroke(canvas, path, width, 950, false);
|
|
}
|
|
|
|
if (fQuadButton.fEnabled) {
|
|
path.reset();
|
|
path.moveTo(fPts[7]);
|
|
path.quadTo(fPts[8], fPts[9]);
|
|
setForSingles();
|
|
draw_stroke(canvas, path, width, 950, false);
|
|
}
|
|
|
|
if (fArcButton.fEnabled) {
|
|
path.reset();
|
|
path.moveTo(fPts[10]);
|
|
path.arcTo(fPts[11], fPts[12], fRadius);
|
|
setForGeometry();
|
|
draw_stroke(canvas, path, width, 950, false);
|
|
SkPath pathPts;
|
|
pathPts.moveTo(fPts[10]);
|
|
pathPts.lineTo(fPts[11]);
|
|
pathPts.lineTo(fPts[12]);
|
|
draw_points(canvas, pathPts, SK_ColorDKGRAY, true);
|
|
}
|
|
|
|
if (fRRectButton.fEnabled) {
|
|
SkScalar rad = 32;
|
|
SkRect r;
|
|
r.setBounds(&fPts[13], 2);
|
|
path.reset();
|
|
SkRRect rr;
|
|
rr.setRectXY(r, rad, rad);
|
|
path.addRRect(rr);
|
|
setForGeometry();
|
|
draw_stroke(canvas, path, width, 950, false);
|
|
|
|
path.reset();
|
|
SkRRect rr2;
|
|
rr.inset(width/2, width/2, &rr2);
|
|
path.addRRect(rr2, SkPathDirection::kCCW);
|
|
rr.inset(-width/2, -width/2, &rr2);
|
|
path.addRRect(rr2, SkPathDirection::kCW);
|
|
SkPaint paint;
|
|
paint.setAntiAlias(true);
|
|
paint.setColor(0x40FF8844);
|
|
canvas->drawPath(path, paint);
|
|
}
|
|
|
|
if (fCircleButton.fEnabled) {
|
|
path.reset();
|
|
SkRect r;
|
|
r.setBounds(&fPts[15], 2);
|
|
path.addOval(r);
|
|
setForGeometry();
|
|
if (fCircleButton.fFill) {
|
|
if (fArcButton.fEnabled) {
|
|
SkPoint center;
|
|
if (arcCenter(¢er)) {
|
|
r.setLTRB(center.fX - fRadius, center.fY - fRadius,
|
|
center.fX + fRadius, center.fY + fRadius);
|
|
}
|
|
}
|
|
draw_fill(canvas, r, width);
|
|
} else {
|
|
draw_stroke(canvas, path, width, 950, false);
|
|
}
|
|
}
|
|
|
|
if (fTextButton.fEnabled) {
|
|
path.reset();
|
|
SkFont font;
|
|
font.setSize(fTextSize);
|
|
SkTextUtils::GetPath(fText.c_str(), fText.size(), SkTextEncoding::kUTF8,
|
|
0, fTextSize, font, &path);
|
|
setForText();
|
|
draw_stroke(canvas, path, width * fWidthScale / fTextSize, fTextSize, true);
|
|
}
|
|
|
|
if (fAnimate) {
|
|
fWidth += fDWidth;
|
|
if (fDWidth > 0 && fWidth > kWidthMax) {
|
|
fDWidth = -fDWidth;
|
|
} else if (fDWidth < 0 && fWidth < kWidthMin) {
|
|
fDWidth = -fDWidth;
|
|
}
|
|
}
|
|
setAsNeeded();
|
|
if (fConicButton.fEnabled) {
|
|
draw_control(canvas, fWeightControl, fWeight, 0, 5, "weight");
|
|
}
|
|
if (fArcButton.fEnabled) {
|
|
draw_control(canvas, fRadiusControl, fRadius, 0, 500, "radius");
|
|
}
|
|
#ifdef SK_DEBUG
|
|
draw_control(canvas, fErrorControl, gDebugStrokerError, kStrokerErrorMin, kStrokerErrorMax,
|
|
"error");
|
|
#endif
|
|
draw_control(canvas, fWidthControl, fWidth * fWidthScale, kWidthMin * fWidthScale,
|
|
kWidthMax * fWidthScale, "width");
|
|
draw_button(canvas, fQuadButton);
|
|
draw_button(canvas, fCubicButton);
|
|
draw_button(canvas, fConicButton);
|
|
draw_button(canvas, fArcButton);
|
|
draw_button(canvas, fRRectButton);
|
|
draw_button(canvas, fCircleButton);
|
|
draw_button(canvas, fTextButton);
|
|
}
|
|
|
|
class MyClick : public Click {
|
|
public:
|
|
int fIndex;
|
|
MyClick(int index) : fIndex(index) {}
|
|
};
|
|
|
|
Sample::Click* onFindClickHandler(SkScalar x, SkScalar y, skui::ModifierKey modi) override {
|
|
for (size_t i = 0; i < std::size(fPts); ++i) {
|
|
if (hittest(fPts[i], x, y)) {
|
|
return new MyClick((int)i);
|
|
}
|
|
}
|
|
const SkRect& rectPt = SkRect::MakeXYWH(x, y, 1, 1);
|
|
if (fWeightControl.contains(rectPt)) {
|
|
return new MyClick((int) std::size(fPts) + 1);
|
|
}
|
|
if (fRadiusControl.contains(rectPt)) {
|
|
return new MyClick((int) std::size(fPts) + 2);
|
|
}
|
|
#ifdef SK_DEBUG
|
|
if (fErrorControl.contains(rectPt)) {
|
|
return new MyClick((int) std::size(fPts) + 3);
|
|
}
|
|
#endif
|
|
if (fWidthControl.contains(rectPt)) {
|
|
return new MyClick((int) std::size(fPts) + 4);
|
|
}
|
|
if (fCubicButton.fBounds.contains(rectPt)) {
|
|
fCubicButton.fEnabled ^= true;
|
|
return new MyClick((int) std::size(fPts) + 5);
|
|
}
|
|
if (fConicButton.fBounds.contains(rectPt)) {
|
|
fConicButton.fEnabled ^= true;
|
|
return new MyClick((int) std::size(fPts) + 6);
|
|
}
|
|
if (fQuadButton.fBounds.contains(rectPt)) {
|
|
fQuadButton.fEnabled ^= true;
|
|
return new MyClick((int) std::size(fPts) + 7);
|
|
}
|
|
if (fArcButton.fBounds.contains(rectPt)) {
|
|
fArcButton.fEnabled ^= true;
|
|
return new MyClick((int) std::size(fPts) + 8);
|
|
}
|
|
if (fRRectButton.fBounds.contains(rectPt)) {
|
|
fRRectButton.fEnabled ^= true;
|
|
return new MyClick((int) std::size(fPts) + 9);
|
|
}
|
|
if (fCircleButton.fBounds.contains(rectPt)) {
|
|
bool wasEnabled = fCircleButton.fEnabled;
|
|
fCircleButton.fEnabled = !fCircleButton.fFill;
|
|
fCircleButton.fFill = wasEnabled && !fCircleButton.fFill;
|
|
return new MyClick((int) std::size(fPts) + 10);
|
|
}
|
|
if (fTextButton.fBounds.contains(rectPt)) {
|
|
fTextButton.fEnabled ^= true;
|
|
return new MyClick((int) std::size(fPts) + 11);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
static SkScalar MapScreenYtoValue(SkScalar y, const SkRect& control, SkScalar min,
|
|
SkScalar max) {
|
|
return (y - control.fTop) / control.height() * (max - min) + min;
|
|
}
|
|
|
|
bool onClick(Click* click) override {
|
|
int index = ((MyClick*)click)->fIndex;
|
|
if (index < (int) std::size(fPts)) {
|
|
fPts[index].offset(click->fCurr.fX - click->fPrev.fX,
|
|
click->fCurr.fY - click->fPrev.fY);
|
|
} else if (index == (int) std::size(fPts) + 1) {
|
|
fWeight = MapScreenYtoValue(click->fCurr.fY, fWeightControl, 0, 5);
|
|
} else if (index == (int) std::size(fPts) + 2) {
|
|
fRadius = MapScreenYtoValue(click->fCurr.fY, fRadiusControl, 0, 500);
|
|
}
|
|
#ifdef SK_DEBUG
|
|
else if (index == (int) std::size(fPts) + 3) {
|
|
gDebugStrokerError = std::max(FLT_EPSILON, MapScreenYtoValue(click->fCurr.fY,
|
|
fErrorControl, kStrokerErrorMin, kStrokerErrorMax));
|
|
gDebugStrokerErrorSet = true;
|
|
}
|
|
#endif
|
|
else if (index == (int) std::size(fPts) + 4) {
|
|
fWidth = std::max(FLT_EPSILON, MapScreenYtoValue(click->fCurr.fY, fWidthControl,
|
|
kWidthMin, kWidthMax));
|
|
fAnimate = fWidth <= kWidthMin;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
private:
|
|
using INHERITED = Sample;
|
|
};
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
DEF_SAMPLE( return new QuadStrokerView(); )
|