reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2015 Google Inc. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license that can be |
| 5 | * found in the LICENSE file. |
| 6 | */ |
| 7 | |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 8 | #include "gm/gm.h" |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 9 | #include "include/core/SkBitmap.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 10 | #include "include/core/SkCanvas.h" |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 11 | #include "include/core/SkColorSpace.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 12 | #include "include/core/SkImage.h" |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 13 | #include "include/core/SkImageInfo.h" |
| 14 | #include "include/core/SkPaint.h" |
| 15 | #include "include/core/SkRect.h" |
| 16 | #include "include/core/SkRefCnt.h" |
| 17 | #include "include/core/SkString.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 18 | #include "include/core/SkSurface.h" |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 19 | #include "include/core/SkTypes.h" |
reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 20 | |
reed | 9ce9d67 | 2016-03-17 10:51:11 -0700 | [diff] [blame] | 21 | static sk_sp<SkImage> make_image() { |
reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 22 | const SkImageInfo info = SkImageInfo::MakeN32Premul(319, 52); |
Kevin Lubick | 5c93acf | 2023-05-09 12:11:43 -0400 | [diff] [blame] | 23 | auto surface(SkSurfaces::Raster(info)); |
reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 24 | SkCanvas* canvas = surface->getCanvas(); |
Mike Klein | d46dce3 | 2018-08-16 10:17:03 -0400 | [diff] [blame] | 25 | canvas->drawColor(0xFFF8F8F8); |
reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 26 | |
| 27 | SkPaint paint; |
| 28 | paint.setAntiAlias(true); |
| 29 | |
| 30 | paint.setStyle(SkPaint::kStroke_Style); |
| 31 | for (int i = 0; i < 20; ++i) { |
| 32 | canvas->drawCircle(-4, 25, 20, paint); |
| 33 | canvas->translate(25, 0); |
| 34 | } |
reed | 9ce9d67 | 2016-03-17 10:51:11 -0700 | [diff] [blame] | 35 | return surface->makeImageSnapshot(); |
reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 36 | } |
| 37 | |
halcanary | 2a24338 | 2015-09-09 08:16:41 -0700 | [diff] [blame] | 38 | DEF_SIMPLE_GM(mipmap, canvas, 400, 200) { |
reed | 9ce9d67 | 2016-03-17 10:51:11 -0700 | [diff] [blame] | 39 | sk_sp<SkImage> img(make_image());//SkImage::NewFromEncoded(data)); |
reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 40 | |
reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 41 | const SkRect dst = SkRect::MakeWH(177, 15); |
| 42 | |
reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 43 | SkString str; |
| 44 | str.printf("scale %g %g", dst.width() / img->width(), dst.height() / img->height()); |
Hal Canary | df2d27e | 2019-01-08 09:38:02 -0500 | [diff] [blame] | 45 | // canvas->drawString(str, 300, 100, SkFont(nullptr, 30), paint); |
reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 46 | |
Mike Reed | 07c5f52 | 2021-01-23 12:23:23 -0500 | [diff] [blame] | 47 | const SkSamplingOptions samplings[] = { |
| 48 | SkSamplingOptions(SkFilterMode::kNearest), |
| 49 | SkSamplingOptions(SkFilterMode::kLinear), |
| 50 | SkSamplingOptions(SkFilterMode::kLinear, SkMipmapMode::kLinear), |
Mike Reed | f3ac2af | 2021-02-05 12:55:38 -0500 | [diff] [blame] | 51 | SkSamplingOptions(SkCubicResampler::Mitchell()), |
Mike Reed | 07c5f52 | 2021-01-23 12:23:23 -0500 | [diff] [blame] | 52 | }; |
| 53 | |
reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 54 | canvas->translate(20, 20); |
Herb Derby | c37b386 | 2022-06-21 09:49:17 -0400 | [diff] [blame] | 55 | for (size_t i = 0; i < std::size(samplings); ++i) { |
Mike Reed | 07c5f52 | 2021-01-23 12:23:23 -0500 | [diff] [blame] | 56 | canvas->drawImageRect(img.get(), dst, samplings[i], nullptr); |
reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 57 | canvas->translate(0, 20); |
| 58 | } |
Mike Reed | 07c5f52 | 2021-01-23 12:23:23 -0500 | [diff] [blame] | 59 | canvas->drawImage(img.get(), 20, 20); |
reed | c8e4765 | 2015-02-19 11:39:46 -0800 | [diff] [blame] | 60 | } |
reed | 6644d93 | 2016-06-10 11:41:47 -0700 | [diff] [blame] | 61 | |
| 62 | /////////////////////////////////////////////////////////////////////////////////////////////////// |
| 63 | |
| 64 | // create a circle image computed raw, so we can wrap it as a linear or srgb image |
| 65 | static sk_sp<SkImage> make(sk_sp<SkColorSpace> cs) { |
| 66 | const int N = 100; |
| 67 | SkImageInfo info = SkImageInfo::Make(N, N, kN32_SkColorType, kPremul_SkAlphaType, cs); |
| 68 | SkBitmap bm; |
| 69 | bm.allocPixels(info); |
| 70 | |
| 71 | for (int y = 0; y < N; ++y) { |
| 72 | for (int x = 0; x < N; ++x) { |
| 73 | *bm.getAddr32(x, y) = (x ^ y) & 1 ? 0xFFFFFFFF : 0xFF000000; |
| 74 | } |
| 75 | } |
| 76 | bm.setImmutable(); |
Mike Reed | ac9f0c9 | 2020-12-23 10:11:33 -0500 | [diff] [blame] | 77 | return bm.asImage(); |
reed | 6644d93 | 2016-06-10 11:41:47 -0700 | [diff] [blame] | 78 | } |
| 79 | |
| 80 | static void show_mips(SkCanvas* canvas, SkImage* img) { |
Mike Reed | 07c5f52 | 2021-01-23 12:23:23 -0500 | [diff] [blame] | 81 | SkSamplingOptions sampling(SkFilterMode::kLinear, |
| 82 | SkMipmapMode::kLinear); |
reed | 6644d93 | 2016-06-10 11:41:47 -0700 | [diff] [blame] | 83 | |
reed | 3cc37d3 | 2016-06-11 04:48:12 -0700 | [diff] [blame] | 84 | // Want to ensure we never draw fractional pixels, so we use an IRect |
| 85 | SkIRect dst = SkIRect::MakeWH(img->width(), img->height()); |
reed | 6644d93 | 2016-06-10 11:41:47 -0700 | [diff] [blame] | 86 | while (dst.width() > 5) { |
Mike Reed | 07c5f52 | 2021-01-23 12:23:23 -0500 | [diff] [blame] | 87 | canvas->drawImageRect(img, SkRect::Make(dst), sampling, nullptr); |
reed | 6644d93 | 2016-06-10 11:41:47 -0700 | [diff] [blame] | 88 | dst.offset(dst.width() + 10, 0); |
reed | 3cc37d3 | 2016-06-11 04:48:12 -0700 | [diff] [blame] | 89 | dst.fRight = dst.fLeft + dst.width()/2; |
| 90 | dst.fBottom = dst.fTop + dst.height()/2; |
reed | 6644d93 | 2016-06-10 11:41:47 -0700 | [diff] [blame] | 91 | } |
| 92 | } |
| 93 | |
| 94 | /* |
| 95 | * Ensure that in L32 drawing mode, both images/mips look the same as each other, and |
| 96 | * their mips are darker than the original (since the mips should ignore the gamma in L32). |
| 97 | * |
| 98 | * Ensure that in S32 drawing mode, all images/mips look the same, and look correct (i.e. |
| 99 | * the mip levels match the original in brightness). |
| 100 | */ |
| 101 | DEF_SIMPLE_GM(mipmap_srgb, canvas, 260, 230) { |
| 102 | sk_sp<SkImage> limg = make(nullptr); |
Matt Sarett | 77a7a1b | 2017-02-07 13:56:11 -0500 | [diff] [blame] | 103 | sk_sp<SkImage> simg = make(SkColorSpace::MakeSRGB()); |
reed | 6644d93 | 2016-06-10 11:41:47 -0700 | [diff] [blame] | 104 | |
| 105 | canvas->translate(10, 10); |
| 106 | show_mips(canvas, limg.get()); |
| 107 | canvas->translate(0, limg->height() + 10.0f); |
| 108 | show_mips(canvas, simg.get()); |
| 109 | } |
| 110 | |
brianosman | 79b15f6 | 2016-06-21 13:40:12 -0700 | [diff] [blame] | 111 | /////////////////////////////////////////////////////////////////////////////////////////////////// |
| 112 | |
| 113 | // create a gradient image computed raw, so we can wrap it as a linear or srgb image |
| 114 | static sk_sp<SkImage> make_g8_gradient(sk_sp<SkColorSpace> cs) { |
| 115 | const int N = 100; |
| 116 | SkImageInfo info = SkImageInfo::Make(N, N, kGray_8_SkColorType, kOpaque_SkAlphaType, cs); |
| 117 | SkBitmap bm; |
| 118 | bm.allocPixels(info); |
| 119 | |
| 120 | for (int y = 0; y < N; ++y) { |
| 121 | for (int x = 0; x < N; ++x) { |
| 122 | *bm.getAddr8(x, y) = static_cast<uint8_t>(255.0f * ((x + y) / (2.0f * (N - 1)))); |
| 123 | } |
| 124 | } |
| 125 | bm.setImmutable(); |
Mike Reed | ac9f0c9 | 2020-12-23 10:11:33 -0500 | [diff] [blame] | 126 | return bm.asImage(); |
brianosman | 79b15f6 | 2016-06-21 13:40:12 -0700 | [diff] [blame] | 127 | } |
| 128 | |
| 129 | static void show_mips_only(SkCanvas* canvas, SkImage* img) { |
Mike Reed | 07c5f52 | 2021-01-23 12:23:23 -0500 | [diff] [blame] | 130 | SkSamplingOptions sampling(SkFilterMode::kLinear, |
| 131 | SkMipmapMode::kLinear); |
brianosman | 79b15f6 | 2016-06-21 13:40:12 -0700 | [diff] [blame] | 132 | |
| 133 | // Want to ensure we never draw fractional pixels, so we use an IRect |
| 134 | SkIRect dst = SkIRect::MakeWH(img->width() / 2, img->height() / 2); |
| 135 | while (dst.width() > 5) { |
Mike Reed | 07c5f52 | 2021-01-23 12:23:23 -0500 | [diff] [blame] | 136 | canvas->drawImageRect(img, SkRect::Make(dst), sampling, nullptr); |
brianosman | 79b15f6 | 2016-06-21 13:40:12 -0700 | [diff] [blame] | 137 | dst.offset(dst.width() + 10, 0); |
| 138 | dst.fRight = dst.fLeft + dst.width() / 2; |
| 139 | dst.fBottom = dst.fTop + dst.height() / 2; |
| 140 | } |
| 141 | } |
| 142 | |
| 143 | /* |
| 144 | * Ensure that in L32 drawing mode, both images/mips look the same as each other, and |
| 145 | * their mips are darker than the original (since the mips should ignore the gamma in L32). |
| 146 | * |
| 147 | * Ensure that in S32 drawing mode, all images/mips look the same, and look correct (i.e. |
| 148 | * the mip levels match the original in brightness). |
brianosman | 79b15f6 | 2016-06-21 13:40:12 -0700 | [diff] [blame] | 149 | */ |
| 150 | DEF_SIMPLE_GM(mipmap_gray8_srgb, canvas, 260, 230) { |
| 151 | sk_sp<SkImage> limg = make_g8_gradient(nullptr); |
Matt Sarett | 77a7a1b | 2017-02-07 13:56:11 -0500 | [diff] [blame] | 152 | sk_sp<SkImage> simg = make_g8_gradient(SkColorSpace::MakeSRGB()); |
brianosman | 79b15f6 | 2016-06-21 13:40:12 -0700 | [diff] [blame] | 153 | |
| 154 | canvas->translate(10, 10); |
| 155 | show_mips_only(canvas, limg.get()); |
| 156 | canvas->translate(0, limg->height() + 10.0f); |
| 157 | show_mips_only(canvas, simg.get()); |
| 158 | } |