| /* |
| * 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 "gm.h" |
| #include "SkCanvas.h" |
| #include "SkColorPriv.h" |
| #include "SkShader.h" |
| |
| static const struct { |
| SkXfermode::Mode fMode; |
| const char* fLabel; |
| } gModes[] = { |
| { SkXfermode::kClear_Mode, "Clear" }, |
| { SkXfermode::kSrc_Mode, "Src" }, |
| { SkXfermode::kDst_Mode, "Dst" }, |
| { SkXfermode::kSrcOver_Mode, "SrcOver" }, |
| { SkXfermode::kDstOver_Mode, "DstOver" }, |
| { SkXfermode::kSrcIn_Mode, "SrcIn" }, |
| { SkXfermode::kDstIn_Mode, "DstIn" }, |
| { SkXfermode::kSrcOut_Mode, "SrcOut" }, |
| { SkXfermode::kDstOut_Mode, "DstOut" }, |
| { SkXfermode::kSrcATop_Mode, "SrcATop" }, |
| { SkXfermode::kDstATop_Mode, "DstATop" }, |
| { SkXfermode::kXor_Mode, "Xor" }, |
| }; |
| |
| const int gWidth = 64; |
| const int gHeight = 64; |
| const SkScalar W = SkIntToScalar(gWidth); |
| const SkScalar H = SkIntToScalar(gHeight); |
| |
| static SkScalar drawCell(SkCanvas* canvas, SkXfermode* mode, SkAlpha a0, SkAlpha a1) { |
| |
| SkPaint paint; |
| paint.setAntiAlias(true); |
| |
| SkRect r = SkRect::MakeWH(W, H); |
| r.inset(W/10, H/10); |
| |
| paint.setColor(SK_ColorBLUE); |
| paint.setAlpha(a0); |
| canvas->drawOval(r, paint); |
| |
| paint.setColor(SK_ColorRED); |
| paint.setAlpha(a1); |
| paint.setXfermode(mode); |
| for (int angle = 0; angle < 24; ++angle) { |
| SkScalar x = SkScalarCos(SkIntToScalar(angle) * (SK_ScalarPI * 2) / 24) * gWidth; |
| SkScalar y = SkScalarSin(SkIntToScalar(angle) * (SK_ScalarPI * 2) / 24) * gHeight; |
| paint.setStrokeWidth(SK_Scalar1 * angle * 2 / 24); |
| canvas->drawLine(W/2, H/2, W/2 + x, H/2 + y, paint); |
| } |
| |
| return H; |
| } |
| |
| static SkShader* make_bg_shader() { |
| SkBitmap bm; |
| bm.setConfig(SkBitmap::kARGB_8888_Config, 2, 2); |
| bm.allocPixels(); |
| *bm.getAddr32(0, 0) = *bm.getAddr32(1, 1) = 0xFFFFFFFF; |
| *bm.getAddr32(1, 0) = *bm.getAddr32(0, 1) = SkPackARGB32(0xFF, 0xCC, 0xCC, 0xCC); |
| |
| SkShader* s = SkShader::CreateBitmapShader(bm, |
| SkShader::kRepeat_TileMode, |
| SkShader::kRepeat_TileMode); |
| |
| SkMatrix m; |
| m.setScale(SkIntToScalar(6), SkIntToScalar(6)); |
| s->setLocalMatrix(m); |
| return s; |
| } |
| |
| namespace skiagm { |
| |
| class HairModesGM : public GM { |
| SkPaint fBGPaint; |
| |
| protected: |
| |
| virtual SkString onShortName() SK_OVERRIDE { |
| return SkString("hairmodes"); |
| } |
| |
| virtual SkISize onISize() { return make_isize(640, 480); } |
| |
| virtual void onOnceBeforeDraw() SK_OVERRIDE { |
| fBGPaint.setShader(make_bg_shader())->unref(); |
| } |
| |
| virtual void onDraw(SkCanvas* canvas) SK_OVERRIDE { |
| const SkRect bounds = SkRect::MakeWH(W, H); |
| static const SkAlpha gAlphaValue[] = { 0xFF, 0x88, 0x88 }; |
| |
| canvas->translate(SkIntToScalar(4), SkIntToScalar(4)); |
| |
| for (int alpha = 0; alpha < 4; ++alpha) { |
| canvas->save(); |
| canvas->save(); |
| for (size_t i = 0; i < SK_ARRAY_COUNT(gModes); ++i) { |
| if (6 == i) { |
| canvas->restore(); |
| canvas->translate(W * 5, 0); |
| canvas->save(); |
| } |
| SkXfermode* mode = SkXfermode::Create(gModes[i].fMode); |
| |
| canvas->drawRect(bounds, fBGPaint); |
| canvas->saveLayer(&bounds, NULL); |
| SkScalar dy = drawCell(canvas, mode, |
| gAlphaValue[alpha & 1], |
| gAlphaValue[alpha & 2]); |
| canvas->restore(); |
| |
| canvas->translate(0, dy * 5 / 4); |
| SkSafeUnref(mode); |
| } |
| canvas->restore(); |
| canvas->restore(); |
| canvas->translate(W * 5 / 4, 0); |
| } |
| } |
| |
| // disable pdf for now, since it crashes on mac |
| virtual uint32_t onGetFlags() const { return kSkipPDF_Flag; } |
| |
| private: |
| typedef GM INHERITED; |
| }; |
| |
| ////////////////////////////////////////////////////////////////////////////// |
| |
| static GM* MyFactory(void*) { return new HairModesGM; } |
| static GMRegistry reg(MyFactory); |
| |
| } |