| /* | 
 |  * Copyright 2012 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 "SkGradientShader.h" | 
 | #include "SkSurface.h" | 
 |  | 
 | #if SK_SUPPORT_GPU | 
 |     #include "SkGpuDevice.h" | 
 | #endif | 
 |  | 
 | #define W   SkIntToScalar(80) | 
 | #define H   SkIntToScalar(60) | 
 |  | 
 | typedef void (*PaintProc)(SkPaint*); | 
 |  | 
 | static void identity_paintproc(SkPaint* paint) { | 
 |     paint->setShader(NULL); | 
 | } | 
 |  | 
 | static void gradient_paintproc(SkPaint* paint) { | 
 |     const SkColor colors[] = { SK_ColorGREEN, SK_ColorBLUE }; | 
 |     const SkPoint pts[] = { { 0, 0 }, { W, H } }; | 
 |     SkShader* s = SkGradientShader::CreateLinear(pts, colors, NULL, | 
 |                                                  SK_ARRAY_COUNT(colors), | 
 |                                                  SkShader::kClamp_TileMode); | 
 |     paint->setShader(s)->unref(); | 
 | } | 
 |  | 
 | typedef void (*Proc)(SkCanvas*, const SkPaint&); | 
 |  | 
 | static void draw_hair(SkCanvas* canvas, const SkPaint& paint) { | 
 |     SkPaint p(paint); | 
 |     p.setStrokeWidth(0); | 
 |     canvas->drawLine(0, 0, W, H, p); | 
 | } | 
 |  | 
 | static void draw_thick(SkCanvas* canvas, const SkPaint& paint) { | 
 |     SkPaint p(paint); | 
 |     p.setStrokeWidth(H/5); | 
 |     canvas->drawLine(0, 0, W, H, p); | 
 | } | 
 |  | 
 | static void draw_rect(SkCanvas* canvas, const SkPaint& paint) { | 
 |     canvas->drawRect(SkRect::MakeWH(W, H), paint); | 
 | } | 
 |  | 
 | static void draw_oval(SkCanvas* canvas, const SkPaint& paint) { | 
 |     canvas->drawOval(SkRect::MakeWH(W, H), paint); | 
 | } | 
 |  | 
 | static void draw_text(SkCanvas* canvas, const SkPaint& paint) { | 
 |     SkPaint p(paint); | 
 |     p.setTextSize(H/4); | 
 |     canvas->drawText("Hamburge", 8, 0, H*2/3, p); | 
 | } | 
 |  | 
 | class SrcModeGM : public skiagm::GM { | 
 |     SkPath fPath; | 
 | public: | 
 |     SrcModeGM() { | 
 |         this->setBGColor(SK_ColorBLACK); | 
 |     } | 
 |  | 
 | protected: | 
 |     virtual SkString onShortName() { | 
 |         return SkString("srcmode"); | 
 |     } | 
 |  | 
 |     virtual SkISize onISize() { | 
 |         return SkISize::Make(640, 760); | 
 |     } | 
 |  | 
 |     void drawContent(SkCanvas* canvas) { | 
 |         canvas->translate(SkIntToScalar(20), SkIntToScalar(20)); | 
 |  | 
 |         SkPaint paint; | 
 |         paint.setColor(0x80FF0000); | 
 |  | 
 |         const Proc procs[] = { | 
 |             draw_hair, draw_thick, draw_rect, draw_oval, draw_text | 
 |         }; | 
 |  | 
 |         const SkXfermode::Mode modes[] = { | 
 |             SkXfermode::kSrcOver_Mode, SkXfermode::kSrc_Mode, SkXfermode::kClear_Mode | 
 |         }; | 
 |  | 
 |         const PaintProc paintProcs[] = { | 
 |             identity_paintproc, gradient_paintproc | 
 |         }; | 
 |  | 
 |         for (int aa = 0; aa <= 1; ++aa) { | 
 |             paint.setAntiAlias(SkToBool(aa)); | 
 |             canvas->save(); | 
 |             for (size_t i = 0; i < SK_ARRAY_COUNT(paintProcs); ++i) { | 
 |                 paintProcs[i](&paint); | 
 |                 for (size_t x = 0; x < SK_ARRAY_COUNT(modes); ++x) { | 
 |                     paint.setXfermodeMode(modes[x]); | 
 |                     canvas->save(); | 
 |                     for (size_t y = 0; y < SK_ARRAY_COUNT(procs); ++y) { | 
 |                         procs[y](canvas, paint); | 
 |                         canvas->translate(0, H * 5 / 4); | 
 |                     } | 
 |                     canvas->restore(); | 
 |                     canvas->translate(W * 5 / 4, 0); | 
 |                 } | 
 |             } | 
 |             canvas->restore(); | 
 |             canvas->translate(0, (H * 5 / 4) * SK_ARRAY_COUNT(procs)); | 
 |         } | 
 |     } | 
 |  | 
 |     static SkSurface* compat_surface(SkCanvas* canvas, const SkISize& size, | 
 |                                      bool skipGPU) { | 
 |         SkImage::Info info = { | 
 |             size.width(), | 
 |             size.height(), | 
 |             SkImage::kPMColor_ColorType, | 
 |             SkImage::kPremul_AlphaType | 
 |         }; | 
 | #if SK_SUPPORT_GPU | 
 |         SkDevice* dev = canvas->getDevice(); | 
 |         if (!skipGPU && dev->accessRenderTarget()) { | 
 |             SkGpuDevice* gd = (SkGpuDevice*)dev; | 
 |             GrContext* ctx = gd->context(); | 
 |             return SkSurface::NewRenderTarget(ctx, info, 0); | 
 |         } | 
 | #endif | 
 |         return SkSurface::NewRaster(info); | 
 |     } | 
 |  | 
 |     virtual void onDraw(SkCanvas* canvas) { | 
 |         SkAutoTUnref<SkSurface> surf(compat_surface(canvas, this->getISize(), | 
 |                                                     this->isCanvasDeferred())); | 
 |         surf->getCanvas()->drawColor(SK_ColorWHITE); | 
 |         this->drawContent(surf->getCanvas()); | 
 |         surf->draw(canvas, 0, 0, NULL); | 
 |     } | 
 |  | 
 | private: | 
 |     typedef skiagm::GM INHERITED; | 
 | }; | 
 |  | 
 | /////////////////////////////////////////////////////////////////////////////// | 
 |  | 
 | DEF_GM(return new SrcModeGM;) | 
 |  |