| OLD | NEW |
| (Empty) |
| 1 | |
| 2 /* | |
| 3 * Copyright 2012 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 #include "gm.h" | |
| 9 | |
| 10 #include "SkColorPriv.h" | |
| 11 #include "SkGeometry.h" | |
| 12 #include "SkShader.h" | |
| 13 | |
| 14 #define WIRE_FRAME_WIDTH 1.1f | |
| 15 | |
| 16 static void tesselate(const SkPath& src, SkPath* dst) { | |
| 17 SkPath::Iter iter(src, true); | |
| 18 SkPoint pts[4]; | |
| 19 SkPath::Verb verb; | |
| 20 while ((verb = iter.next(pts)) != SkPath::kDone_Verb) { | |
| 21 switch (verb) { | |
| 22 case SkPath::kMove_Verb: | |
| 23 dst->moveTo(pts[0]); | |
| 24 break; | |
| 25 case SkPath::kLine_Verb: | |
| 26 dst->lineTo(pts[1]); | |
| 27 break; | |
| 28 case SkPath::kQuad_Verb: { | |
| 29 SkPoint p; | |
| 30 for (int i = 1; i <= 8; ++i) { | |
| 31 SkEvalQuadAt(pts, i / 8.0f, &p, NULL); | |
| 32 dst->lineTo(p); | |
| 33 } | |
| 34 } break; | |
| 35 case SkPath::kCubic_Verb: { | |
| 36 SkPoint p; | |
| 37 for (int i = 1; i <= 8; ++i) { | |
| 38 SkEvalCubicAt(pts, i / 8.0f, &p, NULL, NULL); | |
| 39 dst->lineTo(p); | |
| 40 } | |
| 41 } break; | |
| 42 } | |
| 43 } | |
| 44 } | |
| 45 | |
| 46 static void setFade(SkPaint* paint, bool showGL) { | |
| 47 paint->setAlpha(showGL ? 0x66 : 0xFF); | |
| 48 } | |
| 49 | |
| 50 static void setGLFrame(SkPaint* paint) { | |
| 51 paint->setColor(0xFFFF0000); | |
| 52 paint->setStyle(SkPaint::kStroke_Style); | |
| 53 paint->setAntiAlias(true); | |
| 54 paint->setStrokeWidth(WIRE_FRAME_WIDTH); | |
| 55 } | |
| 56 | |
| 57 static void show_mesh(SkCanvas* canvas, const SkRect& r) { | |
| 58 SkPaint paint; | |
| 59 setGLFrame(&paint); | |
| 60 | |
| 61 canvas->drawRect(r, paint); | |
| 62 canvas->drawLine(r.fLeft, r.fTop, r.fRight, r.fBottom, paint); | |
| 63 } | |
| 64 | |
| 65 static void drawLine(SkCanvas* canvas, const SkPoint& p0, const SkPoint& p1, | |
| 66 const SkPaint& paint) { | |
| 67 canvas->drawLine(p0.fX, p0.fY, p1.fX, p1.fY, paint); | |
| 68 } | |
| 69 | |
| 70 static void show_mesh(SkCanvas* canvas, const SkPoint pts[], | |
| 71 const uint16_t indices[], int count) { | |
| 72 SkPaint paint; | |
| 73 setGLFrame(&paint); | |
| 74 | |
| 75 for (int i = 0; i < count - 2; ++i) { | |
| 76 drawLine(canvas, pts[indices[i]], pts[indices[i+1]], paint); | |
| 77 drawLine(canvas, pts[indices[i+1]], pts[indices[i+2]], paint); | |
| 78 drawLine(canvas, pts[indices[i+2]], pts[indices[i]], paint); | |
| 79 } | |
| 80 } | |
| 81 | |
| 82 static void show_glframe(SkCanvas* canvas, const SkPath& path) { | |
| 83 SkPaint paint; | |
| 84 setGLFrame(&paint); | |
| 85 canvas->drawPath(path, paint); | |
| 86 } | |
| 87 | |
| 88 static void show_mesh_between(SkCanvas* canvas, const SkPath& p0, const SkPath&
p1) { | |
| 89 SkPath d0, d1; | |
| 90 tesselate(p0, &d0); | |
| 91 tesselate(p1, &d1); | |
| 92 | |
| 93 SkPoint pts0[256*2], pts1[256]; | |
| 94 int count = d0.getPoints(pts0, SK_ARRAY_COUNT(pts0)); | |
| 95 int count1 = d1.getPoints(pts1, SK_ARRAY_COUNT(pts1)); | |
| 96 SkASSERT(count == count1); | |
| 97 memcpy(&pts0[count], pts1, count * sizeof(SkPoint)); | |
| 98 | |
| 99 uint16_t indices[256*6]; | |
| 100 uint16_t* ndx = indices; | |
| 101 for (int i = 0; i < count; ++i) { | |
| 102 *ndx++ = i; | |
| 103 *ndx++ = i + count; | |
| 104 } | |
| 105 *ndx++ = 0; | |
| 106 | |
| 107 show_mesh(canvas, pts0, indices, ndx - indices); | |
| 108 } | |
| 109 | |
| 110 static void show_fan(SkCanvas* canvas, const SkPath& path, SkScalar cx, SkScalar
cy) { | |
| 111 SkPaint paint; | |
| 112 setGLFrame(&paint); | |
| 113 | |
| 114 canvas->drawPath(path, paint); | |
| 115 | |
| 116 SkPoint pts[256]; | |
| 117 int count = path.getPoints(pts, SK_ARRAY_COUNT(pts)); | |
| 118 for (int i = 0; i < count; ++i) { | |
| 119 canvas->drawLine(pts[i].fX, pts[i].fY, cx, cy, paint); | |
| 120 } | |
| 121 } | |
| 122 | |
| 123 /////////////////////////////////////////////////////////////////////////////// | |
| 124 | |
| 125 typedef void (*DrawProc)(SkCanvas* canvas, bool showGL, int flags); | |
| 126 | |
| 127 static void draw_line(SkCanvas* canvas, bool showGL, int flags) { | |
| 128 SkPaint paint; | |
| 129 paint.setAntiAlias(true); | |
| 130 | |
| 131 if (showGL) { | |
| 132 setGLFrame(&paint); | |
| 133 } | |
| 134 canvas->drawLine(50, 50, 400, 100, paint); | |
| 135 paint.setColor(SK_ColorBLACK); | |
| 136 | |
| 137 canvas->rotate(40); | |
| 138 setFade(&paint, showGL); | |
| 139 paint.setStrokeWidth(40); | |
| 140 canvas->drawLine(100, 50, 450, 50, paint); | |
| 141 if (showGL) { | |
| 142 show_mesh(canvas, SkRect::MakeLTRB(100, 50-20, 450, 50+20)); | |
| 143 } | |
| 144 } | |
| 145 | |
| 146 static void draw_rect(SkCanvas* canvas, bool showGL, int flags) { | |
| 147 SkPaint paint; | |
| 148 paint.setAntiAlias(true); | |
| 149 | |
| 150 SkRect r = SkRect::MakeLTRB(50, 70, 250, 370); | |
| 151 | |
| 152 setFade(&paint, showGL); | |
| 153 canvas->drawRect(r, paint); | |
| 154 if (showGL) { | |
| 155 show_mesh(canvas, r); | |
| 156 } | |
| 157 | |
| 158 canvas->translate(320, 0); | |
| 159 | |
| 160 paint.setStyle(SkPaint::kStroke_Style); | |
| 161 paint.setStrokeWidth(25); | |
| 162 canvas->drawRect(r, paint); | |
| 163 if (showGL) { | |
| 164 SkScalar rad = paint.getStrokeWidth() / 2; | |
| 165 SkPoint pts[8]; | |
| 166 r.outset(rad, rad); | |
| 167 r.toQuad(&pts[0]); | |
| 168 r.inset(rad*2, rad*2); | |
| 169 r.toQuad(&pts[4]); | |
| 170 | |
| 171 const uint16_t indices[] = { | |
| 172 0, 4, 1, 5, 2, 6, 3, 7, 0, 4 | |
| 173 }; | |
| 174 show_mesh(canvas, pts, indices, SK_ARRAY_COUNT(indices)); | |
| 175 } | |
| 176 } | |
| 177 | |
| 178 static void draw_oval(SkCanvas* canvas, bool showGL, int flags) { | |
| 179 SkPaint paint; | |
| 180 paint.setAntiAlias(true); | |
| 181 | |
| 182 SkRect r = SkRect::MakeLTRB(50, 70, 250, 370); | |
| 183 | |
| 184 setFade(&paint, showGL); | |
| 185 canvas->drawOval(r, paint); | |
| 186 if (showGL) { | |
| 187 switch (flags) { | |
| 188 case 0: { | |
| 189 SkPath path; | |
| 190 path.addOval(r); | |
| 191 show_glframe(canvas, path); | |
| 192 } break; | |
| 193 case 1: | |
| 194 case 3: { | |
| 195 SkPath src, dst; | |
| 196 src.addOval(r); | |
| 197 tesselate(src, &dst); | |
| 198 show_fan(canvas, dst, r.centerX(), r.centerY()); | |
| 199 } break; | |
| 200 case 2: { | |
| 201 SkPaint p(paint); | |
| 202 show_mesh(canvas, r); | |
| 203 setGLFrame(&p); | |
| 204 paint.setStyle(SkPaint::kFill_Style); | |
| 205 canvas->drawCircle(r.centerX(), r.centerY(), 3, p); | |
| 206 } break; | |
| 207 } | |
| 208 } | |
| 209 | |
| 210 canvas->translate(320, 0); | |
| 211 | |
| 212 paint.setStyle(SkPaint::kStroke_Style); | |
| 213 paint.setStrokeWidth(25); | |
| 214 canvas->drawOval(r, paint); | |
| 215 if (showGL) { | |
| 216 switch (flags) { | |
| 217 case 0: { | |
| 218 SkPath path; | |
| 219 SkScalar rad = paint.getStrokeWidth() / 2; | |
| 220 r.outset(rad, rad); | |
| 221 path.addOval(r); | |
| 222 r.inset(rad*2, rad*2); | |
| 223 path.addOval(r); | |
| 224 show_glframe(canvas, path); | |
| 225 } break; | |
| 226 case 1: { | |
| 227 SkPath path0, path1; | |
| 228 SkScalar rad = paint.getStrokeWidth() / 2; | |
| 229 r.outset(rad, rad); | |
| 230 path0.addOval(r); | |
| 231 r.inset(rad*2, rad*2); | |
| 232 path1.addOval(r); | |
| 233 show_mesh_between(canvas, path0, path1); | |
| 234 } break; | |
| 235 case 2: { | |
| 236 SkPath path; | |
| 237 path.addOval(r); | |
| 238 show_glframe(canvas, path); | |
| 239 SkScalar rad = paint.getStrokeWidth() / 2; | |
| 240 r.outset(rad, rad); | |
| 241 show_mesh(canvas, r); | |
| 242 } break; | |
| 243 case 3: { | |
| 244 SkScalar rad = paint.getStrokeWidth() / 2; | |
| 245 r.outset(rad, rad); | |
| 246 SkPaint paint; | |
| 247 paint.setAlpha(0x33); | |
| 248 canvas->drawRect(r, paint); | |
| 249 show_mesh(canvas, r); | |
| 250 } break; | |
| 251 } | |
| 252 } | |
| 253 } | |
| 254 | |
| 255 #include "SkImageDecoder.h" | |
| 256 | |
| 257 static void draw_image(SkCanvas* canvas, bool showGL, int flags) { | |
| 258 SkPaint paint; | |
| 259 paint.setAntiAlias(true); | |
| 260 paint.setFilterBitmap(true); | |
| 261 setFade(&paint, showGL); | |
| 262 | |
| 263 static SkBitmap* gBM; | |
| 264 if (NULL == gBM) { | |
| 265 gBM = new SkBitmap; | |
| 266 SkImageDecoder::DecodeFile("/skimages/startrek.png", gBM); | |
| 267 } | |
| 268 SkRect r = SkRect::MakeWH(gBM->width(), gBM->height()); | |
| 269 | |
| 270 canvas->save(); | |
| 271 canvas->translate(30, 30); | |
| 272 canvas->scale(0.8f, 0.8f); | |
| 273 canvas->drawBitmap(*gBM, 0, 0, &paint); | |
| 274 if (showGL) { | |
| 275 show_mesh(canvas, r); | |
| 276 } | |
| 277 canvas->restore(); | |
| 278 | |
| 279 canvas->translate(210, 290); | |
| 280 canvas->rotate(-35); | |
| 281 canvas->drawBitmap(*gBM, 0, 0, &paint); | |
| 282 if (showGL) { | |
| 283 show_mesh(canvas, r); | |
| 284 } | |
| 285 } | |
| 286 | |
| 287 static void draw_text(SkCanvas* canvas, bool showGL, int flags) { | |
| 288 SkPaint paint; | |
| 289 paint.setAntiAlias(true); | |
| 290 paint.setLCDRenderText(true); | |
| 291 const char text[] = "Graphics at Google"; | |
| 292 size_t len = strlen(text); | |
| 293 setFade(&paint, showGL); | |
| 294 | |
| 295 canvas->translate(40, 50); | |
| 296 for (int i = 0; i < 10; ++i) { | |
| 297 paint.setTextSize(12 + i * 3); | |
| 298 canvas->drawText(text, len, 0, 0, paint); | |
| 299 if (showGL) { | |
| 300 SkRect bounds[256]; | |
| 301 SkScalar widths[256]; | |
| 302 int count = paint.getTextWidths(text, len, widths, bounds); | |
| 303 SkScalar adv = 0; | |
| 304 for (int j = 0; j < count; ++j) { | |
| 305 bounds[j].offset(adv, 0); | |
| 306 show_mesh(canvas, bounds[j]); | |
| 307 adv += widths[j]; | |
| 308 } | |
| 309 } | |
| 310 canvas->translate(0, paint.getTextSize() * 3 / 2); | |
| 311 } | |
| 312 } | |
| 313 | |
| 314 static const struct { | |
| 315 DrawProc fProc; | |
| 316 const char* fName; | |
| 317 } gRec[] = { | |
| 318 { draw_line, "Lines" }, | |
| 319 { draw_rect, "Rects" }, | |
| 320 { draw_oval, "Ovals" }, | |
| 321 { draw_image, "Images" }, | |
| 322 { draw_text, "Text" }, | |
| 323 }; | |
| 324 | |
| 325 class TalkGM : public skiagm::GM { | |
| 326 DrawProc fProc; | |
| 327 SkString fName; | |
| 328 bool fShowGL; | |
| 329 int fFlags; | |
| 330 | |
| 331 public: | |
| 332 TalkGM(int index, bool showGL, int flags = 0) { | |
| 333 fProc = gRec[index].fProc; | |
| 334 fName.set(gRec[index].fName); | |
| 335 if (showGL) { | |
| 336 fName.append("-gl"); | |
| 337 } | |
| 338 fShowGL = showGL; | |
| 339 fFlags = flags; | |
| 340 } | |
| 341 | |
| 342 protected: | |
| 343 virtual SkString onShortName() { | |
| 344 return fName; | |
| 345 } | |
| 346 | |
| 347 virtual SkISize onISize() { | |
| 348 return SkISize::Make(640, 480); | |
| 349 } | |
| 350 | |
| 351 virtual void onDraw(SkCanvas* canvas) { | |
| 352 SkISize size = canvas->getDeviceSize(); | |
| 353 SkRect dst = SkRect::MakeWH(size.width(), size.height()); | |
| 354 SkRect src = SkRect::MakeWH(640, 480); | |
| 355 SkMatrix matrix; | |
| 356 matrix.setRectToRect(src, dst, SkMatrix::kCenter_ScaleToFit); | |
| 357 | |
| 358 canvas->concat(matrix); | |
| 359 fProc(canvas, fShowGL, fFlags); | |
| 360 } | |
| 361 | |
| 362 private: | |
| 363 typedef skiagm::GM INHERITED; | |
| 364 }; | |
| 365 | |
| 366 ////////////////////////////////////////////////////////////////////////////// | |
| 367 | |
| 368 #define GM_CONCAT(X,Y) GM_CONCAT_IMPL(X,Y) | |
| 369 #define GM_CONCAT_IMPL(X,Y) X##Y | |
| 370 | |
| 371 #define FACTORY_NAME GM_CONCAT(Factory, __LINE__) | |
| 372 #define REGISTRY_NAME GM_CONCAT(gReg, __LINE__) | |
| 373 | |
| 374 #define ADD_GM(Class, args) \ | |
| 375 static skiagm::GM* FACTORY_NAME(void*) { return new Class args; } \ | |
| 376 static skiagm::GMRegistry REGISTRY_NAME(FACTORY_NAME); | |
| 377 | |
| 378 ADD_GM(TalkGM, (0, false)) | |
| 379 ADD_GM(TalkGM, (0, true)) | |
| 380 ADD_GM(TalkGM, (1, false)) | |
| 381 ADD_GM(TalkGM, (1, true)) | |
| 382 ADD_GM(TalkGM, (2, false)) | |
| 383 ADD_GM(TalkGM, (2, true)) | |
| 384 ADD_GM(TalkGM, (2, true, 1)) | |
| 385 ADD_GM(TalkGM, (2, true, 2)) | |
| 386 ADD_GM(TalkGM, (2, true, 3)) | |
| 387 ADD_GM(TalkGM, (3, false)) | |
| 388 ADD_GM(TalkGM, (3, true)) | |
| 389 ADD_GM(TalkGM, (4, false)) | |
| 390 ADD_GM(TalkGM, (4, true)) | |
| 391 | |
| 392 //static GM* MyFactory(void*) { return new TalkGM(0, false); } | |
| 393 //static GMRegistry reg(MyFactory); | |
| OLD | NEW |