|  | /* | 
|  | * Copyright 2016 Google Inc. | 
|  | * | 
|  | * Use of this source code is governed by a BSD-style license that can be | 
|  | * found in the LICENSE file. | 
|  | */ | 
|  | #include "Sample.h" | 
|  | #include "SkCanvas.h" | 
|  | #include "SkPath.h" | 
|  |  | 
|  | #include <iostream> | 
|  | #include <cmath> | 
|  |  | 
|  | #define PI SK_ScalarPI | 
|  |  | 
|  | #define LIN_SEGMENTS 10 | 
|  |  | 
|  | class OverstrokeView : public Sample { | 
|  | public: | 
|  | SkScalar fStroke; | 
|  | int fPathType;  // super lazy enum | 
|  | bool fClosePath; | 
|  | bool fDrawFillPath; | 
|  | bool fDumpHex; | 
|  | OverstrokeView() { | 
|  | fStroke = 5; | 
|  | fPathType = 0; | 
|  | fClosePath = false; | 
|  | fDrawFillPath = false; | 
|  | fDumpHex = false; | 
|  | this->setBGColor(0xFFFFFFFF); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | bool onQuery(Sample::Event* evt) override { | 
|  | if (Sample::TitleQ(*evt)) { | 
|  | Sample::TitleR(evt, "PathOverstroke"); | 
|  | return true; | 
|  | } | 
|  | SkUnichar uni; | 
|  | if (Sample::CharQ(*evt, &uni)) { | 
|  | switch (uni) { | 
|  | case ',': | 
|  | fStroke += 1.0; | 
|  | return true; | 
|  | case '.': | 
|  | fStroke -= 1.0; | 
|  | return true; | 
|  | case 'x': | 
|  | fPathType = (fPathType + 1) % 4; | 
|  | return true; | 
|  | case 'c': | 
|  | fClosePath = !fClosePath; | 
|  | return true; | 
|  | case 'f': | 
|  | fDrawFillPath = !fDrawFillPath; | 
|  | return true; | 
|  | case 'D': | 
|  | fDumpHex = !fDumpHex; | 
|  | return true; | 
|  | default: | 
|  | break; | 
|  | } | 
|  | } | 
|  | return this->INHERITED::onQuery(evt); | 
|  | } | 
|  |  | 
|  | SkPath quadPath(SkPoint p1, SkPoint p2) { | 
|  | SkASSERT(p1.y() == p2.y()); | 
|  |  | 
|  | SkPath path; | 
|  | path.moveTo(p1); | 
|  | path.lineTo(p2); | 
|  |  | 
|  | SkPoint p3 = SkPoint::Make((p1.x() + p2.x()) / 2.0f, p1.y() * 0.7f); | 
|  |  | 
|  | path.quadTo(p3, p1); | 
|  |  | 
|  | return path; | 
|  | } | 
|  |  | 
|  | SkPath cubicPath(SkPoint p1, SkPoint p2) { | 
|  | SkASSERT(p1.y() == p2.y()); | 
|  |  | 
|  | SkPath path; | 
|  | path.moveTo(p1); | 
|  |  | 
|  | SkPoint p3 = SkPoint::Make((p1.x() + p2.x()) / 3.0f, p1.y() * 0.7f); | 
|  | SkPoint p4 = SkPoint::Make(2.0f*(p1.x() + p2.x()) / 3.0f, p1.y() * 1.5f); | 
|  |  | 
|  | path.cubicTo(p3, p4, p2); | 
|  |  | 
|  | return path; | 
|  | } | 
|  |  | 
|  | SkPath linSemicirclePath(SkPoint p1, SkPoint p2) { | 
|  | SkASSERT(p1.y() == p2.y()); | 
|  |  | 
|  | SkPath path; | 
|  | path.moveTo(p1); | 
|  | path.lineTo(p2); | 
|  |  | 
|  | SkPoint pt; | 
|  |  | 
|  | for (int i = 0; i < LIN_SEGMENTS; i++) { | 
|  | float theta = i * PI / (LIN_SEGMENTS); | 
|  | SkScalar x = 65 + 15 * cos(theta); | 
|  | SkScalar y = 50 - 15 * sin(theta); | 
|  | pt = SkPoint::Make(x, y); | 
|  | path.lineTo(pt); | 
|  | } | 
|  | path.lineTo(p1); | 
|  |  | 
|  | return path; | 
|  | } | 
|  |  | 
|  | SkPath rectPath(SkPoint p1) { | 
|  | SkRect r = SkRect::MakeXYWH(p1.fX, p1.fY, 20, 20); | 
|  | SkPath path; | 
|  | path.addRect(r); | 
|  |  | 
|  | return path; | 
|  | } | 
|  |  | 
|  | void onDrawContent(SkCanvas* canvas) override { | 
|  | const float SCALE = 1; | 
|  |  | 
|  | canvas->translate(30, 40); | 
|  | canvas->scale(SCALE, SCALE); | 
|  |  | 
|  | SkPoint p1 = SkPoint::Make(50, 50); | 
|  | SkPoint p2 = SkPoint::Make(80, 50); | 
|  |  | 
|  | SkPath path; | 
|  | switch (fPathType) { | 
|  | case 0: | 
|  | path = quadPath(p1, p2); | 
|  | break; | 
|  | case 1: | 
|  | path = cubicPath(p1, p2); | 
|  | break; | 
|  | case 2: | 
|  | path = rectPath(p1); | 
|  | break; | 
|  | case 3: | 
|  | path = linSemicirclePath(p1, p2); | 
|  | break; | 
|  | default: | 
|  | path = quadPath(p1, p2); | 
|  | break; | 
|  | } | 
|  |  | 
|  | if (fClosePath) { | 
|  | path.close(); | 
|  | } | 
|  |  | 
|  | SkPaint p; | 
|  | p.setColor(SK_ColorRED); | 
|  | p.setAntiAlias(true); | 
|  | p.setStyle(SkPaint::kStroke_Style); | 
|  | p.setStrokeWidth(fStroke); | 
|  |  | 
|  | canvas->drawPath(path, p); | 
|  |  | 
|  | if (fDumpHex) { | 
|  | std::cerr << "path dumpHex" << std::endl; | 
|  | path.dumpHex(); | 
|  | } | 
|  |  | 
|  | SkPaint hairp; | 
|  | hairp.setColor(SK_ColorBLACK); | 
|  | hairp.setAntiAlias(true); | 
|  | hairp.setStyle(SkPaint::kStroke_Style); | 
|  |  | 
|  | if (fDrawFillPath) { | 
|  | SkPath fillpath; | 
|  | p.getFillPath(path, &fillpath); | 
|  |  | 
|  | canvas->drawPath(fillpath, hairp); | 
|  |  | 
|  | if (fDumpHex) { | 
|  | std::cerr << "fillpath dumpHex" << std::endl; | 
|  | fillpath.dumpHex(); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (fDumpHex) { | 
|  | std::cerr << std::endl; | 
|  |  | 
|  | fDumpHex = false; | 
|  | } | 
|  |  | 
|  | // draw original path with green hairline | 
|  | hairp.setColor(SK_ColorGREEN); | 
|  | canvas->drawPath(path, hairp); | 
|  | } | 
|  |  | 
|  | private: | 
|  | typedef Sample INHERITED; | 
|  | }; | 
|  |  | 
|  | /////////////////////////////////////////////////////////////////////////////// | 
|  |  | 
|  | DEF_SAMPLE( return new OverstrokeView(); ) |