convexpolyeffect.cpp revision f0539800165314f8bebd8a5ab765ec35012f1b03
1 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 { 47 return make_isize(475, 800); 48 } 49 50 virtual uint32_t onGetFlags() const SK_OVERRIDE { 51 // This is a GPU-specific GM. 52 return kGPUOnly_Flag; 53 } 54 55 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); 60 61 fPaths.addToTail(tri); 62 fPaths.addToTail(SkPath())->reverseAddPath(tri); 63 64 tri.close(); 65 fPaths.addToTail(tri); 66 67 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 } 82 83 fPaths.addToTail(ngon); 84 SkMatrix scaleM; 85 scaleM.setScale(1.1f, 0.4f); 86 ngon.transform(scaleM); 87 fPaths.addToTail(ngon); 88 89 // 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)); 103 104 } 105 106 virtual void onDraw(SkCanvas* canvas) SK_OVERRIDE { 107 SkBaseDevice* device = canvas->getTopDevice(); 108 GrRenderTarget* rt = device->accessRenderTarget(); 109 if (NULL == rt) { 110 return; 111 } 112 GrContext* context = rt->getContext(); 113 if (NULL == context) { 114 return; 115 } 116 117 SkScalar y = 0; 118 for (SkTLList<SkPath>::Iter iter(fPaths, SkTLList<SkPath>::Iter::kHead_IterStart); 119 NULL != iter.get(); 120 iter.next()) { 121 const SkPath* path = iter.get(); 122 SkScalar x = 0; 123 124 for (int et = 0; et < GrConvexPolyEffect::kEdgeTypeCnt; ++et) { 125 GrTestTarget tt; 126 context->getTestTarget(&tt); 127 if (NULL == tt.target()) { 128 SkDEBUGFAIL("Couldn't get Gr test target."); 129 return; 130 } 131 GrDrawState* drawState = tt.target()->drawState(); 132 drawState->setVertexAttribs<kAttribs>(SK_ARRAY_COUNT(kAttribs)); 133 134 SkMatrix m; 135 SkPath p; 136 m.setTranslate(x, y); 137 path->transform(m, &p); 138 139 GrConvexPolyEffect::EdgeType edgeType = (GrConvexPolyEffect::EdgeType) et; 140 SkAutoTUnref<GrEffectRef> effect(GrConvexPolyEffect::Create(edgeType, p)); 141 if (!effect) { 142 SkDEBUGFAIL("Couldn't create convex poly effect."); 143 return; 144 } 145 drawState->addCoverageEffect(effect, 1); 146 drawState->setIdentityViewMatrix(); 147 drawState->setRenderTarget(rt); 148 drawState->setColor(0xff000000); 149 150 SkPoint verts[4]; 151 SkRect bounds = p.getBounds(); 152 // Make sure any artifacts around the exterior of path are visible by using overly 153 // conservative bounding geometry. 154 bounds.outset(5.f, 5.f); 155 bounds.toQuad(verts); 156 157 tt.target()->setVertexSourceToArray(verts, 4); 158 tt.target()->setIndexSourceToBuffer(context->getQuadIndexBuffer()); 159 tt.target()->drawIndexed(kTriangleFan_GrPrimitiveType, 0, 0, 4, 6); 160 161 x += SkScalarCeilToScalar(path->getBounds().width() + 10.f); 162 } 163 164 // Draw AA and non AA paths using normal API for reference. 165 canvas->save(); 166 canvas->translate(x, y); 167 SkPaint paint; 168 canvas->drawPath(*path, paint); 169 canvas->translate(path->getBounds().width() + 10.f, 0); 170 paint.setAntiAlias(true); 171 canvas->drawPath(*path, paint); 172 canvas->restore(); 173 174 y += SkScalarCeilToScalar(path->getBounds().height() + 20.f); 175 } 176 177 // Draw rects. We only have specialized effect code for the AA case, so don't do non-AA. 178 for (SkTLList<SkRect>::Iter iter(fRects, SkTLList<SkRect>::Iter::kHead_IterStart); 179 NULL != iter.get(); 180 iter.next()) { 181 182 SkScalar x = 0; 183 184 GrTestTarget tt; 185 context->getTestTarget(&tt); 186 if (NULL == tt.target()) { 187 SkDEBUGFAIL("Couldn't get Gr test target."); 188 return; 189 } 190 191 SkRect rect = *iter.get(); 192 rect.offset(x, y); 193 SkAutoTUnref<GrEffectRef> effect(GrConvexPolyEffect::CreateForAAFillRect(rect)); 194 if (!effect) { 195 SkDEBUGFAIL("Couldn't create convex poly effect."); 196 return; 197 } 198 199 GrDrawState* drawState = tt.target()->drawState(); 200 drawState->setVertexAttribs<kAttribs>(SK_ARRAY_COUNT(kAttribs)); 201 drawState->addCoverageEffect(effect, 1); 202 drawState->setIdentityViewMatrix(); 203 drawState->setRenderTarget(rt); 204 drawState->setColor(0xff000000); 205 206 SkPoint verts[4]; 207 SkRect bounds = rect; 208 bounds.outset(5.f, 5.f); 209 bounds.toQuad(verts); 210 211 tt.target()->setVertexSourceToArray(verts, 4); 212 tt.target()->setIndexSourceToBuffer(context->getQuadIndexBuffer()); 213 tt.target()->drawIndexed(kTriangleFan_GrPrimitiveType, 0, 0, 4, 6); 214 215 x += SkScalarCeilToScalar(rect.width() + 10.f); 216 217 // Draw AA rect using normal API for reference 218 canvas->save(); 219 canvas->translate(x, y); 220 SkPaint paint; 221 paint.setAntiAlias(true); 222 canvas->drawRect(*iter.get(), paint); 223 canvas->restore(); 224 225 y += SkScalarCeilToScalar(rect.height() + 20.f); 226 } 227 } 228 229private: 230 SkTLList<SkPath> fPaths; 231 SkTLList<SkRect> fRects; 232 233 typedef GM INHERITED; 234}; 235 236DEF_GM( return SkNEW(ConvexPolyEffect); ) 237} 238 239#endif 240