1/* 2 * Copyright 2012 Google Inc. 3 * 4 * Use of this source code is governed by a BSD-style license that can be 5 * found in the LICENSE file. 6 */ 7 8#include "gm.h" 9#include "SkAAClip.h" 10#include "SkCanvas.h" 11#include "SkPath.h" 12 13namespace skiagm { 14 15static void paint_rgn(SkCanvas* canvas, const SkAAClip& clip, 16 const SkPaint& paint) { 17 SkMask mask; 18 SkBitmap bm; 19 20 clip.copyToMask(&mask); 21 22 SkAutoMaskFreeImage amfi(mask.fImage); 23 24 bm.installMaskPixels(mask); 25 26 // need to copy for deferred drawing test to work 27 SkBitmap bm2; 28 29 bm.deepCopyTo(&bm2); 30 31 canvas->drawBitmap(bm2, 32 SK_Scalar1 * mask.fBounds.fLeft, 33 SK_Scalar1 * mask.fBounds.fTop, 34 &paint); 35} 36 37////////////////////////////////////////////////////////////////////////////// 38/* 39 * This GM tests anti aliased single operation booleans with SkAAClips, 40 * SkRect and SkPaths. 41 */ 42class SimpleClipGM : public GM { 43public: 44 enum SkGeomTypes { 45 kRect_GeomType, 46 kPath_GeomType, 47 kAAClip_GeomType 48 }; 49 50 SimpleClipGM(SkGeomTypes geomType) 51 : fGeomType(geomType) { 52 } 53 54protected: 55 void onOnceBeforeDraw() override { 56 // offset the rects a bit so we get anti-aliasing in the rect case 57 fBase.set(100.65f, 58 100.65f, 59 150.65f, 60 150.65f); 61 fRect = fBase; 62 fRect.inset(5, 5); 63 fRect.offset(25, 25); 64 65 fBasePath.addRoundRect(fBase, SkIntToScalar(5), SkIntToScalar(5)); 66 fRectPath.addRoundRect(fRect, SkIntToScalar(5), SkIntToScalar(5)); 67 INHERITED::setBGColor(sk_tool_utils::color_to_565(0xFFDDDDDD)); 68 } 69 70 void buildRgn(SkAAClip* clip, SkRegion::Op op) { 71 clip->setPath(fBasePath, nullptr, true); 72 73 SkAAClip clip2; 74 clip2.setPath(fRectPath, nullptr, true); 75 clip->op(clip2, op); 76 } 77 78 void drawOrig(SkCanvas* canvas) { 79 SkPaint paint; 80 81 paint.setStyle(SkPaint::kStroke_Style); 82 paint.setColor(SK_ColorBLACK); 83 84 canvas->drawRect(fBase, paint); 85 canvas->drawRect(fRect, paint); 86 } 87 88 void drawRgnOped(SkCanvas* canvas, SkRegion::Op op, SkColor color) { 89 90 SkAAClip clip; 91 92 this->buildRgn(&clip, op); 93 this->drawOrig(canvas); 94 95 SkPaint paint; 96 paint.setColor(color); 97 paint_rgn(canvas, clip, paint); 98 } 99 100 void drawPathsOped(SkCanvas* canvas, SkRegion::Op op, SkColor color) { 101 102 this->drawOrig(canvas); 103 104 canvas->save(); 105 106 // create the clip mask with the supplied boolean op 107 if (kPath_GeomType == fGeomType) { 108 // path-based case 109 canvas->clipPath(fBasePath, SkRegion::kReplace_Op, true); 110 canvas->clipPath(fRectPath, op, true); 111 } else { 112 // rect-based case 113 canvas->clipRect(fBase, SkRegion::kReplace_Op, true); 114 canvas->clipRect(fRect, op, true); 115 } 116 117 // draw a rect that will entirely cover the clip mask area 118 SkPaint paint; 119 paint.setColor(color); 120 121 SkRect r = SkRect::MakeLTRB(SkIntToScalar(90), SkIntToScalar(90), 122 SkIntToScalar(180), SkIntToScalar(180)); 123 124 canvas->drawRect(r, paint); 125 126 canvas->restore(); 127 } 128 129 SkString onShortName() override { 130 SkString str; 131 str.printf("simpleaaclip_%s", 132 kRect_GeomType == fGeomType ? "rect" : 133 (kPath_GeomType == fGeomType ? "path" : 134 "aaclip")); 135 return str; 136 } 137 138 SkISize onISize() override { 139 return SkISize::Make(640, 480); 140 } 141 142 void onDraw(SkCanvas* canvas) override { 143 144 static const struct { 145 SkColor fColor; 146 const char* fName; 147 SkRegion::Op fOp; 148 } gOps[] = { 149 { SK_ColorBLACK, "Difference", SkRegion::kDifference_Op }, 150 { SK_ColorRED, "Intersect", SkRegion::kIntersect_Op }, 151 { sk_tool_utils::color_to_565(0xFF008800), "Union", SkRegion::kUnion_Op }, 152 { SK_ColorGREEN, "Rev Diff", SkRegion::kReverseDifference_Op }, 153 { SK_ColorYELLOW, "Replace", SkRegion::kReplace_Op }, 154 { SK_ColorBLUE, "XOR", SkRegion::kXOR_Op }, 155 }; 156 157 SkPaint textPaint; 158 textPaint.setAntiAlias(true); 159 sk_tool_utils::set_portable_typeface(&textPaint); 160 textPaint.setTextSize(SK_Scalar1*24); 161 int xOff = 0; 162 163 for (size_t op = 0; op < SK_ARRAY_COUNT(gOps); op++) { 164 canvas->drawText(gOps[op].fName, strlen(gOps[op].fName), 165 SkIntToScalar(75), SkIntToScalar(50), 166 textPaint); 167 168 if (kAAClip_GeomType == fGeomType) { 169 this->drawRgnOped(canvas, gOps[op].fOp, gOps[op].fColor); 170 } else { 171 this->drawPathsOped(canvas, gOps[op].fOp, gOps[op].fColor); 172 } 173 174 if (xOff >= 400) { 175 canvas->translate(SkIntToScalar(-400), SkIntToScalar(250)); 176 xOff = 0; 177 } else { 178 canvas->translate(SkIntToScalar(200), 0); 179 xOff += 200; 180 } 181 } 182 } 183private: 184 185 SkGeomTypes fGeomType; 186 187 SkRect fBase; 188 SkRect fRect; 189 190 SkPath fBasePath; // fBase as a round rect 191 SkPath fRectPath; // fRect as a round rect 192 193 typedef GM INHERITED; 194}; 195 196////////////////////////////////////////////////////////////////////////////// 197 198// rects 199DEF_GM( return new SimpleClipGM(SimpleClipGM::kRect_GeomType); ) 200DEF_GM( return new SimpleClipGM(SimpleClipGM::kPath_GeomType); ) 201DEF_GM( return new SimpleClipGM(SimpleClipGM::kAAClip_GeomType); ) 202 203} 204