blob: 275193f326435be3ae42c8c77df4944c32d19d49 [file] [log] [blame]
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +00001
2/*
3 * Copyright 2014 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
8
9// This test only works with the GPU backend.
10
11#include "gm.h"
12
13#if SK_SUPPORT_GPU
14
15#include "GrContext.h"
16#include "GrPathUtils.h"
17#include "GrTest.h"
18#include "SkColorPriv.h"
19#include "SkDevice.h"
20#include "SkGeometry.h"
21#include "SkTLList.h"
22
23#include "effects/GrConvexPolyEffect.h"
24
25namespace {
26extern const GrVertexAttrib kAttribs[] = {
27 {kVec2f_GrVertexAttribType, 0, kPosition_GrVertexAttribBinding},
28};
29}
30
31namespace skiagm {
32/**
33 * This GM directly exercises a GrEffect that draws convex polygons.
34 */
35class ConvexPolyEffect : public GM {
36public:
37 ConvexPolyEffect() {
38 this->setBGColor(0xFFFFFFFF);
39 }
40
41protected:
42 virtual SkString onShortName() SK_OVERRIDE {
43 return SkString("convex_poly_effect");
44 }
45
46 virtual SkISize onISize() SK_OVERRIDE {
tfarinaf5393182014-06-09 23:59:03 -070047 return SkISize::Make(720, 800);
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000048 }
49
50 virtual uint32_t onGetFlags() const SK_OVERRIDE {
51 // This is a GPU-specific GM.
52 return kGPUOnly_Flag;
53 }
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +000054
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000055 virtual void onOnceBeforeDraw() SK_OVERRIDE {
56 SkPath tri;
57 tri.moveTo(5.f, 5.f);
58 tri.lineTo(100.f, 20.f);
59 tri.lineTo(15.f, 100.f);
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +000060
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000061 fPaths.addToTail(tri);
62 fPaths.addToTail(SkPath())->reverseAddPath(tri);
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +000063
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000064 tri.close();
65 fPaths.addToTail(tri);
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +000066
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000067 SkPath ngon;
68 static const SkScalar kRadius = 50.f;
69 const SkPoint center = { kRadius, kRadius };
70 for (int i = 0; i < GrConvexPolyEffect::kMaxEdges; ++i) {
71 SkScalar angle = 2 * SK_ScalarPI * i / GrConvexPolyEffect::kMaxEdges;
72 SkPoint point;
73 point.fY = SkScalarSinCos(angle, &point.fX);
74 point.scale(kRadius);
75 point = center + point;
76 if (0 == i) {
77 ngon.moveTo(point);
78 } else {
79 ngon.lineTo(point);
80 }
81 }
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +000082
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +000083 fPaths.addToTail(ngon);
84 SkMatrix scaleM;
85 scaleM.setScale(1.1f, 0.4f);
86 ngon.transform(scaleM);
87 fPaths.addToTail(ngon);
skia.committer@gmail.comf0b0cda2014-02-09 03:02:01 +000088
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +000089 // integer edges
90 fRects.addToTail(SkRect::MakeLTRB(5.f, 1.f, 30.f, 25.f));
91 // half-integer edges
92 fRects.addToTail(SkRect::MakeLTRB(5.5f, 0.5f, 29.5f, 24.5f));
93 // vertically/horizontally thin rects that cover pixel centers
94 fRects.addToTail(SkRect::MakeLTRB(5.25f, 0.5f, 5.75f, 24.5f));
95 fRects.addToTail(SkRect::MakeLTRB(5.5f, 0.5f, 29.5f, 0.75f));
96 // vertically/horizontally thin rects that don't cover pixel centers
97 fRects.addToTail(SkRect::MakeLTRB(5.55f, 0.5f, 5.75f, 24.5f));
98 fRects.addToTail(SkRect::MakeLTRB(5.5f, .05f, 29.5f, .25f));
99 // small in x and y
100 fRects.addToTail(SkRect::MakeLTRB(5.05f, .55f, 5.45f, .85f));
101 // inverted in x and y
102 fRects.addToTail(SkRect::MakeLTRB(100.f, 50.5f, 5.f, 0.5f));
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000103 }
104
105 virtual void onDraw(SkCanvas* canvas) SK_OVERRIDE {
reed@google.com9c135db2014-03-12 18:28:35 +0000106 GrRenderTarget* rt = canvas->internal_private_accessTopLayerRenderTarget();
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000107 if (NULL == rt) {
108 return;
109 }
110 GrContext* context = rt->getContext();
111 if (NULL == context) {
112 return;
113 }
114
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000115 SkScalar y = 0;
116 for (SkTLList<SkPath>::Iter iter(fPaths, SkTLList<SkPath>::Iter::kHead_IterStart);
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000117 NULL != iter.get();
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000118 iter.next()) {
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000119 const SkPath* path = iter.get();
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000120 SkScalar x = 0;
121
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000122 for (int et = 0; et < kGrEffectEdgeTypeCnt; ++et) {
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000123 GrTestTarget tt;
124 context->getTestTarget(&tt);
125 if (NULL == tt.target()) {
126 SkDEBUGFAIL("Couldn't get Gr test target.");
127 return;
128 }
129 GrDrawState* drawState = tt.target()->drawState();
egdaniel7b3d5ee2014-08-28 05:41:14 -0700130 drawState->setVertexAttribs<kAttribs>(SK_ARRAY_COUNT(kAttribs), sizeof(SkPoint));
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000131
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000132 SkMatrix m;
133 SkPath p;
134 m.setTranslate(x, y);
135 path->transform(m, &p);
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000136
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000137 GrEffectEdgeType edgeType = (GrEffectEdgeType) et;
bsalomon83d081a2014-07-08 09:56:10 -0700138 SkAutoTUnref<GrEffect> effect(GrConvexPolyEffect::Create(edgeType, p));
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000139 if (!effect) {
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000140 continue;
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000141 }
142 drawState->addCoverageEffect(effect, 1);
143 drawState->setIdentityViewMatrix();
144 drawState->setRenderTarget(rt);
145 drawState->setColor(0xff000000);
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000146
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000147 SkPoint verts[4];
148 SkRect bounds = p.getBounds();
149 // Make sure any artifacts around the exterior of path are visible by using overly
150 // conservative bounding geometry.
151 bounds.outset(5.f, 5.f);
152 bounds.toQuad(verts);
153
154 tt.target()->setVertexSourceToArray(verts, 4);
155 tt.target()->setIndexSourceToBuffer(context->getQuadIndexBuffer());
156 tt.target()->drawIndexed(kTriangleFan_GrPrimitiveType, 0, 0, 4, 6);
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000157
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000158 x += SkScalarCeilToScalar(path->getBounds().width() + 10.f);
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000159 }
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000160
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000161 // Draw AA and non AA paths using normal API for reference.
162 canvas->save();
163 canvas->translate(x, y);
164 SkPaint paint;
165 canvas->drawPath(*path, paint);
166 canvas->translate(path->getBounds().width() + 10.f, 0);
167 paint.setAntiAlias(true);
168 canvas->drawPath(*path, paint);
169 canvas->restore();
skia.committer@gmail.com4c18e9f2014-01-31 03:01:59 +0000170
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000171 y += SkScalarCeilToScalar(path->getBounds().height() + 20.f);
172 }
173
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000174 for (SkTLList<SkRect>::Iter iter(fRects, SkTLList<SkRect>::Iter::kHead_IterStart);
175 NULL != iter.get();
176 iter.next()) {
177
178 SkScalar x = 0;
179
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000180 for (int et = 0; et < kGrEffectEdgeTypeCnt; ++et) {
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000181 GrTestTarget tt;
182 context->getTestTarget(&tt);
183 if (NULL == tt.target()) {
184 SkDEBUGFAIL("Couldn't get Gr test target.");
185 return;
186 }
187 SkRect rect = *iter.get();
188 rect.offset(x, y);
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000189 GrEffectEdgeType edgeType = (GrEffectEdgeType) et;
bsalomon83d081a2014-07-08 09:56:10 -0700190 SkAutoTUnref<GrEffect> effect(GrConvexPolyEffect::Create(edgeType, rect));
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000191 if (!effect) {
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000192 continue;
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000193 }
194
195 GrDrawState* drawState = tt.target()->drawState();
egdaniel7b3d5ee2014-08-28 05:41:14 -0700196 drawState->setVertexAttribs<kAttribs>(SK_ARRAY_COUNT(kAttribs), sizeof(SkPoint));
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000197 drawState->addCoverageEffect(effect, 1);
198 drawState->setIdentityViewMatrix();
199 drawState->setRenderTarget(rt);
200 drawState->setColor(0xff000000);
201
202 SkPoint verts[4];
203 SkRect bounds = rect;
204 bounds.outset(5.f, 5.f);
205 bounds.toQuad(verts);
206
207 tt.target()->setVertexSourceToArray(verts, 4);
208 tt.target()->setIndexSourceToBuffer(context->getQuadIndexBuffer());
209 tt.target()->drawIndexed(kTriangleFan_GrPrimitiveType, 0, 0, 4, 6);
210
211 x += SkScalarCeilToScalar(rect.width() + 10.f);
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000212 }
213
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000214 // Draw rect without and with AA using normal API for reference
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000215 canvas->save();
216 canvas->translate(x, y);
217 SkPaint paint;
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000218 canvas->drawRect(*iter.get(), paint);
219 x += SkScalarCeilToScalar(iter.get()->width() + 10.f);
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000220 paint.setAntiAlias(true);
221 canvas->drawRect(*iter.get(), paint);
222 canvas->restore();
223
commit-bot@chromium.orgd85f32c2014-02-28 14:43:26 +0000224 y += SkScalarCeilToScalar(iter.get()->height() + 20.f);
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000225 }
226 }
227
228private:
229 SkTLList<SkPath> fPaths;
commit-bot@chromium.orgf0539802014-02-08 19:31:05 +0000230 SkTLList<SkRect> fRects;
skia.committer@gmail.comf0b0cda2014-02-09 03:02:01 +0000231
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000232 typedef GM INHERITED;
233};
234
235DEF_GM( return SkNEW(ConvexPolyEffect); )
commit-bot@chromium.orgc3fe5492014-01-30 18:15:51 +0000236}
237
238#endif