/* * Copyright 2015 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "SkBitmap.h" #include "SkCanvas.h" #include "SkPath.h" #include "SkRect.h" #include "SkRectPriv.h" #include "Test.h" static bool has_green_pixels(const SkBitmap& bm) { for (int j = 0; j < bm.height(); ++j) { for (int i = 0; i < bm.width(); ++i) { if (SkColorGetG(bm.getColor(i, j))) { return true; } } } return false; } static void test_stroke_width_clipping(skiatest::Reporter* reporter) { SkBitmap bm; bm.allocN32Pixels(100, 10); bm.eraseColor(SK_ColorTRANSPARENT); SkCanvas canvas(bm); SkPaint paint; paint.setStyle(SkPaint::kStroke_Style); paint.setStrokeWidth(10); paint.setColor(0xff00ff00); // clip out the left half of our canvas canvas.clipRect(SkRect::MakeXYWH(51, 0, 49, 100)); // no stroke bleed should be visible canvas.drawRect(SkRect::MakeWH(44, 100), paint); REPORTER_ASSERT(reporter, !has_green_pixels(bm)); // right stroke edge should bleed into the visible area canvas.scale(2, 2); canvas.drawRect(SkRect::MakeWH(22, 50), paint); REPORTER_ASSERT(reporter, has_green_pixels(bm)); } static void test_skbug4406(skiatest::Reporter* reporter) { SkBitmap bm; bm.allocN32Pixels(10, 10); bm.eraseColor(SK_ColorTRANSPARENT); SkCanvas canvas(bm); const SkRect r = { 1.5f, 1, 3.5f, 3 }; // draw filled green rect first SkPaint paint; paint.setStyle(SkPaint::kFill_Style); paint.setColor(0xff00ff00); paint.setStrokeWidth(1); paint.setAntiAlias(true); canvas.drawRect(r, paint); // paint black with stroke rect (that asserts in bug 4406) // over the filled rect, it should cover it paint.setStyle(SkPaint::kStroke_Style); paint.setColor(0xff000000); paint.setStrokeWidth(1); canvas.drawRect(r, paint); REPORTER_ASSERT(reporter, !has_green_pixels(bm)); // do it again with thinner stroke paint.setStyle(SkPaint::kFill_Style); paint.setColor(0xff00ff00); paint.setStrokeWidth(1); paint.setAntiAlias(true); canvas.drawRect(r, paint); // paint black with stroke rect (that asserts in bug 4406) // over the filled rect, it doesnt cover it completelly with thinner stroke paint.setStyle(SkPaint::kStroke_Style); paint.setColor(0xff000000); paint.setStrokeWidth(0.99f); canvas.drawRect(r, paint); REPORTER_ASSERT(reporter, has_green_pixels(bm)); } DEF_TEST(Rect, reporter) { test_stroke_width_clipping(reporter); test_skbug4406(reporter); } DEF_TEST(Rect_grow, reporter) { test_stroke_width_clipping(reporter); test_skbug4406(reporter); } DEF_TEST(Rect_path_nan, reporter) { SkRect r = { 0, 0, SK_ScalarNaN, 100 }; SkPath p; p.addRect(r); // path normally just jams its bounds to be r, but it must notice that r is non-finite REPORTER_ASSERT(reporter, !p.isFinite()); } DEF_TEST(Rect_largest, reporter) { REPORTER_ASSERT(reporter, !SkRectPriv::MakeILarge().isEmpty()); REPORTER_ASSERT(reporter, SkRectPriv::MakeILargestInverted().isEmpty()); REPORTER_ASSERT(reporter, !SkRectPriv::MakeLargest().isEmpty()); REPORTER_ASSERT(reporter, !SkRectPriv::MakeLargeS32().isEmpty()); REPORTER_ASSERT(reporter, SkRectPriv::MakeLargestInverted().isEmpty()); } /* * Test the setBounds always handles non-finite values correctly: * - setBoundsCheck should return false, and set the rect to all zeros * - setBoundsNoCheck should ensure that rect.isFinite() is false (definitely NOT all zeros) */ DEF_TEST(Rect_setbounds, reporter) { const SkPoint p0[] = { { SK_ScalarInfinity, 0 }, { 1, 1 }, { 2, 2 }, { 3, 3 } }; const SkPoint p1[] = { { 0, SK_ScalarInfinity }, { 1, 1 }, { 2, 2 }, { 3, 3 } }; const SkPoint p2[] = { { SK_ScalarNaN, 0 }, { 1, 1 }, { 2, 2 }, { 3, 3 } }; const SkPoint p3[] = { { 0, SK_ScalarNaN }, { 1, 1 }, { 2, 2 }, { 3, 3 } }; SkRect r; const SkRect zeror = { 0, 0, 0, 0 }; for (const SkPoint* pts : { p0, p1, p2, p3 }) { for (int n = 1; n <= 4; ++n) { bool isfinite = r.setBoundsCheck(pts, n); REPORTER_ASSERT(reporter, !isfinite); REPORTER_ASSERT(reporter, r == zeror); r.setBoundsNoCheck(pts, n); if (r.isFinite()) r.setBoundsNoCheck(pts, n); REPORTER_ASSERT(reporter, !r.isFinite()); } } } static float make_big_value(skiatest::Reporter* reporter) { // need to make a big value, one that will cause rect.width() to overflow to inf. // however, the windows compiler wants about this if it can see the big value inlined. // hence, this stupid trick to try to fool their compiler. SkASSERT(reporter); return reporter ? SK_ScalarMax * 0.75f : 0; } DEF_TEST(Rect_center, reporter) { // ensure we can compute center even when the width/height might overflow const SkScalar big = make_big_value(reporter); const SkRect r = { -big, -big, big, big }; REPORTER_ASSERT(reporter, r.isFinite()); REPORTER_ASSERT(reporter, SkScalarIsFinite(r.centerX())); REPORTER_ASSERT(reporter, SkScalarIsFinite(r.centerY())); REPORTER_ASSERT(reporter, !SkScalarIsFinite(r.width())); REPORTER_ASSERT(reporter, !SkScalarIsFinite(r.height())); }