blob: 703631ed084a9ac4117bf20fd5ce405bc8acc375 [file] [log] [blame]
bsalomone179a912016-01-20 06:18:10 -08001/*
2 * Copyright 2016 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
Brian Salomonf4ba4ec2020-03-19 15:54:28 -04008// This test only works with the GL backend.
bsalomone179a912016-01-20 06:18:10 -08009
Mike Kleinc0bd9f92019-04-23 12:05:21 -050010#include "gm/gm.h"
Brian Salomonf4ba4ec2020-03-19 15:54:28 -040011
12#ifdef SK_GL
Mike Kleinc0bd9f92019-04-23 12:05:21 -050013#include "include/core/SkBitmap.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040014#include "include/core/SkCanvas.h"
15#include "include/core/SkColor.h"
16#include "include/core/SkFilterQuality.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050017#include "include/core/SkImage.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040018#include "include/core/SkImageInfo.h"
19#include "include/core/SkPaint.h"
20#include "include/core/SkPoint.h"
21#include "include/core/SkRect.h"
22#include "include/core/SkRefCnt.h"
23#include "include/core/SkScalar.h"
24#include "include/core/SkShader.h"
25#include "include/core/SkSize.h"
26#include "include/core/SkString.h"
27#include "include/core/SkTileMode.h"
Robert Phillips08ba0852019-05-22 20:23:43 +000028#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050029#include "include/effects/SkGradientShader.h"
30#include "include/gpu/GrBackendSurface.h"
Robert Phillipsb87b39b2020-07-01 14:45:24 -040031#include "include/gpu/GrDirectContext.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040032#include "include/gpu/GrTypes.h"
Brian Salomon0f396992020-06-19 19:51:21 -040033#include "src/core/SkAutoPixmapStorage.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050034#include "src/gpu/GrContextPriv.h"
35#include "src/gpu/GrGpu.h"
Brian Salomon0f396992020-06-19 19:51:21 -040036#include "src/gpu/gl/GrGLCaps.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040037#include "src/gpu/gl/GrGLDefines.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040038
39#include <algorithm>
40#include <cstdint>
41#include <memory>
42
43class GrRenderTargetContext;
bsalomone179a912016-01-20 06:18:10 -080044
45namespace skiagm {
Chris Dalton3a778372019-02-07 15:23:36 -070046class RectangleTexture : public GpuGM {
bsalomone179a912016-01-20 06:18:10 -080047public:
48 RectangleTexture() {
49 this->setBGColor(0xFFFFFFFF);
50 }
51
Brian Salomon3fcf83a2020-02-23 21:29:01 -050052private:
53 enum class ImageType {
54 kGradientCircle,
55 k2x2
56 };
57
bsalomone179a912016-01-20 06:18:10 -080058 SkString onShortName() override {
59 return SkString("rectangle_texture");
60 }
61
Brian Salomon3fcf83a2020-02-23 21:29:01 -050062 SkISize onISize() override { return SkISize::Make(1180, 710); }
bsalomone179a912016-01-20 06:18:10 -080063
Brian Salomon3fcf83a2020-02-23 21:29:01 -050064 SkBitmap makeImagePixels(int size, ImageType type) {
65 auto ii = SkImageInfo::Make(size, size, kRGBA_8888_SkColorType, kOpaque_SkAlphaType);
66 switch (type) {
67 case ImageType::kGradientCircle: {
68 SkBitmap bmp;
69 bmp.allocPixels(ii);
70 SkPaint paint;
71 SkCanvas canvas(bmp);
72 SkPoint pts[] = {{0, 0}, {0, SkIntToScalar(size)}};
73 SkColor colors0[] = {0xFF1060B0, 0xFF102030};
74 paint.setShader(
75 SkGradientShader::MakeLinear(pts, colors0, nullptr, 2, SkTileMode::kClamp));
76 canvas.drawPaint(paint);
77 SkColor colors1[] = {0xFFA07010, 0xFFA02080};
78 paint.setAntiAlias(true);
79 paint.setShader(
80 SkGradientShader::MakeLinear(pts, colors1, nullptr, 2, SkTileMode::kClamp));
81 canvas.drawCircle(size/2.f, size/2.f, 2.f*size/5, paint);
82 return bmp;
83 }
84 case ImageType::k2x2: {
85 SkBitmap bmp;
86 bmp.allocPixels(ii);
87 *bmp.getAddr32(0, 0) = 0xFF0000FF;
88 *bmp.getAddr32(1, 0) = 0xFF00FF00;
89 *bmp.getAddr32(0, 1) = 0xFFFF0000;
90 *bmp.getAddr32(1, 1) = 0xFFFFFFFF;
91 return bmp;
92 }
93 }
94 SkUNREACHABLE;
bsalomone179a912016-01-20 06:18:10 -080095 }
96
Robert Phillipsb87b39b2020-07-01 14:45:24 -040097 sk_sp<SkImage> createRectangleTextureImg(GrDirectContext* context, GrSurfaceOrigin origin,
Brian Salomon3fcf83a2020-02-23 21:29:01 -050098 const SkBitmap content) {
99 SkASSERT(content.colorType() == kRGBA_8888_SkColorType);
Brian Salomon0f396992020-06-19 19:51:21 -0400100 auto format = GrBackendFormat::MakeGL(GR_GL_RGBA8, GR_GL_TEXTURE_RECTANGLE);
101 auto bet = context->createBackendTexture(content.width(), content.height(), format,
102 GrMipMapped::kNo, GrRenderable::kNo);
103 if (!bet.isValid()) {
Mike Klein16b1efb2019-04-02 10:01:11 -0400104 return nullptr;
105 }
Brian Salomon0f396992020-06-19 19:51:21 -0400106 const SkPixmap* pm = &content.pixmap();
107 SkAutoPixmapStorage tempPM;
Brian Salomon246bc3d2018-12-06 15:33:02 -0500108 if (origin == kBottomLeft_GrSurfaceOrigin) {
Brian Salomon0f396992020-06-19 19:51:21 -0400109 tempPM.alloc(pm->info());
110 const uint32_t* src = pm->addr32();
111 uint32_t* dst = tempPM.writable_addr32(0, content.height() - 1);
112 for (int y = 0; y < content.height(); ++y,
113 src += pm->rowBytesAsPixels(),
114 dst -= tempPM.rowBytesAsPixels()) {
115 std::copy_n(src, content.width(), dst);
Brian Salomon246bc3d2018-12-06 15:33:02 -0500116 }
Brian Salomon0f396992020-06-19 19:51:21 -0400117 pm = &tempPM;
Brian Salomon246bc3d2018-12-06 15:33:02 -0500118 }
Brian Salomon0f396992020-06-19 19:51:21 -0400119 if (!context->updateBackendTexture(bet, pm, 1, nullptr, nullptr)) {
120 context->deleteBackendTexture(bet);
bsalomone179a912016-01-20 06:18:10 -0800121 }
Brian Salomon0f396992020-06-19 19:51:21 -0400122 return SkImage::MakeFromAdoptedTexture(context, bet, origin, kRGBA_8888_SkColorType);
bsalomone179a912016-01-20 06:18:10 -0800123 }
124
Robert Phillipsb87b39b2020-07-01 14:45:24 -0400125 DrawResult onGpuSetup(GrDirectContext* context, SkString* errorMsg) override {
Robert Phillipse3939012020-06-26 08:08:22 -0400126 if (!context || context->abandoned()) {
127 return DrawResult::kSkip;
128 }
129
Brian Salomon0f396992020-06-19 19:51:21 -0400130 if (context->backend() != GrBackendApi::kOpenGL_GrBackend ||
131 !static_cast<const GrGLCaps*>(context->priv().caps())->rectangleTextureSupport()) {
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500132 *errorMsg = "This GM requires an OpenGL context that supports texture rectangles.";
Mike Klein16b1efb2019-04-02 10:01:11 -0400133 return DrawResult::kSkip;
134 }
135
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500136 auto gradCircle = this->makeImagePixels(50, ImageType::kGradientCircle);
bsalomone179a912016-01-20 06:18:10 -0800137
Robert Phillipse3939012020-06-26 08:08:22 -0400138 fGradImgs[0] = this->createRectangleTextureImg(context, kTopLeft_GrSurfaceOrigin,
139 gradCircle);
140 fGradImgs[1] = this->createRectangleTextureImg(context, kBottomLeft_GrSurfaceOrigin,
141 gradCircle);
142 SkASSERT(SkToBool(fGradImgs[0]) == SkToBool(fGradImgs[1]));
143 if (!fGradImgs[0]) {
144 *errorMsg = "Could not create gradient rectangle texture images.";
Chris Dalton50e24d72019-02-07 16:20:09 -0700145 return DrawResult::kFail;
bsalomone179a912016-01-20 06:18:10 -0800146 }
147
Robert Phillipse3939012020-06-26 08:08:22 -0400148 fSmallImg = this->createRectangleTextureImg(context, kTopLeft_GrSurfaceOrigin,
149 this->makeImagePixels(2, ImageType::k2x2));
150 if (!fSmallImg) {
151 *errorMsg = "Could not create 2x2 rectangle texture image.";
152 return DrawResult::kFail;
153 }
154
155 return DrawResult::kOk;
156 }
157
158 void onGpuTeardown() override {
159 fGradImgs[0] = fGradImgs[1] = nullptr;
160 fSmallImg = nullptr;
161 }
162
Robert Phillips95c250c2020-06-29 15:36:12 -0400163 DrawResult onDraw(GrRecordingContext*, GrRenderTargetContext*, SkCanvas* canvas,
164 SkString*) override {
Robert Phillipse3939012020-06-26 08:08:22 -0400165 SkASSERT(fGradImgs[0] && fGradImgs[1] && fSmallImg);
166
167 static constexpr SkScalar kPad = 5.f;
168
mtkleindbfd7ab2016-09-01 11:24:54 -0700169 constexpr SkFilterQuality kQualities[] = {
Brian Salomon246bc3d2018-12-06 15:33:02 -0500170 kNone_SkFilterQuality,
171 kLow_SkFilterQuality,
172 kMedium_SkFilterQuality,
173 kHigh_SkFilterQuality,
bsalomone179a912016-01-20 06:18:10 -0800174 };
175
Brian Salomon246bc3d2018-12-06 15:33:02 -0500176 constexpr SkScalar kScales[] = {1.0f, 1.2f, 0.75f};
bsalomone179a912016-01-20 06:18:10 -0800177
178 canvas->translate(kPad, kPad);
Robert Phillipse3939012020-06-26 08:08:22 -0400179 for (size_t i = 0; i < kNumGradImages; ++i) {
180 auto img = fGradImgs[i];
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500181 int w = img->width();
182 int h = img->height();
Brian Salomon246bc3d2018-12-06 15:33:02 -0500183 for (auto s : kScales) {
184 canvas->save();
185 canvas->scale(s, s);
186 for (auto q : kQualities) {
187 // drawImage
188 SkPaint plainPaint;
189 plainPaint.setFilterQuality(q);
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500190 canvas->drawImage(img, 0, 0, &plainPaint);
191 canvas->translate(w + kPad, 0);
bsalomone179a912016-01-20 06:18:10 -0800192
Brian Salomon246bc3d2018-12-06 15:33:02 -0500193 // clamp/clamp shader
194 SkPaint clampPaint;
195 clampPaint.setFilterQuality(q);
Robert Phillipse3939012020-06-26 08:08:22 -0400196 clampPaint.setShader(fGradImgs[i]->makeShader());
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500197 canvas->drawRect(SkRect::MakeWH(1.5f*w, 1.5f*h), clampPaint);
198 canvas->translate(1.5f*w + kPad, 0);
bsalomone179a912016-01-20 06:18:10 -0800199
Brian Salomon246bc3d2018-12-06 15:33:02 -0500200 // repeat/mirror shader
201 SkPaint repeatPaint;
202 repeatPaint.setFilterQuality(q);
Robert Phillipse3939012020-06-26 08:08:22 -0400203 repeatPaint.setShader(fGradImgs[i]->makeShader(SkTileMode::kRepeat,
204 SkTileMode::kMirror));
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500205 canvas->drawRect(SkRect::MakeWH(1.5f*w, 1.5f*h), repeatPaint);
206 canvas->translate(1.5f*w + kPad, 0);
Brian Salomon246bc3d2018-12-06 15:33:02 -0500207
208 // drawImageRect with kStrict
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500209 auto srcRect = SkRect::MakeXYWH(.25f*w, .25f*h, .50f*w, .50f*h);
210 auto dstRect = SkRect::MakeXYWH( 0, 0, .50f*w, .50f*h);
Robert Phillipse3939012020-06-26 08:08:22 -0400211 canvas->drawImageRect(fGradImgs[i], srcRect, dstRect, &plainPaint,
Brian Salomon246bc3d2018-12-06 15:33:02 -0500212 SkCanvas::kStrict_SrcRectConstraint);
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500213 canvas->translate(.5f*w + kPad, 0);
Brian Salomon246bc3d2018-12-06 15:33:02 -0500214 }
215 canvas->restore();
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500216 canvas->translate(0, kPad + 1.5f*h*s);
bsalomone179a912016-01-20 06:18:10 -0800217 }
bsalomone179a912016-01-20 06:18:10 -0800218 }
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500219
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500220 static constexpr SkScalar kOutset = 25.f;
221 canvas->translate(kOutset, kOutset);
Robert Phillipse3939012020-06-26 08:08:22 -0400222 auto dstRect = SkRect::Make(fSmallImg->dimensions()).makeOutset(kOutset, kOutset);
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500223
224 for (int fq = kNone_SkFilterQuality; fq <= kLast_SkFilterQuality; ++fq) {
225 if (fq == kMedium_SkFilterQuality) {
226 // Medium is the same as Low for upscaling.
227 continue;
228 }
229 canvas->save();
230 for (int ty = 0; ty < kSkTileModeCount; ++ty) {
231 canvas->save();
232 for (int tx = 0; tx < kSkTileModeCount; ++tx) {
233 SkMatrix lm;
234 lm.setRotate(45.f, 1, 1);
235 lm.postScale(6.5f, 6.5f);
Robert Phillipse3939012020-06-26 08:08:22 -0400236 auto shader = fSmallImg->makeShader(static_cast<SkTileMode>(tx),
237 static_cast<SkTileMode>(ty), &lm);
Brian Salomon3fcf83a2020-02-23 21:29:01 -0500238 SkPaint paint;
239 paint.setShader(std::move(shader));
240 paint.setFilterQuality(static_cast<SkFilterQuality>(fq));
241 canvas->drawRect(dstRect, paint);
242 canvas->translate(dstRect.width() + kPad, 0);
243 }
244 canvas->restore();
245 canvas->translate(0, dstRect.height() + kPad);
246 }
247 canvas->restore();
248 canvas->translate((dstRect.width() + kPad)*kSkTileModeCount, 0);
249 }
250
Chris Dalton50e24d72019-02-07 16:20:09 -0700251 return DrawResult::kOk;
bsalomone179a912016-01-20 06:18:10 -0800252 }
253
254private:
Robert Phillipse3939012020-06-26 08:08:22 -0400255 static const int kNumGradImages = 2;
256
257 sk_sp<SkImage> fGradImgs[kNumGradImages];
258 sk_sp<SkImage> fSmallImg;
259
bsalomone179a912016-01-20 06:18:10 -0800260 typedef GM INHERITED;
261};
262
263DEF_GM(return new RectangleTexture;)
264}
Brian Salomonf4ba4ec2020-03-19 15:54:28 -0400265#endif