/* * Copyright 2011 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "SampleCode.h" #include "SkAnimTimer.h" #include "SkView.h" #include "SkCanvas.h" #include "SkGradientShader.h" #include "SkGraphics.h" #include "SkPath.h" #include "SkRegion.h" #include "SkShader.h" #include "SkUtils.h" #include "SkColorPriv.h" #include "SkColorFilter.h" #include "SkParsePath.h" #include "SkTime.h" #include "SkTypeface.h" #include "SkGeometry.h" #include // http://code.google.com/p/skia/issues/detail?id=32 static void test_cubic() { SkPoint src[4] = { { 556.25000f, 523.03003f }, { 556.23999f, 522.96002f }, { 556.21997f, 522.89001f }, { 556.21997f, 522.82001f } }; SkPoint dst[11]; dst[10].set(42, -42); // one past the end, that we don't clobber these SkScalar tval[] = { 0.33333334f, 0.99999994f }; SkChopCubicAt(src, dst, tval, 2); #if 0 for (int i = 0; i < 11; i++) { SkDebugf("--- %d [%g %g]\n", i, dst[i].fX, dst[i].fY); } #endif } static void test_cubic2() { const char* str = "M2242 -590088L-377758 9.94099e+07L-377758 9.94099e+07L2242 -590088Z"; SkPath path; SkParsePath::FromSVGString(str, &path); { SkRect r = path.getBounds(); SkIRect ir; r.round(&ir); SkDebugf("[%g %g %g %g] [%x %x %x %x]\n", SkScalarToDouble(r.fLeft), SkScalarToDouble(r.fTop), SkScalarToDouble(r.fRight), SkScalarToDouble(r.fBottom), ir.fLeft, ir.fTop, ir.fRight, ir.fBottom); } SkBitmap bitmap; bitmap.allocN32Pixels(300, 200); SkCanvas canvas(bitmap); SkPaint paint; paint.setAntiAlias(true); canvas.drawPath(path, paint); } class PathView : public SampleView { SkScalar fPrevSecs; public: SkScalar fDStroke, fStroke, fMinStroke, fMaxStroke; SkPath fPath[6]; bool fShowHairline; bool fOnce; PathView() { fPrevSecs = 0; fOnce = false; } void init() { if (fOnce) { return; } fOnce = true; test_cubic(); test_cubic2(); fShowHairline = false; fDStroke = 1; fStroke = 10; fMinStroke = 10; fMaxStroke = 180; const SkScalar V = 85; fPath[0].moveTo(40, 70); fPath[0].lineTo(70, 70 + SK_ScalarHalf); fPath[0].lineTo(110, 70); fPath[1].moveTo(40, 70); fPath[1].lineTo(70, 70 - SK_ScalarHalf); fPath[1].lineTo(110, 70); fPath[2].moveTo(V, V); fPath[2].lineTo(50, V); fPath[2].lineTo(50, 50); fPath[3].moveTo(50, 50); fPath[3].lineTo(50, V); fPath[3].lineTo(V, V); fPath[4].moveTo(50, 50); fPath[4].lineTo(50, V); fPath[4].lineTo(52, 50); fPath[5].moveTo(52, 50); fPath[5].lineTo(50, V); fPath[5].lineTo(50, 50); this->setBGColor(0xFFDDDDDD); } protected: // overrides from SkEventSink bool onQuery(SkEvent* evt) override { if (SampleCode::TitleQ(*evt)) { SampleCode::TitleR(evt, "Paths"); return true; } return this->INHERITED::onQuery(evt); } void drawPath(SkCanvas* canvas, const SkPath& path, SkPaint::Join j) { SkPaint paint; paint.setAntiAlias(true); paint.setStyle(SkPaint::kStroke_Style); paint.setStrokeJoin(j); paint.setStrokeWidth(fStroke); if (fShowHairline) { SkPath fill; paint.getFillPath(path, &fill); paint.setStrokeWidth(0); canvas->drawPath(fill, paint); } else { canvas->drawPath(path, paint); } paint.setColor(SK_ColorRED); paint.setStrokeWidth(0); canvas->drawPath(path, paint); } void onDrawContent(SkCanvas* canvas) override { this->init(); canvas->translate(50, 50); static const SkPaint::Join gJoins[] = { SkPaint::kBevel_Join, SkPaint::kMiter_Join, SkPaint::kRound_Join }; for (size_t i = 0; i < SK_ARRAY_COUNT(gJoins); i++) { canvas->save(); for (size_t j = 0; j < SK_ARRAY_COUNT(fPath); j++) { this->drawPath(canvas, fPath[j], gJoins[i]); canvas->translate(200, 0); } canvas->restore(); canvas->translate(0, 200); } } bool onAnimate(const SkAnimTimer& timer) override { SkScalar currSecs = timer.scaled(100); SkScalar delta = currSecs - fPrevSecs; fPrevSecs = currSecs; fStroke += fDStroke * delta; if (fStroke > fMaxStroke || fStroke < fMinStroke) { fDStroke = -fDStroke; } return true; } SkView::Click* onFindClickHandler(SkScalar x, SkScalar y, unsigned modi) override { fShowHairline = !fShowHairline; this->inval(nullptr); return this->INHERITED::onFindClickHandler(x, y, modi); } private: typedef SampleView INHERITED; }; DEF_SAMPLE( return new PathView; ) ////////////////////////////////////////////////////////////////////////////// #include "SkCornerPathEffect.h" #include "SkRandom.h" class ArcToView : public SampleView { bool fDoFrame, fDoCorner, fDoConic; SkPaint fPtsPaint, fSkeletonPaint, fCornerPaint; public: enum { N = 4 }; SkPoint fPts[N]; ArcToView() : fDoFrame(false), fDoCorner(false), fDoConic(false) { SkRandom rand; for (int i = 0; i < N; ++i) { fPts[i].fX = 20 + rand.nextUScalar1() * 640; fPts[i].fY = 20 + rand.nextUScalar1() * 480; } const SkScalar rad = 50; fPtsPaint.setAntiAlias(true); fPtsPaint.setStrokeWidth(15); fPtsPaint.setStrokeCap(SkPaint::kRound_Cap); fCornerPaint.setAntiAlias(true); fCornerPaint.setStyle(SkPaint::kStroke_Style); fCornerPaint.setStrokeWidth(13); fCornerPaint.setColor(SK_ColorGREEN); fCornerPaint.setPathEffect(SkCornerPathEffect::Make(rad*2)); fSkeletonPaint.setAntiAlias(true); fSkeletonPaint.setStyle(SkPaint::kStroke_Style); fSkeletonPaint.setColor(SK_ColorRED); } void toggle(bool& value) { value = !value; this->inval(nullptr); } protected: // overrides from SkEventSink bool onQuery(SkEvent* evt) override { if (SampleCode::TitleQ(*evt)) { SampleCode::TitleR(evt, "ArcTo"); return true; } SkUnichar uni; if (SampleCode::CharQ(*evt, &uni)) { switch (uni) { case '1': this->toggle(fDoFrame); return true; case '2': this->toggle(fDoCorner); return true; case '3': this->toggle(fDoConic); return true; default: break; } } return this->INHERITED::onQuery(evt); } void makePath(SkPath* path) { path->moveTo(fPts[0]); for (int i = 1; i < N; ++i) { path->lineTo(fPts[i]); } if (!fDoFrame) { path->close(); } } void onDrawContent(SkCanvas* canvas) override { canvas->drawPoints(SkCanvas::kPoints_PointMode, N, fPts, fPtsPaint); SkPath path; this->makePath(&path); if (fDoCorner) { canvas->drawPath(path, fCornerPaint); } canvas->drawPath(path, fSkeletonPaint); } bool onClick(Click* click) override { int32_t index; if (click->fMeta.findS32("index", &index)) { SkASSERT((unsigned)index < N); fPts[index] = click->fCurr; this->inval(nullptr); return true; } return false; } SkView::Click* onFindClickHandler(SkScalar x, SkScalar y, unsigned modi) override { const SkScalar tol = 4; const SkRect r = SkRect::MakeXYWH(x - tol, y - tol, tol * 2, tol * 2); for (int i = 0; i < N; ++i) { if (r.intersects(SkRect::MakeXYWH(fPts[i].fX, fPts[i].fY, 1, 1))) { Click* click = new Click(this); click->fMeta.setS32("index", i); return click; } } return this->INHERITED::onFindClickHandler(x, y, modi); } private: typedef SampleView INHERITED; }; DEF_SAMPLE( return new ArcToView; ) ///////////// class FatStroke : public SampleView { bool fClosed, fShowStroke, fShowHidden, fShowSkeleton; int fJoinType, fCapType; float fWidth = 30; SkPaint fPtsPaint, fHiddenPaint, fSkeletonPaint, fStrokePaint; public: enum { N = 4 }; SkPoint fPts[N]; FatStroke() : fClosed(false), fShowStroke(true), fShowHidden(false), fShowSkeleton(true), fJoinType(0), fCapType(0) { SkRandom rand; for (int i = 0; i < N; ++i) { fPts[i].fX = 20 + rand.nextUScalar1() * 640; fPts[i].fY = 20 + rand.nextUScalar1() * 480; } fPtsPaint.setAntiAlias(true); fPtsPaint.setStrokeWidth(10); fPtsPaint.setStrokeCap(SkPaint::kRound_Cap); fHiddenPaint.setAntiAlias(true); fHiddenPaint.setStyle(SkPaint::kStroke_Style); fHiddenPaint.setColor(0xFF0000FF); fStrokePaint.setAntiAlias(true); fStrokePaint.setStyle(SkPaint::kStroke_Style); fStrokePaint.setStrokeWidth(50); fStrokePaint.setColor(0x8000FF00); fSkeletonPaint.setAntiAlias(true); fSkeletonPaint.setStyle(SkPaint::kStroke_Style); fSkeletonPaint.setColor(SK_ColorRED); } void toggle(bool& value) { value = !value; this->inval(nullptr); } void toggle3(int& value) { value = (value + 1) % 3; this->inval(nullptr); } protected: // overrides from SkEventSink bool onQuery(SkEvent* evt) override { if (SampleCode::TitleQ(*evt)) { SampleCode::TitleR(evt, "FatStroke"); return true; } SkUnichar uni; if (SampleCode::CharQ(*evt, &uni)) { switch (uni) { case '1': this->toggle(fShowSkeleton); return true; case '2': this->toggle(fShowStroke); return true; case '3': this->toggle(fShowHidden); return true; case '4': this->toggle3(fJoinType); return true; case '5': this->toggle3(fCapType); return true; case '6': this->toggle(fClosed); return true; case '-': fWidth -= 5; this->inval(nullptr); return true; case '=': fWidth += 5; this->inval(nullptr); return true; default: break; } } return this->INHERITED::onQuery(evt); } void makePath(SkPath* path) { path->moveTo(fPts[0]); for (int i = 1; i < N; ++i) { path->lineTo(fPts[i]); } if (fClosed) { path->close(); } } void onDrawContent(SkCanvas* canvas) override { canvas->drawColor(0xFFEEEEEE); SkPath path; this->makePath(&path); fStrokePaint.setStrokeWidth(fWidth); fStrokePaint.setStrokeJoin((SkPaint::Join)fJoinType); fStrokePaint.setStrokeCap((SkPaint::Cap)fCapType); if (fShowStroke) { canvas->drawPath(path, fStrokePaint); } if (fShowHidden) { SkPath hidden; fStrokePaint.getFillPath(path, &hidden); canvas->drawPath(hidden, fHiddenPaint); } if (fShowSkeleton) { canvas->drawPath(path, fSkeletonPaint); } canvas->drawPoints(SkCanvas::kPoints_PointMode, N, fPts, fPtsPaint); } bool onClick(Click* click) override { int32_t index; if (click->fMeta.findS32("index", &index)) { SkASSERT((unsigned)index < N); fPts[index] = click->fCurr; this->inval(nullptr); return true; } return false; } SkView::Click* onFindClickHandler(SkScalar x, SkScalar y, unsigned modi) override { const SkScalar tol = 4; const SkRect r = SkRect::MakeXYWH(x - tol, y - tol, tol * 2, tol * 2); for (int i = 0; i < N; ++i) { if (r.intersects(SkRect::MakeXYWH(fPts[i].fX, fPts[i].fY, 1, 1))) { Click* click = new Click(this); click->fMeta.setS32("index", i); return click; } } return this->INHERITED::onFindClickHandler(x, y, modi); } private: typedef SampleView INHERITED; }; DEF_SAMPLE( return new FatStroke; )