/* * 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/SkBitmap.h" #include "include/core/SkBlendMode.h" #include "include/core/SkCanvas.h" #include "include/core/SkColor.h" #include "include/core/SkMatrix.h" #include "include/core/SkPaint.h" #include "include/core/SkPathBuilder.h" #include "include/core/SkRect.h" #include "include/core/SkScalar.h" #include "include/core/SkSize.h" #include "include/core/SkString.h" #include "src/core/SkCanvasPriv.h" #include "tools/ToolUtils.h" /** Draw a 2px border around the target, then red behind the target; set the clip to match the target, then draw >> the target in blue. */ static void draw(SkCanvas* canvas, SkRect& target, int x, int y) { SkPaint borderPaint; borderPaint.setColor(SkColorSetRGB(0x0, 0xDD, 0x0)); borderPaint.setAntiAlias(true); SkPaint backgroundPaint; backgroundPaint.setColor(SkColorSetRGB(0xDD, 0x0, 0x0)); backgroundPaint.setAntiAlias(true); SkPaint foregroundPaint; foregroundPaint.setColor(SkColorSetRGB(0x0, 0x0, 0xDD)); foregroundPaint.setAntiAlias(true); canvas->save(); canvas->translate(SkIntToScalar(x), SkIntToScalar(y)); target.inset(SkIntToScalar(-2), SkIntToScalar(-2)); canvas->drawRect(target, borderPaint); target.inset(SkIntToScalar(2), SkIntToScalar(2)); canvas->drawRect(target, backgroundPaint); canvas->clipRect(target, true); target.inset(SkIntToScalar(-4), SkIntToScalar(-4)); canvas->drawRect(target, foregroundPaint); canvas->restore(); } static void draw_square(SkCanvas* canvas, int x, int y) { SkRect target (SkRect::MakeWH(10 * SK_Scalar1, 10 * SK_Scalar1)); draw(canvas, target, x, y); } static void draw_column(SkCanvas* canvas, int x, int y) { SkRect target (SkRect::MakeWH(1 * SK_Scalar1, 10 * SK_Scalar1)); draw(canvas, target, x, y); } static void draw_bar(SkCanvas* canvas, int x, int y) { SkRect target (SkRect::MakeWH(10 * SK_Scalar1, 1 * SK_Scalar1)); draw(canvas, target, x, y); } static void draw_rect_tests(SkCanvas* canvas) { draw_square(canvas, 10, 10); draw_column(canvas, 30, 10); draw_bar(canvas, 10, 30); } /** Test a set of clipping problems discovered while writing blitAntiRect, and test all the code paths through the clipping blitters. Each region should show as a blue center surrounded by a 2px green border, with no red. */ DEF_SIMPLE_GM(aaclip, canvas, 240, 120) { // Initial pixel-boundary-aligned draw draw_rect_tests(canvas); // Repeat 4x with .2, .4, .6, .8 px offsets canvas->translate(SK_Scalar1 / 5, SK_Scalar1 / 5); canvas->translate(SkIntToScalar(50), 0); draw_rect_tests(canvas); canvas->translate(SK_Scalar1 / 5, SK_Scalar1 / 5); canvas->translate(SkIntToScalar(50), 0); draw_rect_tests(canvas); canvas->translate(SK_Scalar1 / 5, SK_Scalar1 / 5); canvas->translate(SkIntToScalar(50), 0); draw_rect_tests(canvas); canvas->translate(SK_Scalar1 / 5, SK_Scalar1 / 5); canvas->translate(SkIntToScalar(50), 0); draw_rect_tests(canvas); } ///////////////////////////////////////////////////////////////////////// #ifdef SK_BUILD_FOR_MAC #include "include/utils/mac/SkCGUtils.h" static std::unique_ptr make_canvas(const SkBitmap& bm) { return SkCanvas::MakeRasterDirect(bm.info(), bm.getPixels(), bm.rowBytes()); } static void test_image(SkCanvas* canvas, const SkImageInfo& info) { SkBitmap bm; bm.allocPixels(info); if (info.isOpaque()) { bm.eraseColor(SK_ColorGREEN); } else { bm.eraseColor(0); } SkPaint paint; paint.setAntiAlias(true); paint.setColor(SK_ColorBLUE); make_canvas(bm)->drawCircle(50, 50, 49, paint); canvas->drawImage(bm.asImage(), 10, 10); CGImageRef image = SkCreateCGImageRefWithColorspace(bm, nullptr); SkBitmap bm2; SkCreateBitmapFromCGImage(&bm2, image); canvas->drawImage(bm2.asImage(), 10, 120); canvas->drawImage(SkMakeImageFromCGImage(image), 10, 120 + bm2.height() + 10); CGImageRelease(image); } DEF_SIMPLE_GM(cgimage, canvas, 800, 250) { const struct { SkColorType fCT; SkAlphaType fAT; } rec[] = { { kRGB_565_SkColorType, kOpaque_SkAlphaType }, { kRGBA_8888_SkColorType, kPremul_SkAlphaType }, { kRGBA_8888_SkColorType, kUnpremul_SkAlphaType }, { kRGBA_8888_SkColorType, kOpaque_SkAlphaType }, { kBGRA_8888_SkColorType, kPremul_SkAlphaType }, { kBGRA_8888_SkColorType, kUnpremul_SkAlphaType }, { kBGRA_8888_SkColorType, kOpaque_SkAlphaType }, }; for (size_t i = 0; i < SK_ARRAY_COUNT(rec); ++i) { SkImageInfo info = SkImageInfo::Make(100, 100, rec[i].fCT, rec[i].fAT); test_image(canvas, info); canvas->translate(info.width() + 10, 0); } } #endif /////////////////////////////////////////////////////////////////////////////////////////////////// // https://bug.skia.org/3716 class ClipCubicGM : public skiagm::GM { const SkScalar W = 100; const SkScalar H = 240; SkPath fVPath, fHPath; public: ClipCubicGM() { fVPath = SkPathBuilder().moveTo(W, 0) .cubicTo(W, H-10, 0, 10, 0, H) .detach(); SkMatrix pivot; pivot.setRotate(90, W/2, H/2); fHPath = fVPath.makeTransform(pivot); } protected: SkString onShortName() override { return SkString("clipcubic"); } SkISize onISize() override { return SkISize::Make(400, 410); } void doDraw(SkCanvas* canvas, const SkPath& path) { SkPaint paint; paint.setAntiAlias(true); paint.setColor(0xFFCCCCCC); canvas->drawPath(path, paint); paint.setColor(SK_ColorRED); paint.setStyle(SkPaint::kStroke_Style); canvas->drawPath(path, paint); } void drawAndClip(SkCanvas* canvas, const SkPath& path, SkScalar dx, SkScalar dy) { SkAutoCanvasRestore acr(canvas, true); SkRect r = SkRect::MakeXYWH(0, H/4, W, H/2); SkPaint paint; paint.setColor(ToolUtils::color_to_565(0xFF8888FF)); canvas->drawRect(r, paint); this->doDraw(canvas, path); canvas->translate(dx, dy); canvas->drawRect(r, paint); canvas->clipRect(r); this->doDraw(canvas, path); } void onDraw(SkCanvas* canvas) override { canvas->translate(80, 10); this->drawAndClip(canvas, fVPath, 200, 0); canvas->translate(0, 200); this->drawAndClip(canvas, fHPath, 200, 0); } private: using INHERITED = skiagm::GM; }; DEF_GM(return new ClipCubicGM;)