|  |  | 
|  | /* | 
|  | * 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 "SkView.h" | 
|  | #include "SkCanvas.h" | 
|  | #include "SkGradientShader.h" | 
|  | #include "SkGraphics.h" | 
|  | #include "SkImageDecoder.h" | 
|  | #include "SkPath.h" | 
|  | #include "SkRegion.h" | 
|  | #include "SkShader.h" | 
|  | #include "SkUtils.h" | 
|  | #include "SkXfermode.h" | 
|  | #include "SkColorPriv.h" | 
|  | #include "SkColorFilter.h" | 
|  | #include "SkParsePath.h" | 
|  | #include "SkTime.h" | 
|  | #include "SkTypeface.h" | 
|  |  | 
|  | #include "SkGeometry.h" | 
|  |  | 
|  | // 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); | 
|  |  | 
|  | { | 
|  | #ifdef SK_BUILD_FOR_WIN | 
|  | // windows doesn't have strtof | 
|  | float x = (float)strtod("9.94099e+07", NULL); | 
|  | #else | 
|  | float x = strtof("9.94099e+07", NULL); | 
|  | #endif | 
|  | int ix = (int)x; | 
|  | int fx = (int)(x * 65536); | 
|  | int ffx = SkScalarToFixed(x); | 
|  | SkDebugf("%g %x %x %x\n", x, ix, fx, ffx); | 
|  |  | 
|  | 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.setConfig(SkBitmap::kARGB_8888_Config, 300, 200); | 
|  | bitmap.allocPixels(); | 
|  |  | 
|  | SkCanvas canvas(bitmap); | 
|  | SkPaint paint; | 
|  | paint.setAntiAlias(true); | 
|  | canvas.drawPath(path, paint); | 
|  | } | 
|  |  | 
|  | class PathView : public SampleView { | 
|  | public: | 
|  | int fDStroke, fStroke, fMinStroke, fMaxStroke; | 
|  | SkPath fPath[6]; | 
|  | bool fShowHairline; | 
|  | bool fOnce; | 
|  |  | 
|  | PathView() { | 
|  | fOnce = false; | 
|  | } | 
|  |  | 
|  | void init() { | 
|  | if (fOnce) { | 
|  | return; | 
|  | } | 
|  | fOnce = true; | 
|  |  | 
|  | test_cubic(); | 
|  | test_cubic2(); | 
|  |  | 
|  | fShowHairline = false; | 
|  |  | 
|  | fDStroke = 1; | 
|  | fStroke = 10; | 
|  | fMinStroke = 10; | 
|  | fMaxStroke = 180; | 
|  |  | 
|  | const int V = 85; | 
|  |  | 
|  | fPath[0].moveTo(SkIntToScalar(40), SkIntToScalar(70)); | 
|  | fPath[0].lineTo(SkIntToScalar(70), SkIntToScalar(70) + SK_Scalar1/1); | 
|  | fPath[0].lineTo(SkIntToScalar(110), SkIntToScalar(70)); | 
|  |  | 
|  | fPath[1].moveTo(SkIntToScalar(40), SkIntToScalar(70)); | 
|  | fPath[1].lineTo(SkIntToScalar(70), SkIntToScalar(70) - SK_Scalar1/1); | 
|  | fPath[1].lineTo(SkIntToScalar(110), SkIntToScalar(70)); | 
|  |  | 
|  | fPath[2].moveTo(SkIntToScalar(V), SkIntToScalar(V)); | 
|  | fPath[2].lineTo(SkIntToScalar(50), SkIntToScalar(V)); | 
|  | fPath[2].lineTo(SkIntToScalar(50), SkIntToScalar(50)); | 
|  |  | 
|  | fPath[3].moveTo(SkIntToScalar(50), SkIntToScalar(50)); | 
|  | fPath[3].lineTo(SkIntToScalar(50), SkIntToScalar(V)); | 
|  | fPath[3].lineTo(SkIntToScalar(V), SkIntToScalar(V)); | 
|  |  | 
|  | fPath[4].moveTo(SkIntToScalar(50), SkIntToScalar(50)); | 
|  | fPath[4].lineTo(SkIntToScalar(50), SkIntToScalar(V)); | 
|  | fPath[4].lineTo(SkIntToScalar(52), SkIntToScalar(50)); | 
|  |  | 
|  | fPath[5].moveTo(SkIntToScalar(52), SkIntToScalar(50)); | 
|  | fPath[5].lineTo(SkIntToScalar(50), SkIntToScalar(V)); | 
|  | fPath[5].lineTo(SkIntToScalar(50), SkIntToScalar(50)); | 
|  |  | 
|  | this->setBGColor(0xFFDDDDDD); | 
|  | } | 
|  |  | 
|  | void nextStroke() { | 
|  | fStroke += fDStroke; | 
|  | if (fStroke > fMaxStroke || fStroke < fMinStroke) | 
|  | fDStroke = -fDStroke; | 
|  | } | 
|  |  | 
|  | protected: | 
|  | // overrides from SkEventSink | 
|  | virtual bool onQuery(SkEvent* evt) { | 
|  | 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(SkIntToScalar(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); | 
|  | } | 
|  |  | 
|  | virtual void onDrawContent(SkCanvas* canvas) { | 
|  | this->init(); | 
|  | canvas->translate(SkIntToScalar(50), SkIntToScalar(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(SkIntToScalar(200), 0); | 
|  | } | 
|  | canvas->restore(); | 
|  |  | 
|  | canvas->translate(0, SkIntToScalar(200)); | 
|  | } | 
|  |  | 
|  | this->nextStroke(); | 
|  | this->inval(NULL); | 
|  | } | 
|  |  | 
|  | virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y, unsigned modi) SK_OVERRIDE { | 
|  | fShowHairline = !fShowHairline; | 
|  | this->inval(NULL); | 
|  | return this->INHERITED::onFindClickHandler(x, y, modi); | 
|  | } | 
|  |  | 
|  | private: | 
|  | typedef SampleView INHERITED; | 
|  | }; | 
|  |  | 
|  | ////////////////////////////////////////////////////////////////////////////// | 
|  |  | 
|  | static SkView* MyFactory() { return new PathView; } | 
|  | static SkViewRegister reg(MyFactory); |