SamplePath.cpp revision f7834221ac5342a3446b9b299398ea3ff7976946
1 2/* 3 * Copyright 2011 Google Inc. 4 * 5 * Use of this source code is governed by a BSD-style license that can be 6 * found in the LICENSE file. 7 */ 8 9#include "SampleCode.h" 10#include "SkAnimTimer.h" 11#include "SkView.h" 12#include "SkCanvas.h" 13#include "SkGradientShader.h" 14#include "SkGraphics.h" 15#include "SkImageDecoder.h" 16#include "SkPath.h" 17#include "SkRegion.h" 18#include "SkShader.h" 19#include "SkUtils.h" 20#include "SkXfermode.h" 21#include "SkColorPriv.h" 22#include "SkColorFilter.h" 23#include "SkParsePath.h" 24#include "SkTime.h" 25#include "SkTypeface.h" 26 27#include "SkGeometry.h" 28 29// http://code.google.com/p/skia/issues/detail?id=32 30static void test_cubic() { 31 SkPoint src[4] = { 32 { 556.25000f, 523.03003f }, 33 { 556.23999f, 522.96002f }, 34 { 556.21997f, 522.89001f }, 35 { 556.21997f, 522.82001f } 36 }; 37 SkPoint dst[11]; 38 dst[10].set(42, -42); // one past the end, that we don't clobber these 39 SkScalar tval[] = { 0.33333334f, 0.99999994f }; 40 41 SkChopCubicAt(src, dst, tval, 2); 42 43#if 0 44 for (int i = 0; i < 11; i++) { 45 SkDebugf("--- %d [%g %g]\n", i, dst[i].fX, dst[i].fY); 46 } 47#endif 48} 49 50static void test_cubic2() { 51 const char* str = "M2242 -590088L-377758 9.94099e+07L-377758 9.94099e+07L2242 -590088Z"; 52 SkPath path; 53 SkParsePath::FromSVGString(str, &path); 54 55 { 56#ifdef SK_BUILD_FOR_WIN 57 // windows doesn't have strtof 58 float x = (float)strtod("9.94099e+07", NULL); 59#else 60 float x = strtof("9.94099e+07", NULL); 61#endif 62 int ix = (int)x; 63 int fx = (int)(x * 65536); 64 int ffx = SkScalarToFixed(x); 65 SkDebugf("%g %x %x %x\n", x, ix, fx, ffx); 66 67 SkRect r = path.getBounds(); 68 SkIRect ir; 69 r.round(&ir); 70 SkDebugf("[%g %g %g %g] [%x %x %x %x]\n", 71 SkScalarToDouble(r.fLeft), SkScalarToDouble(r.fTop), 72 SkScalarToDouble(r.fRight), SkScalarToDouble(r.fBottom), 73 ir.fLeft, ir.fTop, ir.fRight, ir.fBottom); 74 } 75 76 SkBitmap bitmap; 77 bitmap.allocN32Pixels(300, 200); 78 79 SkCanvas canvas(bitmap); 80 SkPaint paint; 81 paint.setAntiAlias(true); 82 canvas.drawPath(path, paint); 83} 84 85class PathView : public SampleView { 86 SkScalar fPrevSecs; 87public: 88 SkScalar fDStroke, fStroke, fMinStroke, fMaxStroke; 89 SkPath fPath[6]; 90 bool fShowHairline; 91 bool fOnce; 92 93 PathView() { 94 fPrevSecs = 0; 95 fOnce = false; 96 } 97 98 void init() { 99 if (fOnce) { 100 return; 101 } 102 fOnce = true; 103 104 test_cubic(); 105 test_cubic2(); 106 107 fShowHairline = false; 108 109 fDStroke = 1; 110 fStroke = 10; 111 fMinStroke = 10; 112 fMaxStroke = 180; 113 114 const SkScalar V = 85; 115 116 fPath[0].moveTo(40, 70); 117 fPath[0].lineTo(70, 70 + SK_ScalarHalf); 118 fPath[0].lineTo(110, 70); 119 120 fPath[1].moveTo(40, 70); 121 fPath[1].lineTo(70, 70 - SK_ScalarHalf); 122 fPath[1].lineTo(110, 70); 123 124 fPath[2].moveTo(V, V); 125 fPath[2].lineTo(50, V); 126 fPath[2].lineTo(50, 50); 127 128 fPath[3].moveTo(50, 50); 129 fPath[3].lineTo(50, V); 130 fPath[3].lineTo(V, V); 131 132 fPath[4].moveTo(50, 50); 133 fPath[4].lineTo(50, V); 134 fPath[4].lineTo(52, 50); 135 136 fPath[5].moveTo(52, 50); 137 fPath[5].lineTo(50, V); 138 fPath[5].lineTo(50, 50); 139 140 this->setBGColor(0xFFDDDDDD); 141 } 142 143protected: 144 // overrides from SkEventSink 145 bool onQuery(SkEvent* evt) override { 146 if (SampleCode::TitleQ(*evt)) { 147 SampleCode::TitleR(evt, "Paths"); 148 return true; 149 } 150 return this->INHERITED::onQuery(evt); 151 } 152 153 void drawPath(SkCanvas* canvas, const SkPath& path, SkPaint::Join j) { 154 SkPaint paint; 155 156 paint.setAntiAlias(true); 157 paint.setStyle(SkPaint::kStroke_Style); 158 paint.setStrokeJoin(j); 159 paint.setStrokeWidth(fStroke); 160 161 if (fShowHairline) { 162 SkPath fill; 163 164 paint.getFillPath(path, &fill); 165 paint.setStrokeWidth(0); 166 canvas->drawPath(fill, paint); 167 } else { 168 canvas->drawPath(path, paint); 169 } 170 171 paint.setColor(SK_ColorRED); 172 paint.setStrokeWidth(0); 173 canvas->drawPath(path, paint); 174 } 175 176 void onDrawContent(SkCanvas* canvas) override { 177 this->init(); 178 canvas->translate(50, 50); 179 180 static const SkPaint::Join gJoins[] = { 181 SkPaint::kBevel_Join, 182 SkPaint::kMiter_Join, 183 SkPaint::kRound_Join 184 }; 185 186 for (size_t i = 0; i < SK_ARRAY_COUNT(gJoins); i++) { 187 canvas->save(); 188 for (size_t j = 0; j < SK_ARRAY_COUNT(fPath); j++) { 189 this->drawPath(canvas, fPath[j], gJoins[i]); 190 canvas->translate(200, 0); 191 } 192 canvas->restore(); 193 194 canvas->translate(0, 200); 195 } 196 } 197 198 bool onAnimate(const SkAnimTimer& timer) override { 199 SkScalar currSecs = timer.scaled(100); 200 SkScalar delta = currSecs - fPrevSecs; 201 fPrevSecs = currSecs; 202 203 fStroke += fDStroke * delta; 204 if (fStroke > fMaxStroke || fStroke < fMinStroke) { 205 fDStroke = -fDStroke; 206 } 207 return true; 208 } 209 210 SkView::Click* onFindClickHandler(SkScalar x, SkScalar y, unsigned modi) override { 211 fShowHairline = !fShowHairline; 212 this->inval(NULL); 213 return this->INHERITED::onFindClickHandler(x, y, modi); 214 } 215 216private: 217 typedef SampleView INHERITED; 218}; 219DEF_SAMPLE( return new PathView; ) 220 221////////////////////////////////////////////////////////////////////////////// 222 223#include "SkArcToPathEffect.h" 224#include "SkCornerPathEffect.h" 225#include "SkRandom.h" 226 227class ArcToView : public SampleView { 228 bool fDoFrame, fDoArcTo, fDoCorner, fDoConic; 229 SkPaint fPtsPaint, fArcToPaint, fSkeletonPaint, fCornerPaint; 230public: 231 enum { 232 N = 4 233 }; 234 SkPoint fPts[N]; 235 236 ArcToView() 237 : fDoFrame(false), fDoArcTo(false), fDoCorner(false), fDoConic(false) 238 { 239 SkRandom rand; 240 for (int i = 0; i < N; ++i) { 241 fPts[i].fX = 20 + rand.nextUScalar1() * 640; 242 fPts[i].fY = 20 + rand.nextUScalar1() * 480; 243 } 244 245 const SkScalar rad = 50; 246 247 fPtsPaint.setAntiAlias(true); 248 fPtsPaint.setStrokeWidth(15); 249 fPtsPaint.setStrokeCap(SkPaint::kRound_Cap); 250 251 fArcToPaint.setAntiAlias(true); 252 fArcToPaint.setStyle(SkPaint::kStroke_Style); 253 fArcToPaint.setStrokeWidth(9); 254 fArcToPaint.setColor(0x800000FF); 255 fArcToPaint.setPathEffect(SkArcToPathEffect::Create(rad))->unref(); 256 257 fCornerPaint.setAntiAlias(true); 258 fCornerPaint.setStyle(SkPaint::kStroke_Style); 259 fCornerPaint.setStrokeWidth(13); 260 fCornerPaint.setColor(SK_ColorGREEN); 261 fCornerPaint.setPathEffect(SkCornerPathEffect::Create(rad*2))->unref(); 262 263 fSkeletonPaint.setAntiAlias(true); 264 fSkeletonPaint.setStyle(SkPaint::kStroke_Style); 265 fSkeletonPaint.setColor(SK_ColorRED); 266 } 267 268 void toggle(bool& value) { 269 value = !value; 270 this->inval(NULL); 271 } 272 273protected: 274 // overrides from SkEventSink 275 bool onQuery(SkEvent* evt) override { 276 if (SampleCode::TitleQ(*evt)) { 277 SampleCode::TitleR(evt, "ArcTo"); 278 return true; 279 } 280 SkUnichar uni; 281 if (SampleCode::CharQ(*evt, &uni)) { 282 switch (uni) { 283 case '1': this->toggle(fDoFrame); return true; 284 case '2': this->toggle(fDoArcTo); return true; 285 case '3': this->toggle(fDoCorner); return true; 286 case '4': this->toggle(fDoConic); return true; 287 default: break; 288 } 289 } 290 return this->INHERITED::onQuery(evt); 291 } 292 293 void makePath(SkPath* path) { 294 path->moveTo(fPts[0]); 295 for (int i = 1; i < N; ++i) { 296 path->lineTo(fPts[i]); 297 } 298 if (!fDoFrame) { 299 path->close(); 300 } 301 } 302 303 void onDrawContent(SkCanvas* canvas) override { 304 canvas->drawPoints(SkCanvas::kPoints_PointMode, N, fPts, fPtsPaint); 305 306 SkPath path; 307 this->makePath(&path); 308 309 if (fDoCorner) { 310 canvas->drawPath(path, fCornerPaint); 311 } 312 if (fDoArcTo) { 313 canvas->drawPath(path, fArcToPaint); 314 } 315 316 canvas->drawPath(path, fSkeletonPaint); 317 } 318 319 bool onClick(Click* click) override { 320 int32_t index; 321 if (click->fMeta.findS32("index", &index)) { 322 SkASSERT((unsigned)index < N); 323 fPts[index] = click->fCurr; 324 this->inval(NULL); 325 return true; 326 } 327 return false; 328 } 329 330 SkView::Click* onFindClickHandler(SkScalar x, SkScalar y, unsigned modi) override { 331 const SkScalar tol = 4; 332 const SkRect r = SkRect::MakeXYWH(x - tol, y - tol, tol * 2, tol * 2); 333 for (int i = 0; i < N; ++i) { 334 if (r.intersects(SkRect::MakeXYWH(fPts[i].fX, fPts[i].fY, 1, 1))) { 335 Click* click = new Click(this); 336 click->fMeta.setS32("index", i); 337 return click; 338 } 339 } 340 return this->INHERITED::onFindClickHandler(x, y, modi); 341 } 342 343private: 344 typedef SampleView INHERITED; 345}; 346DEF_SAMPLE( return new ArcToView; ) 347 348