| /* |
| * Copyright 2013 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/SkColor.h" |
| #include "include/core/SkImage.h" |
| #include "include/core/SkPaint.h" |
| #include "include/core/SkRRect.h" |
| #include "include/core/SkRect.h" |
| #include "include/core/SkSize.h" |
| #include "include/core/SkString.h" |
| #include "include/core/SkSurface.h" |
| #include "include/core/SkTypes.h" |
| |
| namespace skiagm { |
| |
| // Draw various width thin rects at 1/8 horizontal pixel increments |
| class ThinRectsGM : public GM { |
| public: |
| ThinRectsGM(bool round) : fRound(round) { |
| this->setBGColor(0xFF000000); |
| } |
| |
| protected: |
| SkString onShortName() override { |
| return SkString(fRound ? "thinroundrects" : "thinrects"); |
| } |
| |
| SkISize onISize() override { |
| return SkISize::Make(240, 320); |
| } |
| |
| void onDraw(SkCanvas* canvas) override { |
| |
| SkPaint white; |
| white.setColor(SK_ColorWHITE); |
| white.setAntiAlias(true); |
| |
| SkPaint green; |
| green.setColor(SK_ColorGREEN); |
| green.setAntiAlias(true); |
| |
| for (int i = 0; i < 8; ++i) { |
| canvas->save(); |
| canvas->translate(i*0.125f, i*40.0f); |
| this->drawVertRects(canvas, white); |
| |
| canvas->translate(40.0f, 0.0f); |
| this->drawVertRects(canvas, green); |
| canvas->restore(); |
| |
| canvas->save(); |
| canvas->translate(80.0f, i*40.0f + i*0.125f); |
| this->drawHorizRects(canvas, white); |
| |
| canvas->translate(40.0f, 0.0f); |
| this->drawHorizRects(canvas, green); |
| canvas->restore(); |
| |
| canvas->save(); |
| canvas->translate(160.0f + i*0.125f, |
| i*40.0f + i*0.125f); |
| this->drawSquares(canvas, white); |
| |
| canvas->translate(40.0f, 0.0f); |
| this->drawSquares(canvas, green); |
| canvas->restore(); |
| } |
| } |
| |
| private: |
| void drawVertRects(SkCanvas* canvas, const SkPaint& p) { |
| constexpr SkRect vertRects[] = { |
| { 1, 1, 5.0f, 21 }, // 4 pix wide |
| { 8, 1, 10.0f, 21 }, // 2 pix wide |
| { 13, 1, 14.0f, 21 }, // 1 pix wide |
| { 17, 1, 17.5f, 21 }, // 1/2 pix wide |
| { 21, 1, 21.25f, 21 }, // 1/4 pix wide |
| { 25, 1, 25.125f, 21 }, // 1/8 pix wide |
| { 29, 1, 29.0f, 21 } // 0 pix wide |
| }; |
| |
| static constexpr SkVector radii[4] = {{1/32.f, 2/32.f}, {3/32.f, 1/32.f}, {2/32.f, 3/32.f}, |
| {1/32.f, 3/32.f}}; |
| SkRRect rrect; |
| for (size_t j = 0; j < SK_ARRAY_COUNT(vertRects); ++j) { |
| if (fRound) { |
| rrect.setRectRadii(vertRects[j], radii); |
| canvas->drawRRect(rrect, p); |
| } else { |
| canvas->drawRect(vertRects[j], p); |
| } |
| } |
| } |
| |
| void drawHorizRects(SkCanvas* canvas, const SkPaint& p) { |
| constexpr SkRect horizRects[] = { |
| { 1, 1, 21, 5.0f }, // 4 pix high |
| { 1, 8, 21, 10.0f }, // 2 pix high |
| { 1, 13, 21, 14.0f }, // 1 pix high |
| { 1, 17, 21, 17.5f }, // 1/2 pix high |
| { 1, 21, 21, 21.25f }, // 1/4 pix high |
| { 1, 25, 21, 25.125f }, // 1/8 pix high |
| { 1, 29, 21, 29.0f } // 0 pix high |
| }; |
| |
| SkRRect rrect; |
| for (size_t j = 0; j < SK_ARRAY_COUNT(horizRects); ++j) { |
| if (fRound) { |
| rrect.setNinePatch(horizRects[j], 1/32.f, 2/32.f, 3/32.f, 4/32.f); |
| canvas->drawRRect(rrect, p); |
| } else { |
| canvas->drawRect(horizRects[j], p); |
| } |
| } |
| } |
| |
| void drawSquares(SkCanvas* canvas, const SkPaint& p) { |
| constexpr SkRect squares[] = { |
| { 1, 1, 5.0f, 5.0f }, // 4 pix |
| { 8, 8, 10.0f, 10.0f }, // 2 pix |
| { 13, 13, 14.0f, 14.0f }, // 1 pix |
| { 17, 17, 17.5f, 17.5f }, // 1/2 pix |
| { 21, 21, 21.25f, 21.25f }, // 1/4 pix |
| { 25, 25, 25.125f, 25.125f }, // 1/8 pix |
| { 29, 29, 29.0f, 29.0f } // 0 pix |
| }; |
| |
| SkRRect rrect; |
| for (size_t j = 0; j < SK_ARRAY_COUNT(squares); ++j) { |
| if (fRound) { |
| rrect.setRectXY(squares[j], 1/32.f, 2/32.f); |
| canvas->drawRRect(rrect, p); |
| } else { |
| canvas->drawRect(squares[j], p); |
| } |
| } |
| } |
| |
| const bool fRound; |
| |
| using INHERITED = GM; |
| }; |
| |
| ////////////////////////////////////////////////////////////////////////////// |
| |
| DEF_GM( return new ThinRectsGM(false); ) |
| DEF_GM( return new ThinRectsGM(true); ) |
| |
| } // namespace skiagm |
| |
| DEF_SIMPLE_GM_CAN_FAIL(clipped_thinrect, canvas, errorMsg, 256, 256) { |
| auto zoomed = canvas->makeSurface(canvas->imageInfo().makeWH(10, 10)); |
| if (!zoomed) { |
| errorMsg->printf("makeSurface not supported"); |
| return skiagm::DrawResult::kSkip; |
| } |
| auto zoomedCanvas = zoomed->getCanvas(); |
| |
| SkPaint p; |
| p.setColor(SK_ColorRED); |
| p.setAntiAlias(true); |
| p.setStyle(SkPaint::kFill_Style); |
| zoomedCanvas->save(); |
| zoomedCanvas->clipRect(SkRect::MakeXYWH(0, 5, 256, 10), true /*doAntialias*/); |
| zoomedCanvas->drawRect(SkRect::MakeXYWH(0, 0, 100, 5.5), p); |
| zoomedCanvas->restore(); |
| |
| // Zoom-in. Should see one line of red representing zoomed in 1/2px coverage and *not* |
| // two lines of varying coverage from hairline rendering. |
| auto img = zoomed->makeImageSnapshot(); |
| canvas->drawImageRect(img, SkRect::MakeXYWH(0, 10, 200, 200), SkSamplingOptions()); |
| return skiagm::DrawResult::kOk; |
| } |