| epoger@google.com | ec3ed6a | 2011-07-28 14:26:00 +0000 | [diff] [blame] | 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 | */ |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 8 | #include "gm.h" |
| 9 | #include "SkBitmap.h" |
| 10 | #include "SkShader.h" |
| 11 | #include "SkXfermode.h" |
| 12 | |
| 13 | namespace skiagm { |
| 14 | |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 15 | static void make_bitmaps(int w, int h, SkBitmap* src, SkBitmap* dst) { |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 16 | src->setConfig(SkBitmap::kARGB_8888_Config, w, h); |
| 17 | src->allocPixels(); |
| junov@google.com | dbfac8a | 2012-12-06 21:47:40 +0000 | [diff] [blame] | 18 | src->eraseColor(SK_ColorTRANSPARENT); |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 19 | |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 20 | SkPaint p; |
| mike@reedtribe.org | 3bd2173 | 2012-09-26 02:45:10 +0000 | [diff] [blame] | 21 | p.setAntiAlias(true); |
| 22 | |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 23 | SkRect r; |
| 24 | SkScalar ww = SkIntToScalar(w); |
| 25 | SkScalar hh = SkIntToScalar(h); |
| 26 | |
| mike@reedtribe.org | 3bd2173 | 2012-09-26 02:45:10 +0000 | [diff] [blame] | 27 | { |
| 28 | SkCanvas c(*src); |
| 29 | p.setColor(0xFFFFCC44); |
| 30 | r.set(0, 0, ww*3/4, hh*3/4); |
| 31 | c.drawOval(r, p); |
| 32 | } |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 33 | |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 34 | dst->setConfig(SkBitmap::kARGB_8888_Config, w, h); |
| 35 | dst->allocPixels(); |
| junov@google.com | dbfac8a | 2012-12-06 21:47:40 +0000 | [diff] [blame] | 36 | dst->eraseColor(SK_ColorTRANSPARENT); |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 37 | |
| mike@reedtribe.org | 3bd2173 | 2012-09-26 02:45:10 +0000 | [diff] [blame] | 38 | { |
| 39 | SkCanvas c(*dst); |
| 40 | p.setColor(0xFF66AAFF); |
| 41 | r.set(ww/3, hh/3, ww*19/20, hh*19/20); |
| 42 | c.drawRect(r, p); |
| 43 | } |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 44 | } |
| 45 | |
| scroggo@google.com | 7312a18 | 2013-01-28 21:14:21 +0000 | [diff] [blame] | 46 | static uint16_t gData[] = { 0xFFFF, 0xCCCF, 0xCCCF, 0xFFFF }; |
| 47 | |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 48 | class XfermodesGM : public GM { |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 49 | enum SrcType { |
| 50 | //! A WxH image with a rectangle in the lower right. |
| 51 | kRectangleImage_SrcType = 0x01, |
| 52 | //! kRectangleImage_SrcType with an alpha of 34.5%. |
| 53 | kRectangleImageWithAlpha_SrcType = 0x02, |
| 54 | //! kRectnagleImageWithAlpha_SrcType scaled down by half. |
| 55 | kSmallRectangleImageWithAlpha_SrcType = 0x04, |
| 56 | //! kRectangleImage_SrcType drawn directly instead in an image. |
| 57 | kRectangle_SrcType = 0x08, |
| 58 | //! Two rectangles, first on the right half, second on the bottom half. |
| 59 | kQuarterClear_SrcType = 0x10, |
| 60 | //! kQuarterClear_SrcType in a layer. |
| 61 | kQuarterClearInLayer_SrcType = 0x20, |
| 62 | |
| 63 | kAll_SrcType = 0x3F, //!< All the source types. |
| 64 | kBasic_SrcType = 0x03, //!< Just basic source types. |
| 65 | }; |
| 66 | |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 67 | SkBitmap fBG; |
| 68 | SkBitmap fSrcB, fDstB; |
| 69 | |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 70 | /* The srcType argument indicates what to draw for the source part. Skia |
| 71 | * uses the implied shape of the drawing command and these modes |
| 72 | * demonstrate that. |
| 73 | */ |
| 74 | void draw_mode(SkCanvas* canvas, SkXfermode* mode, SrcType srcType, |
| reed@google.com | f3c1cc9 | 2010-12-23 16:45:33 +0000 | [diff] [blame] | 75 | SkScalar x, SkScalar y) { |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 76 | SkPaint p; |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 77 | SkMatrix m; |
| 78 | bool restoreNeeded = false; |
| 79 | m.setTranslate(x, y); |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 80 | |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 81 | canvas->drawBitmapMatrix(fSrcB, m, &p); |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 82 | p.setXfermode(mode); |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 83 | switch (srcType) { |
| 84 | case kQuarterClearInLayer_SrcType: { |
| 85 | SkRect bounds = SkRect::MakeXYWH(x, y, W, H); |
| 86 | canvas->saveLayer(&bounds, &p); |
| 87 | restoreNeeded = true; |
| 88 | p.setXfermodeMode(SkXfermode::kSrcOver_Mode); |
| 89 | // Fall through. |
| 90 | } |
| 91 | case kQuarterClear_SrcType: { |
| 92 | SkScalar halfW = SkIntToScalar(W) / 2; |
| 93 | SkScalar halfH = SkIntToScalar(H) / 2; |
| 94 | p.setColor(0xFF66AAFF); |
| 95 | SkRect r = SkRect::MakeXYWH(x + halfW, y, halfW, H); |
| 96 | canvas->drawRect(r, p); |
| 97 | p.setColor(0xFFAA66FF); |
| 98 | r = SkRect::MakeXYWH(x, y + halfH, W, halfH); |
| 99 | canvas->drawRect(r, p); |
| 100 | break; |
| 101 | } |
| 102 | case kRectangle_SrcType: { |
| 103 | SkScalar w = SkIntToScalar(W); |
| 104 | SkScalar h = SkIntToScalar(H); |
| 105 | SkRect r = SkRect::MakeXYWH(x + w / 3, y + h / 3, |
| 106 | w * 37 / 60, h * 37 / 60); |
| 107 | p.setColor(0xFF66AAFF); |
| 108 | canvas->drawRect(r, p); |
| 109 | break; |
| 110 | } |
| 111 | case kSmallRectangleImageWithAlpha_SrcType: |
| 112 | m.postScale(SK_ScalarHalf, SK_ScalarHalf, x, y); |
| 113 | // Fall through. |
| 114 | case kRectangleImageWithAlpha_SrcType: |
| 115 | p.setAlpha(0x88); |
| 116 | // Fall through. |
| 117 | case kRectangleImage_SrcType: |
| 118 | canvas->drawBitmapMatrix(fDstB, m, &p); |
| 119 | break; |
| 120 | default: |
| 121 | break; |
| 122 | } |
| 123 | |
| 124 | if (restoreNeeded) { |
| 125 | canvas->restore(); |
| 126 | } |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 127 | } |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 128 | |
| scroggo@google.com | 7312a18 | 2013-01-28 21:14:21 +0000 | [diff] [blame] | 129 | virtual void onOnceBeforeDraw() SK_OVERRIDE { |
| reed@google.com | 383a697 | 2013-10-21 14:00:07 +0000 | [diff] [blame] | 130 | fBG.setConfig(SkBitmap::kARGB_4444_Config, 2, 2, 4, kOpaque_SkAlphaType); |
| scroggo@google.com | 7312a18 | 2013-01-28 21:14:21 +0000 | [diff] [blame] | 131 | fBG.setPixels(gData); |
| rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 132 | |
| scroggo@google.com | 7312a18 | 2013-01-28 21:14:21 +0000 | [diff] [blame] | 133 | make_bitmaps(W, H, &fSrcB, &fDstB); |
| reed@google.com | 5213c04 | 2011-12-20 13:41:09 +0000 | [diff] [blame] | 134 | } |
| 135 | |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 136 | public: |
| reed@google.com | f3c1cc9 | 2010-12-23 16:45:33 +0000 | [diff] [blame] | 137 | const static int W = 64; |
| 138 | const static int H = 64; |
| scroggo@google.com | 7312a18 | 2013-01-28 21:14:21 +0000 | [diff] [blame] | 139 | XfermodesGM() {} |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 140 | |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 141 | protected: |
| reed@android.com | 048522d | 2009-06-23 12:19:41 +0000 | [diff] [blame] | 142 | virtual SkString onShortName() { |
| reed@android.com | 8015dd8 | 2009-06-21 00:49:18 +0000 | [diff] [blame] | 143 | return SkString("xfermodes"); |
| 144 | } |
| 145 | |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 146 | virtual SkISize onISize() { |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 147 | return make_isize(1590, 640); |
| reed@android.com | 048522d | 2009-06-23 12:19:41 +0000 | [diff] [blame] | 148 | } |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 149 | |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 150 | virtual void onDraw(SkCanvas* canvas) { |
| reed@google.com | f3c1cc9 | 2010-12-23 16:45:33 +0000 | [diff] [blame] | 151 | canvas->translate(SkIntToScalar(10), SkIntToScalar(20)); |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 152 | |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 153 | const struct { |
| reed@google.com | f3c1cc9 | 2010-12-23 16:45:33 +0000 | [diff] [blame] | 154 | SkXfermode::Mode fMode; |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 155 | const char* fLabel; |
| 156 | int fSourceTypeMask; // The source types to use this |
| 157 | // mode with. See draw_mode for |
| 158 | // an explanation of each type. |
| 159 | // PDF has to play some tricks |
| 160 | // to support the base modes, |
| 161 | // test those more extensively. |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 162 | } gModes[] = { |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 163 | { SkXfermode::kClear_Mode, "Clear", kAll_SrcType }, |
| 164 | { SkXfermode::kSrc_Mode, "Src", kAll_SrcType }, |
| 165 | { SkXfermode::kDst_Mode, "Dst", kAll_SrcType }, |
| 166 | { SkXfermode::kSrcOver_Mode, "SrcOver", kAll_SrcType }, |
| 167 | { SkXfermode::kDstOver_Mode, "DstOver", kAll_SrcType }, |
| 168 | { SkXfermode::kSrcIn_Mode, "SrcIn", kAll_SrcType }, |
| 169 | { SkXfermode::kDstIn_Mode, "DstIn", kAll_SrcType }, |
| 170 | { SkXfermode::kSrcOut_Mode, "SrcOut", kAll_SrcType }, |
| 171 | { SkXfermode::kDstOut_Mode, "DstOut", kAll_SrcType }, |
| 172 | { SkXfermode::kSrcATop_Mode, "SrcATop", kAll_SrcType }, |
| 173 | { SkXfermode::kDstATop_Mode, "DstATop", kAll_SrcType }, |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 174 | |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 175 | { SkXfermode::kXor_Mode, "Xor", kBasic_SrcType }, |
| 176 | { SkXfermode::kPlus_Mode, "Plus", kBasic_SrcType }, |
| 177 | { SkXfermode::kModulate_Mode, "Modulate", kAll_SrcType }, |
| 178 | { SkXfermode::kScreen_Mode, "Screen", kBasic_SrcType }, |
| 179 | { SkXfermode::kOverlay_Mode, "Overlay", kBasic_SrcType }, |
| 180 | { SkXfermode::kDarken_Mode, "Darken", kBasic_SrcType }, |
| 181 | { SkXfermode::kLighten_Mode, "Lighten", kBasic_SrcType }, |
| 182 | { SkXfermode::kColorDodge_Mode, "ColorDodge", kBasic_SrcType }, |
| 183 | { SkXfermode::kColorBurn_Mode, "ColorBurn", kBasic_SrcType }, |
| 184 | { SkXfermode::kHardLight_Mode, "HardLight", kBasic_SrcType }, |
| 185 | { SkXfermode::kSoftLight_Mode, "SoftLight", kBasic_SrcType }, |
| 186 | { SkXfermode::kDifference_Mode, "Difference", kBasic_SrcType }, |
| 187 | { SkXfermode::kExclusion_Mode, "Exclusion", kBasic_SrcType }, |
| 188 | { SkXfermode::kMultiply_Mode, "Multiply", kAll_SrcType }, |
| 189 | { SkXfermode::kHue_Mode, "Hue", kBasic_SrcType }, |
| 190 | { SkXfermode::kSaturation_Mode, "Saturation", kBasic_SrcType }, |
| 191 | { SkXfermode::kColor_Mode, "Color", kBasic_SrcType }, |
| 192 | { SkXfermode::kLuminosity_Mode, "Luminosity", kBasic_SrcType }, |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 193 | }; |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 194 | |
| reed@google.com | f3c1cc9 | 2010-12-23 16:45:33 +0000 | [diff] [blame] | 195 | const SkScalar w = SkIntToScalar(W); |
| 196 | const SkScalar h = SkIntToScalar(H); |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 197 | SkShader* s = SkShader::CreateBitmapShader(fBG, |
| 198 | SkShader::kRepeat_TileMode, |
| 199 | SkShader::kRepeat_TileMode); |
| 200 | SkMatrix m; |
| 201 | m.setScale(SkIntToScalar(6), SkIntToScalar(6)); |
| 202 | s->setLocalMatrix(m); |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 203 | |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 204 | SkPaint labelP; |
| 205 | labelP.setAntiAlias(true); |
| 206 | labelP.setTextAlign(SkPaint::kCenter_Align); |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 207 | |
| reed@android.com | c9c6422 | 2009-06-23 00:09:12 +0000 | [diff] [blame] | 208 | const int W = 5; |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 209 | |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 210 | SkScalar x0 = 0; |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 211 | SkScalar y0 = 0; |
| 212 | for (int sourceType = 1; sourceType & kAll_SrcType; sourceType <<= 1) { |
| 213 | SkScalar x = x0, y = y0; |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 214 | for (size_t i = 0; i < SK_ARRAY_COUNT(gModes); i++) { |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 215 | if ((gModes[i].fSourceTypeMask & sourceType) == 0) { |
| 216 | continue; |
| 217 | } |
| reed@android.com | c9c6422 | 2009-06-23 00:09:12 +0000 | [diff] [blame] | 218 | SkXfermode* mode = SkXfermode::Create(gModes[i].fMode); |
| reed@google.com | f3c1cc9 | 2010-12-23 16:45:33 +0000 | [diff] [blame] | 219 | SkAutoUnref aur(mode); |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 220 | SkRect r; |
| 221 | r.set(x, y, x+w, y+h); |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 222 | |
| reed@google.com | f3c1cc9 | 2010-12-23 16:45:33 +0000 | [diff] [blame] | 223 | SkPaint p; |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 224 | p.setStyle(SkPaint::kFill_Style); |
| 225 | p.setShader(s); |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 226 | canvas->drawRect(r, p); |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 227 | |
| reed@google.com | f3c1cc9 | 2010-12-23 16:45:33 +0000 | [diff] [blame] | 228 | canvas->saveLayer(&r, NULL, SkCanvas::kARGB_ClipLayer_SaveFlag); |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 229 | draw_mode(canvas, mode, static_cast<SrcType>(sourceType), |
| 230 | r.fLeft, r.fTop); |
| reed@google.com | f3c1cc9 | 2010-12-23 16:45:33 +0000 | [diff] [blame] | 231 | canvas->restore(); |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 232 | |
| reed@google.com | f3c1cc9 | 2010-12-23 16:45:33 +0000 | [diff] [blame] | 233 | r.inset(-SK_ScalarHalf, -SK_ScalarHalf); |
| 234 | p.setStyle(SkPaint::kStroke_Style); |
| 235 | p.setShader(NULL); |
| 236 | canvas->drawRect(r, p); |
| tomhudson@google.com | 86bf5c9 | 2011-04-26 18:15:03 +0000 | [diff] [blame] | 237 | |
| reed@google.com | f3c1cc9 | 2010-12-23 16:45:33 +0000 | [diff] [blame] | 238 | #if 1 |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 239 | canvas->drawText(gModes[i].fLabel, strlen(gModes[i].fLabel), |
| 240 | x + w/2, y - labelP.getTextSize()/2, labelP); |
| reed@google.com | f3c1cc9 | 2010-12-23 16:45:33 +0000 | [diff] [blame] | 241 | #endif |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 242 | x += w + SkIntToScalar(10); |
| reed@android.com | c9c6422 | 2009-06-23 00:09:12 +0000 | [diff] [blame] | 243 | if ((i % W) == W - 1) { |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 244 | x = x0; |
| 245 | y += h + SkIntToScalar(30); |
| 246 | } |
| 247 | } |
| vandebo@chromium.org | 3b41621 | 2013-10-30 20:48:05 +0000 | [diff] [blame^] | 248 | if (y < 320) { |
| 249 | if (x > x0) { |
| 250 | y += h + SkIntToScalar(30); |
| 251 | } |
| 252 | y0 = y; |
| 253 | } else { |
| 254 | x0 += SkIntToScalar(400); |
| 255 | y0 = 0; |
| 256 | } |
| reed@android.com | dd0ac28 | 2009-06-20 02:38:16 +0000 | [diff] [blame] | 257 | } |
| 258 | s->unref(); |
| 259 | } |
| 260 | |
| 261 | private: |
| 262 | typedef GM INHERITED; |
| 263 | }; |
| 264 | |
| 265 | ////////////////////////////////////////////////////////////////////////////// |
| 266 | |
| 267 | static GM* MyFactory(void*) { return new XfermodesGM; } |
| 268 | static GMRegistry reg(MyFactory); |
| 269 | |
| 270 | } |