1 /* 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 */ 7 8 #ifndef GrCoordTransform_DEFINED 9 #define GrCoordTransform_DEFINED 10 11 #include "include/core/SkMatrix.h" 12 #include "src/gpu/GrSurfaceProxyPriv.h" 13 #include "src/gpu/GrTextureProxy.h" 14 15 class GrTexture; 16 17 /** 18 * A class representing a linear transformation of local coordinates. GrFragnentProcessors 19 * these transformations, and the GrGeometryProcessor implements the transformation. 20 */ 21 class GrCoordTransform { 22 public: GrCoordTransform()23 GrCoordTransform() 24 : fProxy(nullptr) 25 , fNormalize(false) 26 , fReverseY(false) { 27 SkDEBUGCODE(fInProcessor = false); 28 } 29 30 GrCoordTransform(const GrCoordTransform&) = default; 31 32 /** 33 * Create a transformation that maps [0, 1] to a proxy's boundaries. The proxy origin also 34 * implies whether a y-reversal should be performed. 35 */ GrCoordTransform(GrTextureProxy * proxy)36 GrCoordTransform(GrTextureProxy* proxy) { 37 SkASSERT(proxy); 38 SkDEBUGCODE(fInProcessor = false); 39 this->reset(SkMatrix::I(), proxy); 40 } 41 42 /** 43 * Create a transformation from a matrix. The proxy origin also implies whether a y-reversal 44 * should be performed. 45 */ GrCoordTransform(const SkMatrix & m,GrTextureProxy * proxy)46 GrCoordTransform(const SkMatrix& m, GrTextureProxy* proxy) { 47 SkASSERT(proxy); 48 SkDEBUGCODE(fInProcessor = false); 49 this->reset(m, proxy); 50 } 51 52 /** 53 * Create a transformation that applies the matrix to a coord set. 54 */ GrCoordTransform(const SkMatrix & m)55 GrCoordTransform(const SkMatrix& m) { 56 SkDEBUGCODE(fInProcessor = false); 57 this->reset(m); 58 } 59 60 GrCoordTransform& operator= (const GrCoordTransform& that) { 61 SkASSERT(!fInProcessor); 62 fMatrix = that.fMatrix; 63 fProxy = that.fProxy; 64 fNormalize = that.fNormalize; 65 fReverseY = that.fReverseY; 66 return *this; 67 } 68 69 /** 70 * Access the matrix for editing. Note, this must be done before adding the transform to an 71 * effect, since effects are immutable. 72 */ accessMatrix()73 SkMatrix* accessMatrix() { 74 SkASSERT(!fInProcessor); 75 return &fMatrix; 76 } 77 hasSameEffectAs(const GrCoordTransform & that)78 bool hasSameEffectAs(const GrCoordTransform& that) const { 79 if (fNormalize != that.fNormalize || 80 fReverseY != that.fReverseY || 81 !fMatrix.cheapEqualTo(that.fMatrix)) { 82 return false; 83 } 84 85 if (fNormalize) { 86 if (fProxy->underlyingUniqueID() != that.fProxy->underlyingUniqueID()) { 87 return false; 88 } 89 } 90 91 return true; 92 } 93 getMatrix()94 const SkMatrix& getMatrix() const { return fMatrix; } proxy()95 const GrTextureProxy* proxy() const { return fProxy; } normalize()96 bool normalize() const { return fNormalize; } reverseY()97 bool reverseY() const { return fReverseY; } 98 99 // This should only ever be called at flush time after the backing texture has been 100 // successfully instantiated peekTexture()101 GrTexture* peekTexture() const { return fProxy->peekTexture(); } 102 103 private: 104 void reset(const SkMatrix& m, GrTextureProxy* proxy = nullptr) { 105 SkASSERT(!fInProcessor); 106 107 fMatrix = m; 108 fProxy = proxy; 109 fNormalize = proxy && proxy->textureType() != GrTextureType::kRectangle; 110 fReverseY = proxy && kBottomLeft_GrSurfaceOrigin == proxy->origin(); 111 } 112 113 // The textures' effect is to optionally normalize the final matrix, so a blind 114 // equality check could be misleading 115 bool operator==(const GrCoordTransform& that) const; 116 bool operator!=(const GrCoordTransform& that) const; 117 118 SkMatrix fMatrix; 119 const GrTextureProxy* fProxy; 120 bool fNormalize; 121 bool fReverseY; 122 123 #ifdef SK_DEBUG 124 public: setInProcessor()125 void setInProcessor() const { fInProcessor = true; } 126 private: 127 mutable bool fInProcessor; 128 #endif 129 }; 130 131 #endif 132