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 #include "SampleCode.h"
9 #include "SkView.h"
10 #include "SkCanvas.h"
11 #include "SkDevice.h"
12 #include "SkPaint.h"
13
14 #define BG_COLOR 0xFFDDDDDD
15
16 typedef void (*SlideProc)(SkCanvas*);
17
18 ///////////////////////////////////////////////////////////////////////////////
19
20 #include "Sk1DPathEffect.h"
21 #include "Sk2DPathEffect.h"
22 #include "SkCornerPathEffect.h"
23 #include "SkDashPathEffect.h"
24 #include "SkDiscretePathEffect.h"
25
compose_pe(SkPaint * paint)26 static void compose_pe(SkPaint* paint) {
27 SkPathEffect* pe = paint->getPathEffect();
28 SkPathEffect* corner = new SkCornerPathEffect(25);
29 SkPathEffect* compose;
30 if (pe) {
31 compose = new SkComposePathEffect(pe, corner);
32 corner->unref();
33 } else {
34 compose = corner;
35 }
36 paint->setPathEffect(compose)->unref();
37 }
38
hair_pe(SkPaint * paint)39 static void hair_pe(SkPaint* paint) {
40 paint->setStrokeWidth(0);
41 }
42
hair2_pe(SkPaint * paint)43 static void hair2_pe(SkPaint* paint) {
44 paint->setStrokeWidth(0);
45 compose_pe(paint);
46 }
47
stroke_pe(SkPaint * paint)48 static void stroke_pe(SkPaint* paint) {
49 paint->setStrokeWidth(12);
50 compose_pe(paint);
51 }
52
dash_pe(SkPaint * paint)53 static void dash_pe(SkPaint* paint) {
54 SkScalar inter[] = { 20, 10, 10, 10 };
55 paint->setStrokeWidth(12);
56 paint->setPathEffect(new SkDashPathEffect(inter, SK_ARRAY_COUNT(inter),
57 0))->unref();
58 compose_pe(paint);
59 }
60
61 static const int gXY[] = {
62 4, 0, 0, -4, 8, -4, 12, 0, 8, 4, 0, 4
63 };
64
scale(SkPath * path,SkScalar scale)65 static void scale(SkPath* path, SkScalar scale) {
66 SkMatrix m;
67 m.setScale(scale, scale);
68 path->transform(m);
69 }
70
one_d_pe(SkPaint * paint)71 static void one_d_pe(SkPaint* paint) {
72 SkPath path;
73 path.moveTo(SkIntToScalar(gXY[0]), SkIntToScalar(gXY[1]));
74 for (unsigned i = 2; i < SK_ARRAY_COUNT(gXY); i += 2)
75 path.lineTo(SkIntToScalar(gXY[i]), SkIntToScalar(gXY[i+1]));
76 path.close();
77 path.offset(SkIntToScalar(-6), 0);
78 scale(&path, 1.5);
79
80 paint->setPathEffect(new SkPath1DPathEffect(path, SkIntToScalar(21), 0,
81 SkPath1DPathEffect::kRotate_Style))->unref();
82 compose_pe(paint);
83 }
84
85 typedef void (*PE_Proc)(SkPaint*);
86 static const PE_Proc gPE[] = { hair_pe, hair2_pe, stroke_pe, dash_pe, one_d_pe };
87
fill_pe(SkPaint * paint)88 static void fill_pe(SkPaint* paint) {
89 paint->setStyle(SkPaint::kFill_Style);
90 paint->setPathEffect(NULL);
91 }
92
discrete_pe(SkPaint * paint)93 static void discrete_pe(SkPaint* paint) {
94 paint->setPathEffect(new SkDiscretePathEffect(10, 4))->unref();
95 }
96
MakeTileEffect()97 static SkPathEffect* MakeTileEffect() {
98 SkMatrix m;
99 m.setScale(SkIntToScalar(12), SkIntToScalar(12));
100
101 SkPath path;
102 path.addCircle(0, 0, SkIntToScalar(5));
103
104 return new SkPath2DPathEffect(m, path);
105 }
106
tile_pe(SkPaint * paint)107 static void tile_pe(SkPaint* paint) {
108 paint->setPathEffect(MakeTileEffect())->unref();
109 }
110
111 static const PE_Proc gPE2[] = { fill_pe, discrete_pe, tile_pe };
112
patheffect_slide(SkCanvas * canvas)113 static void patheffect_slide(SkCanvas* canvas) {
114 SkPaint paint;
115 paint.setAntiAlias(true);
116 paint.setStyle(SkPaint::kStroke_Style);
117
118 SkPath path;
119 path.moveTo(20, 20);
120 path.lineTo(70, 120);
121 path.lineTo(120, 30);
122 path.lineTo(170, 80);
123 path.lineTo(240, 50);
124
125 size_t i;
126 canvas->save();
127 for (i = 0; i < SK_ARRAY_COUNT(gPE); i++) {
128 gPE[i](&paint);
129 canvas->drawPath(path, paint);
130 canvas->translate(0, 75);
131 }
132 canvas->restore();
133
134 path.reset();
135 SkRect r = { 0, 0, 250, 120 };
136 path.addOval(r, SkPath::kCW_Direction);
137 r.inset(50, 50);
138 path.addRect(r, SkPath::kCCW_Direction);
139
140 canvas->translate(320, 20);
141 for (i = 0; i < SK_ARRAY_COUNT(gPE2); i++) {
142 gPE2[i](&paint);
143 canvas->drawPath(path, paint);
144 canvas->translate(0, 160);
145 }
146 }
147
148 ///////////////////////////////////////////////////////////////////////////////
149
150 #include "SkGradientShader.h"
151
152 struct GradData {
153 int fCount;
154 const SkColor* fColors;
155 const SkScalar* fPos;
156 };
157
158 static const SkColor gColors[] = {
159 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK
160 };
161 static const SkScalar gPos0[] = { 0, SK_Scalar1 };
162 static const SkScalar gPos1[] = { SK_Scalar1/4, SK_Scalar1*3/4 };
163 static const SkScalar gPos2[] = {
164 0, SK_Scalar1/8, SK_Scalar1/2, SK_Scalar1*7/8, SK_Scalar1
165 };
166
167 static const GradData gGradData[] = {
168 { 2, gColors, NULL },
169 { 2, gColors, gPos0 },
170 { 2, gColors, gPos1 },
171 { 5, gColors, NULL },
172 { 5, gColors, gPos2 }
173 };
174
MakeLinear(const SkPoint pts[2],const GradData & data,SkShader::TileMode tm,SkUnitMapper * mapper)175 static SkShader* MakeLinear(const SkPoint pts[2], const GradData& data,
176 SkShader::TileMode tm, SkUnitMapper* mapper) {
177 return SkGradientShader::CreateLinear(pts, data.fColors, data.fPos,
178 data.fCount, tm, mapper);
179 }
180
MakeRadial(const SkPoint pts[2],const GradData & data,SkShader::TileMode tm,SkUnitMapper * mapper)181 static SkShader* MakeRadial(const SkPoint pts[2], const GradData& data,
182 SkShader::TileMode tm, SkUnitMapper* mapper) {
183 SkPoint center;
184 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
185 SkScalarAve(pts[0].fY, pts[1].fY));
186 return SkGradientShader::CreateRadial(center, center.fX, data.fColors,
187 data.fPos, data.fCount, tm, mapper);
188 }
189
MakeSweep(const SkPoint pts[2],const GradData & data,SkShader::TileMode tm,SkUnitMapper * mapper)190 static SkShader* MakeSweep(const SkPoint pts[2], const GradData& data,
191 SkShader::TileMode tm, SkUnitMapper* mapper) {
192 SkPoint center;
193 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
194 SkScalarAve(pts[0].fY, pts[1].fY));
195 return SkGradientShader::CreateSweep(center.fX, center.fY, data.fColors,
196 data.fPos, data.fCount, mapper);
197 }
198
Make2Radial(const SkPoint pts[2],const GradData & data,SkShader::TileMode tm,SkUnitMapper * mapper)199 static SkShader* Make2Radial(const SkPoint pts[2], const GradData& data,
200 SkShader::TileMode tm, SkUnitMapper* mapper) {
201 SkPoint center0, center1;
202 center0.set(SkScalarAve(pts[0].fX, pts[1].fX),
203 SkScalarAve(pts[0].fY, pts[1].fY));
204 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5),
205 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4));
206 return SkGradientShader::CreateTwoPointRadial(
207 center1, (pts[1].fX - pts[0].fX) / 7,
208 center0, (pts[1].fX - pts[0].fX) / 2,
209 data.fColors, data.fPos, data.fCount, tm, mapper);
210 }
211
212 typedef SkShader* (*GradMaker)(const SkPoint pts[2], const GradData& data,
213 SkShader::TileMode tm, SkUnitMapper* mapper);
214 static const GradMaker gGradMakers[] = {
215 MakeLinear, MakeRadial, MakeSweep, Make2Radial
216 };
217
gradient_slide(SkCanvas * canvas)218 static void gradient_slide(SkCanvas* canvas) {
219 SkPoint pts[2] = {
220 { 0, 0 },
221 { SkIntToScalar(100), SkIntToScalar(100) }
222 };
223 SkShader::TileMode tm = SkShader::kClamp_TileMode;
224 SkRect r = { 0, 0, SkIntToScalar(100), SkIntToScalar(100) };
225 SkPaint paint;
226 paint.setAntiAlias(true);
227 paint.setDither(true);
228
229 canvas->translate(SkIntToScalar(20), SkIntToScalar(10));
230 for (size_t i = 0; i < SK_ARRAY_COUNT(gGradData); i++) {
231 canvas->save();
232 for (size_t j = 0; j < SK_ARRAY_COUNT(gGradMakers); j++) {
233 SkShader* shader = gGradMakers[j](pts, gGradData[i], tm, NULL);
234 paint.setShader(shader);
235 canvas->drawRect(r, paint);
236 shader->unref();
237 canvas->translate(0, SkIntToScalar(120));
238 }
239 canvas->restore();
240 canvas->translate(SkIntToScalar(120), 0);
241 }
242 }
243
244 ///////////////////////////////////////////////////////////////////////////////
245
246 #include "SkPathMeasure.h"
247
getpathlen(const SkPath & path)248 static SkScalar getpathlen(const SkPath& path) {
249 SkPathMeasure meas(path, false);
250 return meas.getLength();
251 }
252
textonpath_slide(SkCanvas * canvas)253 static void textonpath_slide(SkCanvas* canvas) {
254 const char* text = "Displacement";
255 size_t len =strlen(text);
256 SkPath path;
257 path.moveTo(100, 300);
258 path.quadTo(300, 100, 500, 300);
259 path.offset(0, -100);
260
261 SkPaint paint;
262 paint.setAntiAlias(true);
263 paint.setTextSize(40);
264
265 paint.setStyle(SkPaint::kStroke_Style);
266 canvas->drawPath(path, paint);
267 paint.setStyle(SkPaint::kFill_Style);
268
269 SkScalar x = 50;
270 paint.setColor(0xFF008800);
271 canvas->drawTextOnPathHV(text, len, path,
272 x, paint.getTextSize()*2/3, paint);
273 paint.setColor(SK_ColorRED);
274 canvas->drawTextOnPathHV(text, len, path,
275 x + 60, 0, paint);
276 paint.setColor(SK_ColorBLUE);
277 canvas->drawTextOnPathHV(text, len, path,
278 x + 120, -paint.getTextSize()*2/3, paint);
279
280 path.offset(0, 200);
281 paint.setTextAlign(SkPaint::kRight_Align);
282
283 text = "Matrices";
284 len = strlen(text);
285 SkScalar pathLen = getpathlen(path);
286 SkMatrix matrix;
287
288 paint.setColor(SK_ColorBLACK);
289 paint.setStyle(SkPaint::kStroke_Style);
290 canvas->drawPath(path, paint);
291 paint.setStyle(SkPaint::kFill_Style);
292
293 paint.setTextSize(50);
294 canvas->drawTextOnPath(text, len, path, NULL, paint);
295
296 paint.setColor(SK_ColorRED);
297 matrix.setScale(-SK_Scalar1, SK_Scalar1);
298 matrix.postTranslate(pathLen, 0);
299 canvas->drawTextOnPath(text, len, path, &matrix, paint);
300
301 paint.setColor(SK_ColorBLUE);
302 matrix.setScale(SK_Scalar1, -SK_Scalar1);
303 canvas->drawTextOnPath(text, len, path, &matrix, paint);
304
305 paint.setColor(0xFF008800);
306 matrix.setScale(-SK_Scalar1, -SK_Scalar1);
307 matrix.postTranslate(pathLen, 0);
308 canvas->drawTextOnPath(text, len, path, &matrix, paint);
309 }
310
311 ///////////////////////////////////////////////////////////////////////////////
312
313 #include "SkImageDecoder.h"
314 #include "SkOSFile.h"
315 #include "SkRandom.h"
316 #include "SkStream.h"
317
make_shader0(SkIPoint * size)318 static SkShader* make_shader0(SkIPoint* size) {
319 SkBitmap bm;
320
321 SkImageDecoder::DecodeFile("/skimages/logo.gif", &bm);
322 size->set(bm.width(), bm.height());
323 return SkShader::CreateBitmapShader(bm, SkShader::kClamp_TileMode,
324 SkShader::kClamp_TileMode);
325 }
326
make_shader1(const SkIPoint & size)327 static SkShader* make_shader1(const SkIPoint& size) {
328 SkPoint pts[] = { { 0, 0 },
329 { SkIntToScalar(size.fX), SkIntToScalar(size.fY) } };
330 SkColor colors[] = { SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorRED };
331 return SkGradientShader::CreateLinear(pts, colors, NULL,
332 SK_ARRAY_COUNT(colors), SkShader::kMirror_TileMode, NULL);
333 }
334
335 class Rec {
336 public:
337 SkCanvas::VertexMode fMode;
338 int fCount;
339 SkPoint* fVerts;
340 SkPoint* fTexs;
341
Rec()342 Rec() : fCount(0), fVerts(NULL), fTexs(NULL) {}
~Rec()343 ~Rec() { delete[] fVerts; delete[] fTexs; }
344 };
345
make_tris(Rec * rec)346 void make_tris(Rec* rec) {
347 int n = 10;
348 SkRandom rand;
349
350 rec->fMode = SkCanvas::kTriangles_VertexMode;
351 rec->fCount = n * 3;
352 rec->fVerts = new SkPoint[rec->fCount];
353
354 for (int i = 0; i < n; i++) {
355 SkPoint* v = &rec->fVerts[i*3];
356 for (int j = 0; j < 3; j++) {
357 v[j].set(rand.nextUScalar1() * 250, rand.nextUScalar1() * 250);
358 }
359 }
360 }
361
make_fan(Rec * rec,int texWidth,int texHeight)362 void make_fan(Rec* rec, int texWidth, int texHeight) {
363 const SkScalar tx = SkIntToScalar(texWidth);
364 const SkScalar ty = SkIntToScalar(texHeight);
365 const int n = 24;
366
367 rec->fMode = SkCanvas::kTriangleFan_VertexMode;
368 rec->fCount = n + 2;
369 rec->fVerts = new SkPoint[rec->fCount];
370 rec->fTexs = new SkPoint[rec->fCount];
371
372 SkPoint* v = rec->fVerts;
373 SkPoint* t = rec->fTexs;
374
375 v[0].set(0, 0);
376 t[0].set(0, 0);
377 for (int i = 0; i < n; i++) {
378 SkScalar cos;
379 SkScalar sin = SkScalarSinCos(SK_ScalarPI * 2 * i / n, &cos);
380 v[i+1].set(cos, sin);
381 t[i+1].set(i*tx/n, ty);
382 }
383 v[n+1] = v[1];
384 t[n+1].set(tx, ty);
385
386 SkMatrix m;
387 m.setScale(SkIntToScalar(100), SkIntToScalar(100));
388 m.postTranslate(SkIntToScalar(110), SkIntToScalar(110));
389 m.mapPoints(v, rec->fCount);
390 }
391
make_strip(Rec * rec,int texWidth,int texHeight)392 void make_strip(Rec* rec, int texWidth, int texHeight) {
393 const SkScalar tx = SkIntToScalar(texWidth);
394 const SkScalar ty = SkIntToScalar(texHeight);
395 const int n = 24;
396
397 rec->fMode = SkCanvas::kTriangleStrip_VertexMode;
398 rec->fCount = 2 * (n + 1);
399 rec->fVerts = new SkPoint[rec->fCount];
400 rec->fTexs = new SkPoint[rec->fCount];
401
402 SkPoint* v = rec->fVerts;
403 SkPoint* t = rec->fTexs;
404
405 for (int i = 0; i < n; i++) {
406 SkScalar cos;
407 SkScalar sin = SkScalarSinCos(SK_ScalarPI * 2 * i / n, &cos);
408 v[i*2 + 0].set(cos/2, sin/2);
409 v[i*2 + 1].set(cos, sin);
410
411 t[i*2 + 0].set(tx * i / n, ty);
412 t[i*2 + 1].set(tx * i / n, 0);
413 }
414 v[2*n + 0] = v[0];
415 v[2*n + 1] = v[1];
416
417 t[2*n + 0].set(tx, ty);
418 t[2*n + 1].set(tx, 0);
419
420 SkMatrix m;
421 m.setScale(SkIntToScalar(100), SkIntToScalar(100));
422 m.postTranslate(SkIntToScalar(110), SkIntToScalar(110));
423 m.mapPoints(v, rec->fCount);
424 }
425
mesh_slide(SkCanvas * canvas)426 static void mesh_slide(SkCanvas* canvas) {
427 Rec fRecs[3];
428 SkIPoint size;
429
430 SkShader* fShader0 = make_shader0(&size);
431 SkShader* fShader1 = make_shader1(size);
432
433 SkAutoUnref aur0(fShader0);
434 SkAutoUnref aur1(fShader1);
435
436 make_strip(&fRecs[0], size.fX, size.fY);
437 make_fan(&fRecs[1], size.fX, size.fY);
438 make_tris(&fRecs[2]);
439
440 SkPaint paint;
441 paint.setDither(true);
442 paint.setFilterBitmap(true);
443
444 for (size_t i = 0; i < SK_ARRAY_COUNT(fRecs); i++) {
445 canvas->save();
446
447 paint.setShader(NULL);
448 canvas->drawVertices(fRecs[i].fMode, fRecs[i].fCount,
449 fRecs[i].fVerts, fRecs[i].fTexs,
450 NULL, NULL, NULL, 0, paint);
451
452 canvas->translate(SkIntToScalar(210), 0);
453
454 paint.setShader(fShader0);
455 canvas->drawVertices(fRecs[i].fMode, fRecs[i].fCount,
456 fRecs[i].fVerts, fRecs[i].fTexs,
457 NULL, NULL, NULL, 0, paint);
458
459 canvas->translate(SkIntToScalar(210), 0);
460
461 paint.setShader(fShader1);
462 canvas->drawVertices(fRecs[i].fMode, fRecs[i].fCount,
463 fRecs[i].fVerts, fRecs[i].fTexs,
464 NULL, NULL, NULL, 0, paint);
465 canvas->restore();
466
467 canvas->translate(0, SkIntToScalar(250));
468 }
469 }
470
471 ///////////////////////////////////////////////////////////////////////////////
472
473 #include "SkGradientShader.h"
474 #include "SkLayerRasterizer.h"
475 #include "SkBlurMaskFilter.h"
476
r0(SkLayerRasterizer * rast,SkPaint & p)477 static void r0(SkLayerRasterizer* rast, SkPaint& p)
478 {
479 p.setMaskFilter(SkBlurMaskFilter::Create(SkIntToScalar(3),
480 SkBlurMaskFilter::kNormal_BlurStyle))->unref();
481 rast->addLayer(p, SkIntToScalar(3), SkIntToScalar(3));
482
483 p.setMaskFilter(NULL);
484 p.setStyle(SkPaint::kStroke_Style);
485 p.setStrokeWidth(SK_Scalar1);
486 rast->addLayer(p);
487
488 p.setAlpha(0x11);
489 p.setStyle(SkPaint::kFill_Style);
490 p.setXfermodeMode(SkXfermode::kSrc_Mode);
491 rast->addLayer(p);
492 }
493
r1(SkLayerRasterizer * rast,SkPaint & p)494 static void r1(SkLayerRasterizer* rast, SkPaint& p)
495 {
496 rast->addLayer(p);
497
498 p.setAlpha(0x40);
499 p.setXfermodeMode(SkXfermode::kSrc_Mode);
500 p.setStyle(SkPaint::kStroke_Style);
501 p.setStrokeWidth(SK_Scalar1*2);
502 rast->addLayer(p);
503 }
504
r2(SkLayerRasterizer * rast,SkPaint & p)505 static void r2(SkLayerRasterizer* rast, SkPaint& p)
506 {
507 p.setStyle(SkPaint::kStrokeAndFill_Style);
508 p.setStrokeWidth(SK_Scalar1*4);
509 rast->addLayer(p);
510
511 p.setStyle(SkPaint::kStroke_Style);
512 p.setStrokeWidth(SK_Scalar1*3/2);
513 p.setXfermodeMode(SkXfermode::kClear_Mode);
514 rast->addLayer(p);
515 }
516
r3(SkLayerRasterizer * rast,SkPaint & p)517 static void r3(SkLayerRasterizer* rast, SkPaint& p)
518 {
519 p.setStyle(SkPaint::kStroke_Style);
520 p.setStrokeWidth(SK_Scalar1*3);
521 rast->addLayer(p);
522
523 p.setAlpha(0x20);
524 p.setStyle(SkPaint::kFill_Style);
525 p.setXfermodeMode(SkXfermode::kSrc_Mode);
526 rast->addLayer(p);
527 }
528
r4(SkLayerRasterizer * rast,SkPaint & p)529 static void r4(SkLayerRasterizer* rast, SkPaint& p)
530 {
531 p.setAlpha(0x60);
532 rast->addLayer(p, SkIntToScalar(3), SkIntToScalar(3));
533
534 p.setAlpha(0xFF);
535 p.setXfermodeMode(SkXfermode::kClear_Mode);
536 rast->addLayer(p, SK_Scalar1*3/2, SK_Scalar1*3/2);
537
538 p.setXfermode(NULL);
539 rast->addLayer(p);
540 }
541
542 #include "SkDiscretePathEffect.h"
543
r5(SkLayerRasterizer * rast,SkPaint & p)544 static void r5(SkLayerRasterizer* rast, SkPaint& p)
545 {
546 rast->addLayer(p);
547
548 p.setPathEffect(new SkDiscretePathEffect(SK_Scalar1*4, SK_Scalar1*3))->unref();
549 p.setXfermodeMode(SkXfermode::kSrcOut_Mode);
550 rast->addLayer(p);
551 }
552
r6(SkLayerRasterizer * rast,SkPaint & p)553 static void r6(SkLayerRasterizer* rast, SkPaint& p)
554 {
555 rast->addLayer(p);
556
557 p.setAntiAlias(false);
558 SkLayerRasterizer* rast2 = new SkLayerRasterizer;
559 r5(rast2, p);
560 p.setRasterizer(rast2)->unref();
561 p.setXfermodeMode(SkXfermode::kClear_Mode);
562 rast->addLayer(p);
563 }
564
565 #include "Sk2DPathEffect.h"
566
MakeDotEffect(SkScalar radius,const SkMatrix & matrix)567 static SkPathEffect* MakeDotEffect(SkScalar radius, const SkMatrix& matrix) {
568 SkPath path;
569 path.addCircle(0, 0, radius);
570 return new SkPath2DPathEffect(matrix, path);
571 }
572
r7(SkLayerRasterizer * rast,SkPaint & p)573 static void r7(SkLayerRasterizer* rast, SkPaint& p)
574 {
575 SkMatrix lattice;
576 lattice.setScale(SK_Scalar1*6, SK_Scalar1*6, 0, 0);
577 lattice.postSkew(SK_Scalar1/3, 0, 0, 0);
578 p.setPathEffect(MakeDotEffect(SK_Scalar1*4, lattice))->unref();
579 rast->addLayer(p);
580 }
581
r8(SkLayerRasterizer * rast,SkPaint & p)582 static void r8(SkLayerRasterizer* rast, SkPaint& p)
583 {
584 rast->addLayer(p);
585
586 SkMatrix lattice;
587 lattice.setScale(SK_Scalar1*6, SK_Scalar1*6, 0, 0);
588 lattice.postSkew(SK_Scalar1/3, 0, 0, 0);
589 p.setPathEffect(MakeDotEffect(SK_Scalar1*2, lattice))->unref();
590 p.setXfermodeMode(SkXfermode::kClear_Mode);
591 rast->addLayer(p);
592
593 p.setPathEffect(NULL);
594 p.setXfermode(NULL);
595 p.setStyle(SkPaint::kStroke_Style);
596 p.setStrokeWidth(SK_Scalar1);
597 rast->addLayer(p);
598 }
599
600 class Line2DPathEffect : public Sk2DPathEffect {
601 public:
Line2DPathEffect(SkScalar width,const SkMatrix & matrix)602 Line2DPathEffect(SkScalar width, const SkMatrix& matrix)
603 : Sk2DPathEffect(matrix), fWidth(width) {}
604
filterPath(SkPath * dst,const SkPath & src,SkScalar * width)605 virtual bool filterPath(SkPath* dst, const SkPath& src, SkScalar* width)
606 {
607 if (this->INHERITED::filterPath(dst, src, width))
608 {
609 *width = fWidth;
610 return true;
611 }
612 return false;
613 }
614
getFactory()615 virtual Factory getFactory() { return CreateProc; }
flatten(SkFlattenableWriteBuffer & buffer)616 virtual void flatten(SkFlattenableWriteBuffer& buffer)
617 {
618 this->INHERITED::flatten(buffer);
619 buffer.writeScalar(fWidth);
620 }
621 protected:
nextSpan(int u,int v,int ucount,SkPath * dst)622 virtual void nextSpan(int u, int v, int ucount, SkPath* dst)
623 {
624 if (ucount > 1)
625 {
626 SkPoint src[2], dstP[2];
627
628 src[0].set(SkIntToScalar(u) + SK_ScalarHalf,
629 SkIntToScalar(v) + SK_ScalarHalf);
630 src[1].set(SkIntToScalar(u+ucount) + SK_ScalarHalf,
631 SkIntToScalar(v) + SK_ScalarHalf);
632 this->getMatrix().mapPoints(dstP, src, 2);
633
634 dst->moveTo(dstP[0]);
635 dst->lineTo(dstP[1]);
636 }
637 }
638
Line2DPathEffect(SkFlattenableReadBuffer & buffer)639 Line2DPathEffect(SkFlattenableReadBuffer& buffer) : Sk2DPathEffect(buffer)
640 {
641 fWidth = buffer.readScalar();
642 }
643
644 private:
645 SkScalar fWidth;
646
CreateProc(SkFlattenableReadBuffer & buffer)647 static SkFlattenable* CreateProc(SkFlattenableReadBuffer& buffer)
648 {
649 return new Line2DPathEffect(buffer);
650 }
651
652 typedef Sk2DPathEffect INHERITED;
653 };
654
r9(SkLayerRasterizer * rast,SkPaint & p)655 static void r9(SkLayerRasterizer* rast, SkPaint& p)
656 {
657 rast->addLayer(p);
658
659 SkMatrix lattice;
660 lattice.setScale(SK_Scalar1, SK_Scalar1*6, 0, 0);
661 lattice.postRotate(SkIntToScalar(30), 0, 0);
662 p.setPathEffect(new Line2DPathEffect(SK_Scalar1*2, lattice))->unref();
663 p.setXfermodeMode(SkXfermode::kClear_Mode);
664 rast->addLayer(p);
665
666 p.setPathEffect(NULL);
667 p.setXfermode(NULL);
668 p.setStyle(SkPaint::kStroke_Style);
669 p.setStrokeWidth(SK_Scalar1);
670 rast->addLayer(p);
671 }
672
673 typedef void (*raster_proc)(SkLayerRasterizer*, SkPaint&);
674
675 static const raster_proc gRastProcs[] = {
676 r0, r1, r2, r3, r4, r5, r6, r7, r8, r9
677 };
678
apply_shader(SkPaint * paint,int index)679 static void apply_shader(SkPaint* paint, int index) {
680 raster_proc proc = gRastProcs[index];
681 SkPaint p;
682 SkLayerRasterizer* rast = new SkLayerRasterizer;
683
684 p.setAntiAlias(true);
685 proc(rast, p);
686 paint->setRasterizer(rast)->unref();
687 paint->setColor(SK_ColorBLUE);
688 }
689
690 #include "SkTypeface.h"
691
texteffect_slide(SkCanvas * canvas)692 static void texteffect_slide(SkCanvas* canvas) {
693 const char* str = "Google";
694 size_t len = strlen(str);
695 SkScalar x = 20;
696 SkScalar y = 80;
697 SkPaint paint;
698 paint.setTypeface(SkTypeface::CreateFromName("Georgia", SkTypeface::kItalic));
699 paint.setTextSize(75);
700 paint.setAntiAlias(true);
701 paint.setColor(SK_ColorBLUE);
702 for (size_t i = 0; i < SK_ARRAY_COUNT(gRastProcs); i++) {
703 apply_shader(&paint, i);
704 canvas->drawText(str, len, x, y, paint);
705 y += 80;
706 if (i == 4) {
707 x += 320;
708 y = 80;
709 }
710 }
711 }
712
713 ///////////////////////////////////////////////////////////////////////////////
714
715 #include "SkImageEncoder.h"
716
717 static const SlideProc gProc[] = {
718 patheffect_slide,
719 gradient_slide,
720 textonpath_slide,
721 mesh_slide,
722 texteffect_slide
723 };
724
725 class SlideView : public SampleView {
726 int fIndex;
727 public:
SlideView()728 SlideView() {
729 fIndex = 0;
730
731 SkBitmap bm;
732 bm.setConfig(SkBitmap::kARGB_8888_Config, 1024, 768);
733 bm.allocPixels();
734 SkCanvas canvas(bm);
735 SkScalar s = SkIntToScalar(1024) / 640;
736 canvas.scale(s, s);
737 for (size_t i = 0; i < SK_ARRAY_COUNT(gProc); i++) {
738 canvas.save();
739 canvas.drawColor(BG_COLOR);
740 gProc[i](&canvas);
741 canvas.restore();
742 SkString str;
743 str.printf("/skimages/slide_%d.png", i);
744 SkImageEncoder::EncodeFile(str.c_str(), bm, SkImageEncoder::kPNG_Type, 100);
745 }
746 this->setBGColor(BG_COLOR);
747 }
748
749 protected:
750 // overrides from SkEventSink
onQuery(SkEvent * evt)751 virtual bool onQuery(SkEvent* evt) {
752 if (SampleCode::TitleQ(*evt)) {
753 SampleCode::TitleR(evt, "Slides");
754 return true;
755 }
756 return this->INHERITED::onQuery(evt);
757 }
758
onDrawContent(SkCanvas * canvas)759 virtual void onDrawContent(SkCanvas* canvas) {
760 gProc[fIndex](canvas);
761 }
762
onFindClickHandler(SkScalar x,SkScalar y)763 virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y) {
764 fIndex = (fIndex + 1) % SK_ARRAY_COUNT(gProc);
765 this->inval(NULL);
766 return NULL;
767 }
768
769 private:
770 typedef SampleView INHERITED;
771 };
772
773 //////////////////////////////////////////////////////////////////////////////
774
MyFactory()775 static SkView* MyFactory() { return new SlideView; }
776 static SkViewRegister reg(MyFactory);
777
778