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 "gm.h" 9 #include "SkCanvas.h" 10 //#include "SkParsePath.h" 11 #include "SkPath.h" 12 //#include "SkRandom.h" 13 14 namespace skiagm { 15 16 static const SkColor gPathColor = SK_ColorBLACK; 17 static const SkColor gClipAColor = SK_ColorBLUE; 18 static const SkColor gClipBColor = SK_ColorRED; 19 20 class ComplexClipGM : public GM { 21 public: ComplexClipGM(bool aaclip,bool saveLayer,bool invertDraw)22 ComplexClipGM(bool aaclip, bool saveLayer, bool invertDraw) 23 : fDoAAClip(aaclip) 24 , fDoSaveLayer(saveLayer) 25 , fInvertDraw(invertDraw) { 26 this->setBGColor(0xFFDEDFDE); 27 } 28 29 protected: 30 31 onShortName()32 SkString onShortName() { 33 SkString str; 34 str.printf("complexclip_%s%s%s", 35 fDoAAClip ? "aa" : "bw", 36 fDoSaveLayer ? "_layer" : "", 37 fInvertDraw ? "_invert" : ""); 38 return str; 39 } 40 onISize()41 SkISize onISize() { return SkISize::Make(970, 780); } 42 onDraw(SkCanvas * canvas)43 virtual void onDraw(SkCanvas* canvas) { 44 SkPath path; 45 path.moveTo(SkIntToScalar(0), SkIntToScalar(50)); 46 path.quadTo(SkIntToScalar(0), SkIntToScalar(0), SkIntToScalar(50), SkIntToScalar(0)); 47 path.lineTo(SkIntToScalar(175), SkIntToScalar(0)); 48 path.quadTo(SkIntToScalar(200), SkIntToScalar(0), SkIntToScalar(200), SkIntToScalar(25)); 49 path.lineTo(SkIntToScalar(200), SkIntToScalar(150)); 50 path.quadTo(SkIntToScalar(200), SkIntToScalar(200), SkIntToScalar(150), SkIntToScalar(200)); 51 path.lineTo(SkIntToScalar(0), SkIntToScalar(200)); 52 path.close(); 53 path.moveTo(SkIntToScalar(50), SkIntToScalar(50)); 54 path.lineTo(SkIntToScalar(150), SkIntToScalar(50)); 55 path.lineTo(SkIntToScalar(150), SkIntToScalar(125)); 56 path.quadTo(SkIntToScalar(150), SkIntToScalar(150), SkIntToScalar(125), SkIntToScalar(150)); 57 path.lineTo(SkIntToScalar(50), SkIntToScalar(150)); 58 path.close(); 59 if (fInvertDraw) { 60 path.setFillType(SkPath::kInverseEvenOdd_FillType); 61 } else { 62 path.setFillType(SkPath::kEvenOdd_FillType); 63 } 64 SkPaint pathPaint; 65 pathPaint.setAntiAlias(true); 66 pathPaint.setColor(gPathColor); 67 68 SkPath clipA; 69 clipA.moveTo(SkIntToScalar(10), SkIntToScalar(20)); 70 clipA.lineTo(SkIntToScalar(165), SkIntToScalar(22)); 71 clipA.lineTo(SkIntToScalar(70), SkIntToScalar(105)); 72 clipA.lineTo(SkIntToScalar(165), SkIntToScalar(177)); 73 clipA.lineTo(SkIntToScalar(-5), SkIntToScalar(180)); 74 clipA.close(); 75 76 SkPath clipB; 77 clipB.moveTo(SkIntToScalar(40), SkIntToScalar(10)); 78 clipB.lineTo(SkIntToScalar(190), SkIntToScalar(15)); 79 clipB.lineTo(SkIntToScalar(195), SkIntToScalar(190)); 80 clipB.lineTo(SkIntToScalar(40), SkIntToScalar(185)); 81 clipB.lineTo(SkIntToScalar(155), SkIntToScalar(100)); 82 clipB.close(); 83 84 SkPaint paint; 85 paint.setAntiAlias(true); 86 sk_tool_utils::set_portable_typeface(&paint); 87 paint.setTextSize(SkIntToScalar(20)); 88 89 static const struct { 90 SkRegion::Op fOp; 91 const char* fName; 92 } gOps[] = { //extra spaces in names for measureText 93 {SkRegion::kIntersect_Op, "Isect "}, 94 {SkRegion::kDifference_Op, "Diff " }, 95 {SkRegion::kUnion_Op, "Union "}, 96 {SkRegion::kXOR_Op, "Xor " }, 97 {SkRegion::kReverseDifference_Op, "RDiff "} 98 }; 99 100 canvas->translate(SkIntToScalar(20), SkIntToScalar(20)); 101 canvas->scale(3 * SK_Scalar1 / 4, 3 * SK_Scalar1 / 4); 102 103 if (fDoSaveLayer) { 104 // We want the layer to appear symmetric relative to actual 105 // device boundaries so we need to "undo" the effect of the 106 // scale and translate 107 SkRect bounds = SkRect::MakeLTRB( 108 4.0f/3.0f * -20, 109 4.0f/3.0f * -20, 110 4.0f/3.0f * (this->getISize().fWidth - 20), 111 4.0f/3.0f * (this->getISize().fHeight - 20)); 112 113 bounds.inset(SkIntToScalar(100), SkIntToScalar(100)); 114 SkPaint boundPaint; 115 boundPaint.setColor(SK_ColorRED); 116 boundPaint.setStyle(SkPaint::kStroke_Style); 117 canvas->drawRect(bounds, boundPaint); 118 canvas->saveLayer(&bounds, nullptr); 119 } 120 121 for (int invBits = 0; invBits < 4; ++invBits) { 122 canvas->save(); 123 for (size_t op = 0; op < SK_ARRAY_COUNT(gOps); ++op) { 124 this->drawHairlines(canvas, path, clipA, clipB); 125 126 bool doInvA = SkToBool(invBits & 1); 127 bool doInvB = SkToBool(invBits & 2); 128 canvas->save(); 129 // set clip 130 clipA.setFillType(doInvA ? SkPath::kInverseEvenOdd_FillType : 131 SkPath::kEvenOdd_FillType); 132 clipB.setFillType(doInvB ? SkPath::kInverseEvenOdd_FillType : 133 SkPath::kEvenOdd_FillType); 134 canvas->clipPath(clipA, SkRegion::kIntersect_Op, fDoAAClip); 135 canvas->clipPath(clipB, gOps[op].fOp, fDoAAClip); 136 137 // In the inverse case we need to prevent the draw from covering the whole 138 // canvas. 139 if (fInvertDraw) { 140 SkRect rectClip = clipA.getBounds(); 141 rectClip.join(path.getBounds()); 142 rectClip.join(path.getBounds()); 143 rectClip.outset(5, 5); 144 canvas->clipRect(rectClip); 145 } 146 147 // draw path clipped 148 canvas->drawPath(path, pathPaint); 149 canvas->restore(); 150 151 152 SkScalar txtX = SkIntToScalar(45); 153 paint.setColor(gClipAColor); 154 const char* aTxt = doInvA ? "InvA " : "A "; 155 canvas->drawText(aTxt, strlen(aTxt), txtX, SkIntToScalar(220), paint); 156 txtX += paint.measureText(aTxt, strlen(aTxt)); 157 paint.setColor(SK_ColorBLACK); 158 canvas->drawText(gOps[op].fName, strlen(gOps[op].fName), 159 txtX, SkIntToScalar(220), paint); 160 txtX += paint.measureText(gOps[op].fName, strlen(gOps[op].fName)); 161 paint.setColor(gClipBColor); 162 const char* bTxt = doInvB ? "InvB " : "B "; 163 canvas->drawText(bTxt, strlen(bTxt), txtX, SkIntToScalar(220), paint); 164 165 canvas->translate(SkIntToScalar(250),0); 166 } 167 canvas->restore(); 168 canvas->translate(0, SkIntToScalar(250)); 169 } 170 171 if (fDoSaveLayer) { 172 canvas->restore(); 173 } 174 } 175 private: drawHairlines(SkCanvas * canvas,const SkPath & path,const SkPath & clipA,const SkPath & clipB)176 void drawHairlines(SkCanvas* canvas, const SkPath& path, 177 const SkPath& clipA, const SkPath& clipB) { 178 SkPaint paint; 179 paint.setAntiAlias(true); 180 paint.setStyle(SkPaint::kStroke_Style); 181 const SkAlpha fade = 0x33; 182 183 // draw path in hairline 184 paint.setColor(gPathColor); paint.setAlpha(fade); 185 canvas->drawPath(path, paint); 186 187 // draw clips in hair line 188 paint.setColor(gClipAColor); paint.setAlpha(fade); 189 canvas->drawPath(clipA, paint); 190 paint.setColor(gClipBColor); paint.setAlpha(fade); 191 canvas->drawPath(clipB, paint); 192 } 193 194 bool fDoAAClip; 195 bool fDoSaveLayer; 196 bool fInvertDraw; 197 198 typedef GM INHERITED; 199 }; 200 201 ////////////////////////////////////////////////////////////////////////////// 202 203 DEF_GM(return new ComplexClipGM(false, false, false);) 204 DEF_GM(return new ComplexClipGM(false, false, true);) 205 DEF_GM(return new ComplexClipGM(false, true, false);) 206 DEF_GM(return new ComplexClipGM(false, true, true);) 207 DEF_GM(return new ComplexClipGM(true, false, false);) 208 DEF_GM(return new ComplexClipGM(true, false, true);) 209 DEF_GM(return new ComplexClipGM(true, true, false);) 210 DEF_GM(return new ComplexClipGM(true, true, true);) 211 } 212