blob: 64f3bbc64cefeb57ae52fa0444f1fb2e101b4a26 [file] [log] [blame]
joshualitt5ce33c12015-01-28 11:08:00 -08001/*
2 * Copyright 2013 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 */
Ben Wagner7fde8e12019-05-01 17:28:53 -04007
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "gm/gm.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -04009#include "include/core/SkBlendMode.h"
10#include "include/core/SkCanvas.h"
11#include "include/core/SkColor.h"
12#include "include/core/SkColorSpace.h"
13#include "include/core/SkFont.h"
14#include "include/core/SkImageInfo.h"
15#include "include/core/SkMatrix.h"
16#include "include/core/SkPaint.h"
Mike Reed06d7c9d2020-08-26 12:56:51 -040017#include "include/core/SkPathBuilder.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040018#include "include/core/SkPoint.h"
19#include "include/core/SkRect.h"
20#include "include/core/SkRefCnt.h"
21#include "include/core/SkScalar.h"
22#include "include/core/SkSize.h"
23#include "include/core/SkString.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050024#include "include/core/SkSurface.h"
Ben Wagner7fde8e12019-05-01 17:28:53 -040025#include "include/core/SkTypeface.h"
26#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050027#include "include/utils/SkRandom.h"
28#include "tools/ToolUtils.h"
joshualitt5ce33c12015-01-28 11:08:00 -080029
30namespace skiagm {
31
32/**
33 * Renders overlapping shapes with colorburn against a checkerboard.
34 */
35class DstReadShuffle : public GM {
36public:
Brian Salomond02b6f32017-03-17 10:41:39 -040037 DstReadShuffle() { this->setBGColor(kBackground); }
joshualitt5ce33c12015-01-28 11:08:00 -080038
39protected:
40 enum ShapeType {
41 kCircle_ShapeType,
42 kRoundRect_ShapeType,
43 kRect_ShapeType,
44 kConvexPath_ShapeType,
45 kConcavePath_ShapeType,
46 kText_ShapeType,
47 kNumShapeTypes
48 };
49
mtklein36352bf2015-03-25 18:17:31 -070050 SkString onShortName() override {
joshualitt5ce33c12015-01-28 11:08:00 -080051 return SkString("dstreadshuffle");
52 }
53
mtklein36352bf2015-03-25 18:17:31 -070054 SkISize onISize() override {
Brian Salomon8336e942017-03-15 15:37:43 -040055 return SkISize::Make(530, 680);
joshualitt5ce33c12015-01-28 11:08:00 -080056 }
57
Brian Salomon8336e942017-03-15 15:37:43 -040058 void drawShape(SkCanvas* canvas, SkPaint* paint, ShapeType type) {
59 const SkRect kRect = SkRect::MakeXYWH(0, 0, 75.f, 85.f);
joshualitt5ce33c12015-01-28 11:08:00 -080060 switch (type) {
61 case kCircle_ShapeType:
Brian Salomon8336e942017-03-15 15:37:43 -040062 canvas->drawCircle(kRect.centerX(), kRect.centerY(), kRect.width() / 2.f, *paint);
joshualitt5ce33c12015-01-28 11:08:00 -080063 break;
64 case kRoundRect_ShapeType:
Brian Salomon8336e942017-03-15 15:37:43 -040065 canvas->drawRoundRect(kRect, 15.f, 15.f, *paint);
joshualitt5ce33c12015-01-28 11:08:00 -080066 break;
67 case kRect_ShapeType:
68 canvas->drawRect(kRect, *paint);
69 break;
70 case kConvexPath_ShapeType:
71 if (fConvexPath.isEmpty()) {
72 SkPoint points[4];
73 kRect.toQuad(points);
Mike Reed06d7c9d2020-08-26 12:56:51 -040074 fConvexPath = SkPathBuilder().moveTo(points[0])
75 .quadTo(points[1], points[2])
76 .quadTo(points[3], points[0])
77 .detach();
joshualitt5ce33c12015-01-28 11:08:00 -080078 SkASSERT(fConvexPath.isConvex());
79 }
80 canvas->drawPath(fConvexPath, *paint);
81 break;
82 case kConcavePath_ShapeType:
83 if (fConcavePath.isEmpty()) {
Mike Reed06d7c9d2020-08-26 12:56:51 -040084 SkPathBuilder b;
Brian Salomon8336e942017-03-15 15:37:43 -040085 SkPoint points[5] = {{50.f, 0.f}};
joshualitt5ce33c12015-01-28 11:08:00 -080086 SkMatrix rot;
Brian Salomon8336e942017-03-15 15:37:43 -040087 rot.setRotate(360.f / 5, 50.f, 70.f);
joshualitt5ce33c12015-01-28 11:08:00 -080088 for (int i = 1; i < 5; ++i) {
89 rot.mapPoints(points + i, points + i - 1, 1);
90 }
Mike Reed06d7c9d2020-08-26 12:56:51 -040091 b.moveTo(points[0]);
joshualitt5ce33c12015-01-28 11:08:00 -080092 for (int i = 0; i < 5; ++i) {
Mike Reed06d7c9d2020-08-26 12:56:51 -040093 b.lineTo(points[(2 * i) % 5]);
joshualitt5ce33c12015-01-28 11:08:00 -080094 }
Mike Reed06d7c9d2020-08-26 12:56:51 -040095 fConcavePath = b.setFillType(SkPathFillType::kEvenOdd)
96 .detach();
joshualitt5ce33c12015-01-28 11:08:00 -080097 SkASSERT(!fConcavePath.isConvex());
98 }
99 canvas->drawPath(fConcavePath, *paint);
100 break;
101 case kText_ShapeType: {
Brian Salomon8336e942017-03-15 15:37:43 -0400102 const char* text = "N";
Mike Kleinea3f0142019-03-20 11:12:10 -0500103 SkFont font(ToolUtils::create_portable_typeface(), 100);
Hal Canary6ac0df82019-01-07 16:01:22 -0500104 font.setEmbolden(true);
105 canvas->drawString(text, 0.f, 100.f, font, *paint);
John Stiles30212b72020-06-11 17:55:07 -0400106 break;
joshualitt5ce33c12015-01-28 11:08:00 -0800107 }
108 default:
109 break;
110 }
111 }
112
Brian Salomon8336e942017-03-15 15:37:43 -0400113 static SkColor GetColor(SkRandom* random) {
Mike Kleinea3f0142019-03-20 11:12:10 -0500114 SkColor color = ToolUtils::color_to_565(random->nextU() | 0xFF000000);
Brian Salomon8336e942017-03-15 15:37:43 -0400115 return SkColorSetA(color, 0x80);
joshualitt5ce33c12015-01-28 11:08:00 -0800116 }
117
Brian Salomon8336e942017-03-15 15:37:43 -0400118 static void DrawHairlines(SkCanvas* canvas) {
Brian Salomond02b6f32017-03-17 10:41:39 -0400119 if (canvas->imageInfo().alphaType() == kOpaque_SkAlphaType) {
120 canvas->clear(kBackground);
121 } else {
122 canvas->clear(SK_ColorTRANSPARENT);
123 }
Brian Salomon8336e942017-03-15 15:37:43 -0400124 SkPaint hairPaint;
125 hairPaint.setStyle(SkPaint::kStroke_Style);
126 hairPaint.setStrokeWidth(0);
127 hairPaint.setAntiAlias(true);
128 static constexpr int kNumHairlines = 12;
129 SkPoint pts[] = {{3.f, 7.f}, {29.f, 7.f}};
130 SkRandom colorRandom;
131 SkMatrix rot;
132 rot.setRotate(360.f / kNumHairlines, 15.5f, 12.f);
133 rot.postTranslate(3.f, 0);
134 for (int i = 0; i < 12; ++i) {
135 hairPaint.setColor(GetColor(&colorRandom));
Hal Canary23e474c2017-05-15 13:35:35 -0400136 canvas->drawLine(pts[0], pts[1], hairPaint);
Brian Salomon8336e942017-03-15 15:37:43 -0400137 rot.mapPoints(pts, 2);
joshualitt5ce33c12015-01-28 11:08:00 -0800138 }
139 }
140
mtklein36352bf2015-03-25 18:17:31 -0700141 void onDraw(SkCanvas* canvas) override {
Brian Salomon8336e942017-03-15 15:37:43 -0400142 SkScalar y = 5;
joshualitt5ce33c12015-01-28 11:08:00 -0800143 for (int i = 0; i < kNumShapeTypes; i++) {
Brian Salomon8336e942017-03-15 15:37:43 -0400144 SkRandom colorRandom;
joshualitt5ce33c12015-01-28 11:08:00 -0800145 ShapeType shapeType = static_cast<ShapeType>(i);
Brian Salomon8336e942017-03-15 15:37:43 -0400146 SkScalar x = 5;
147 for (int r = 0; r <= 15; r++) {
148 SkPaint p;
149 p.setAntiAlias(true);
150 p.setColor(GetColor(&colorRandom));
151 // In order to get some op combining on the GPU backend we do 2 src over
152 // for each xfer mode which requires a dst read
153 p.setBlendMode(r % 3 == 0 ? SkBlendMode::kColorBurn : SkBlendMode::kSrcOver);
154 canvas->save();
155 canvas->translate(x, y);
156 this->drawShape(canvas, &p, shapeType);
157 canvas->restore();
158 x += 15;
joshualitt5ce33c12015-01-28 11:08:00 -0800159 }
Brian Salomon8336e942017-03-15 15:37:43 -0400160 y += 110;
joshualitt5ce33c12015-01-28 11:08:00 -0800161 }
Brian Salomon8336e942017-03-15 15:37:43 -0400162 // Draw hairlines to a surface and then draw that to the main canvas with a zoom so that
163 // it is easier to see how they blend.
164 SkImageInfo info;
Brian Salomond02b6f32017-03-17 10:41:39 -0400165 // Recording canvases don't have a color type.
166 if (SkColorType::kUnknown_SkColorType == canvas->imageInfo().colorType()) {
167 info = SkImageInfo::MakeN32Premul(35, 35);
168 } else {
Brian Salomon8336e942017-03-15 15:37:43 -0400169 info = SkImageInfo::Make(35, 35,
170 canvas->imageInfo().colorType(),
171 canvas->imageInfo().alphaType(),
172 canvas->imageInfo().refColorSpace());
Brian Salomon8336e942017-03-15 15:37:43 -0400173 }
174 auto surf = canvas->makeSurface(info);
Cary Clarka24712e2018-09-05 18:41:40 +0000175 if (!surf) {
176 // Fall back to raster. Raster supports only one of the 8 bit per-channel RGBA or BGRA
177 // formats. This fall back happens when running with --preAbandonGpuContext.
178 if ((info.colorType() == kRGBA_8888_SkColorType ||
179 info.colorType() == kBGRA_8888_SkColorType) &&
180 info.colorType() != kN32_SkColorType) {
181 info = SkImageInfo::Make(35, 35,
182 kN32_SkColorType,
183 canvas->imageInfo().alphaType(),
184 canvas->imageInfo().refColorSpace());
185 }
186 surf = SkSurface::MakeRaster(info);
187 SkASSERT(surf);
188 }
Brian Salomon8336e942017-03-15 15:37:43 -0400189 canvas->scale(5.f, 5.f);
190 canvas->translate(67.f, 10.f);
191 DrawHairlines(surf->getCanvas());
192 canvas->drawImage(surf->makeImageSnapshot(), 0.f, 0.f);
joshualitt5ce33c12015-01-28 11:08:00 -0800193 }
194
195private:
Brian Salomond02b6f32017-03-17 10:41:39 -0400196 static constexpr SkColor kBackground = SK_ColorLTGRAY;
Brian Salomon8336e942017-03-15 15:37:43 -0400197 SkPath fConcavePath;
198 SkPath fConvexPath;
John Stiles7571f9e2020-09-02 22:42:33 -0400199 using INHERITED = GM;
joshualitt5ce33c12015-01-28 11:08:00 -0800200};
201
202//////////////////////////////////////////////////////////////////////////////
203
Hal Canarye964c182019-01-23 10:22:01 -0500204DEF_GM( return new DstReadShuffle; )
joshualitt5ce33c12015-01-28 11:08:00 -0800205
John Stilesa6841be2020-08-06 14:11:56 -0400206} // namespace skiagm