| /* |
| * 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/gm.h" |
| #include "include/core/SkCanvas.h" |
| #include "include/core/SkClipOp.h" |
| #include "include/core/SkColor.h" |
| #include "include/core/SkPaint.h" |
| #include "include/core/SkPath.h" |
| #include "include/core/SkRRect.h" |
| #include "include/core/SkRect.h" |
| #include "include/core/SkScalar.h" |
| #include "include/core/SkSize.h" |
| #include "include/core/SkString.h" |
| #include "include/core/SkTypes.h" |
| #include "include/utils/SkRandom.h" |
| #include "src/core/SkClipOpPriv.h" |
| |
| namespace skiagm { |
| |
| class ComplexClip2GM : public GM { |
| public: |
| enum Clip { |
| kRect_Clip, |
| kRRect_Clip, |
| kPath_Clip |
| }; |
| |
| ComplexClip2GM(Clip clip, bool antiAlias) |
| : fClip(clip) |
| , fAntiAlias(antiAlias) { |
| SkScalar xA = 0.65f; |
| SkScalar xF = 50.65f; |
| |
| SkScalar yA = 0.65f; |
| SkScalar yF = 50.65f; |
| |
| fWidth = xF - xA; |
| fHeight = yF - yA; |
| |
| fTotalWidth = kCols * fWidth + SK_Scalar1 * (kCols + 1) * kPadX; |
| fTotalHeight = kRows * fHeight + SK_Scalar1 * (kRows + 1) * kPadY; |
| } |
| |
| protected: |
| void onOnceBeforeDraw() override { |
| this->setBGColor(SkColorSetRGB(0xDD,0xA0,0xDD)); |
| |
| // offset the rects a bit so we get antialiasing even in the rect case |
| SkScalar xA = 0.65f; |
| SkScalar xB = 10.65f; |
| SkScalar xC = 20.65f; |
| SkScalar xD = 30.65f; |
| SkScalar xE = 40.65f; |
| SkScalar xF = 50.65f; |
| |
| SkScalar yA = 0.65f; |
| SkScalar yB = 10.65f; |
| SkScalar yC = 20.65f; |
| SkScalar yD = 30.65f; |
| SkScalar yE = 40.65f; |
| SkScalar yF = 50.65f; |
| |
| fRects[0].setLTRB(xB, yB, xE, yE); |
| fRRects[0].setRectXY(fRects[0], 7, 7); |
| fPaths[0].addRoundRect(fRects[0], 5, 5); |
| fRectColors[0] = SK_ColorRED; |
| |
| fRects[1].setLTRB(xA, yA, xD, yD); |
| fRRects[1].setRectXY(fRects[1], 7, 7); |
| fPaths[1].addRoundRect(fRects[1], 5, 5); |
| fRectColors[1] = SK_ColorGREEN; |
| |
| fRects[2].setLTRB(xC, yA, xF, yD); |
| fRRects[2].setRectXY(fRects[2], 7, 7); |
| fPaths[2].addRoundRect(fRects[2], 5, 5); |
| fRectColors[2] = SK_ColorBLUE; |
| |
| fRects[3].setLTRB(xA, yC, xD, yF); |
| fRRects[3].setRectXY(fRects[3], 7, 7); |
| fPaths[3].addRoundRect(fRects[3], 5, 5); |
| fRectColors[3] = SK_ColorYELLOW; |
| |
| fRects[4].setLTRB(xC, yC, xF, yF); |
| fRRects[4].setRectXY(fRects[4], 7, 7); |
| fPaths[4].addRoundRect(fRects[4], 5, 5); |
| fRectColors[4] = SK_ColorCYAN; |
| |
| const SkClipOp ops[] = { |
| kDifference_SkClipOp, |
| kIntersect_SkClipOp, |
| kUnion_SkClipOp, |
| kXOR_SkClipOp, |
| kReverseDifference_SkClipOp, |
| kReplace_SkClipOp, |
| }; |
| |
| SkRandom r; |
| for (int i = 0; i < kRows; ++i) { |
| for (int j = 0; j < kCols; ++j) { |
| for (int k = 0; k < 5; ++k) { |
| fOps[j*kRows+i][k] = ops[r.nextU() % SK_ARRAY_COUNT(ops)]; |
| } |
| } |
| } |
| } |
| |
| static constexpr int kRows = 5; |
| static constexpr int kCols = 5; |
| static constexpr int kPadX = 20; |
| static constexpr int kPadY = 20; |
| |
| static const char* ClipStr(Clip clip) { |
| switch (clip) { |
| case kRect_Clip: |
| return "rect"; |
| case kRRect_Clip: |
| return "rrect"; |
| case kPath_Clip: |
| return "path"; |
| } |
| SkDEBUGFAIL("Unknown clip type."); |
| return ""; |
| } |
| |
| SkString onShortName() override { |
| if (kRect_Clip == fClip && !fAntiAlias) { |
| return SkString("complexclip2"); |
| } |
| |
| SkString str; |
| str.printf("complexclip2_%s_%s", |
| ClipStr(fClip), |
| fAntiAlias ? "aa" : "bw"); |
| return str; |
| } |
| |
| SkISize onISize() override { |
| return SkISize::Make(SkScalarRoundToInt(fTotalWidth), |
| SkScalarRoundToInt(fTotalHeight)); |
| } |
| |
| void onDraw(SkCanvas* canvas) override { |
| SkPaint rectPaint; |
| rectPaint.setStyle(SkPaint::kStroke_Style); |
| rectPaint.setStrokeWidth(-1); |
| |
| SkPaint fillPaint; |
| fillPaint.setColor(SkColorSetRGB(0xA0,0xDD,0xA0)); |
| |
| for (int i = 0; i < kRows; ++i) { |
| for (int j = 0; j < kCols; ++j) { |
| canvas->save(); |
| |
| canvas->translate(kPadX * SK_Scalar1 + (fWidth + kPadX * SK_Scalar1)*j, |
| kPadY * SK_Scalar1 + (fHeight + kPadY * SK_Scalar1)*i); |
| |
| // draw the original shapes first so we can see the |
| // antialiasing on the clipped draw |
| for (int k = 0; k < 5; ++k) { |
| rectPaint.setColor(fRectColors[k]); |
| switch (fClip) { |
| case kRect_Clip: |
| canvas->drawRect(fRects[k], rectPaint); |
| break; |
| case kRRect_Clip: |
| canvas->drawRRect(fRRects[k], rectPaint); |
| break; |
| case kPath_Clip: |
| canvas->drawPath(fPaths[k], rectPaint); |
| break; |
| } |
| } |
| |
| for (int k = 0; k < 5; ++k) { |
| switch (fClip) { |
| case kRect_Clip: |
| canvas->clipRect(fRects[k], |
| fOps[j*kRows+i][k], |
| fAntiAlias); |
| break; |
| case kRRect_Clip: |
| canvas->clipRRect(fRRects[k], |
| fOps[j*kRows+i][k], |
| fAntiAlias); |
| break; |
| case kPath_Clip: |
| canvas->clipPath(fPaths[k], |
| fOps[j*kRows+i][k], |
| fAntiAlias); |
| break; |
| } |
| } |
| canvas->drawRect(SkRect::MakeWH(fWidth, fHeight), fillPaint); |
| canvas->restore(); |
| } |
| } |
| } |
| private: |
| Clip fClip; |
| bool fAntiAlias; |
| SkRect fRects[5]; |
| SkRRect fRRects[5]; |
| SkPath fPaths[5]; |
| SkColor fRectColors[5]; |
| SkClipOp fOps[kRows * kCols][5]; |
| SkScalar fWidth; |
| SkScalar fHeight; |
| SkScalar fTotalWidth; |
| SkScalar fTotalHeight; |
| |
| typedef GM INHERITED; |
| }; |
| |
| ////////////////////////////////////////////////////////////////////////////// |
| |
| // bw |
| DEF_GM( return new ComplexClip2GM(ComplexClip2GM::kRect_Clip, false); ) |
| DEF_GM( return new ComplexClip2GM(ComplexClip2GM::kRRect_Clip, false); ) |
| DEF_GM( return new ComplexClip2GM(ComplexClip2GM::kPath_Clip, false); ) |
| |
| // aa |
| DEF_GM( return new ComplexClip2GM(ComplexClip2GM::kRect_Clip, true); ) |
| DEF_GM( return new ComplexClip2GM(ComplexClip2GM::kRRect_Clip, true); ) |
| DEF_GM( return new ComplexClip2GM(ComplexClip2GM::kPath_Clip, true); ) |
| |
| } |