1 /* 2 * Copyright 2014 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 GrStencilAndCoverTextContext_DEFINED 9 #define GrStencilAndCoverTextContext_DEFINED 10 11 #include "GrRenderTargetContext.h" 12 #include "GrStyle.h" 13 #include "SkDrawFilter.h" 14 #include "SkOpts.h" 15 #include "SkTHash.h" 16 #include "SkTInternalLList.h" 17 #include "SkTLList.h" 18 #include "SkTextBlob.h" 19 #include "ops/GrDrawPathOp.h" 20 21 class GrAtlasTextContext; 22 class GrTextStrike; 23 class GrPath; 24 class SkSurfaceProps; 25 26 /* 27 * This class implements text rendering using stencil and cover path rendering 28 * (by the means of GrOpList::drawPath). 29 */ 30 class GrStencilAndCoverTextContext { 31 public: 32 static GrStencilAndCoverTextContext* Create(GrAtlasTextContext* fallbackTextContext); 33 34 void drawText(GrContext*, GrRenderTargetContext* rtc, const GrClip&, const SkPaint&, 35 const SkMatrix& viewMatrix, const SkSurfaceProps&, const char text[], 36 size_t byteLength, SkScalar x, SkScalar y, const SkIRect& clipBounds); 37 void drawPosText(GrContext*, GrRenderTargetContext*, const GrClip&, const SkPaint&, 38 const SkMatrix& viewMatrix, const SkSurfaceProps&, const char text[], 39 size_t byteLength, const SkScalar pos[], int scalarsPerPosition, 40 const SkPoint& offset, const SkIRect& clipBounds); 41 void drawTextBlob(GrContext*, GrRenderTargetContext*, const GrClip&, const SkPaint&, 42 const SkMatrix& viewMatrix, const SkSurfaceProps&, const SkTextBlob*, 43 SkScalar x, SkScalar y, 44 SkDrawFilter*, const SkIRect& clipBounds); 45 46 virtual ~GrStencilAndCoverTextContext(); 47 48 private: 49 GrStencilAndCoverTextContext(GrAtlasTextContext* fallbackTextContext); 50 canDraw(const SkPaint & skPaint,const SkMatrix &)51 bool canDraw(const SkPaint& skPaint, const SkMatrix&) { 52 return this->internalCanDraw(skPaint); 53 } 54 55 bool internalCanDraw(const SkPaint&); 56 57 void uncachedDrawTextBlob(GrContext*, GrRenderTargetContext* rtc, 58 const GrClip& clip, const SkPaint& skPaint, 59 const SkMatrix& viewMatrix, 60 const SkSurfaceProps&, 61 const SkTextBlob* blob, 62 SkScalar x, SkScalar y, 63 SkDrawFilter* drawFilter, 64 const SkIRect& clipBounds); 65 66 class FallbackBlobBuilder; 67 68 class TextRun { 69 public: 70 TextRun(const SkPaint& fontAndStroke); 71 ~TextRun(); 72 73 void setText(const char text[], size_t byteLength, SkScalar x, SkScalar y); 74 75 void setPosText(const char text[], size_t byteLength, const SkScalar pos[], 76 int scalarsPerPosition, const SkPoint& offset); 77 78 void draw(GrContext*, GrRenderTargetContext*, const GrClip&, const SkMatrix&, 79 const SkSurfaceProps&, SkScalar x, SkScalar y, const SkIRect& clipBounds, 80 GrAtlasTextContext* fallbackTextContext, const SkPaint& originalSkPaint) const; 81 82 void releaseGlyphCache() const; 83 84 size_t computeSizeInCache() const; 85 isAntiAlias()86 GrAA isAntiAlias() const { return fFont.isAntiAlias() ? GrAA::kYes : GrAA::kNo; } 87 88 private: 89 typedef GrDrawPathRangeOp::InstanceData InstanceData; 90 91 SkGlyphCache* getGlyphCache() const; 92 GrPathRange* createGlyphs(GrResourceProvider*) const; 93 void appendGlyph(const SkGlyph&, const SkPoint&, FallbackBlobBuilder*); 94 95 GrStyle fStyle; 96 SkPaint fFont; 97 SkScalar fTextRatio; 98 float fTextInverseRatio; 99 bool fUsingRawGlyphPaths; 100 GrUniqueKey fGlyphPathsKey; 101 int fTotalGlyphCount; 102 sk_sp<InstanceData> fInstanceData; 103 int fFallbackGlyphCount; 104 sk_sp<SkTextBlob> fFallbackTextBlob; 105 mutable SkGlyphCache* fDetachedGlyphCache; 106 mutable GrGpuResource::UniqueID fLastDrawnGlyphsID; 107 }; 108 109 // Text blobs/caches. 110 111 class TextBlob : public SkTLList<TextRun, 1> { 112 public: 113 typedef SkTArray<uint32_t, true> Key; 114 GetKey(const TextBlob * blob)115 static const Key& GetKey(const TextBlob* blob) { return blob->key(); } 116 Hash(const Key & key)117 static uint32_t Hash(const Key& key) { 118 SkASSERT(key.count() > 1); // 1-length keys should be using the blob-id hash map. 119 return SkOpts::hash(key.begin(), sizeof(uint32_t) * key.count()); 120 } 121 TextBlob(uint32_t blobId,const SkTextBlob * skBlob,const SkPaint & skPaint)122 TextBlob(uint32_t blobId, const SkTextBlob* skBlob, const SkPaint& skPaint) 123 : fKey(&blobId, 1) { this->init(skBlob, skPaint); } 124 TextBlob(const Key & key,const SkTextBlob * skBlob,const SkPaint & skPaint)125 TextBlob(const Key& key, const SkTextBlob* skBlob, const SkPaint& skPaint) 126 : fKey(key) { 127 // 1-length keys are unterstood to be the blob id and must use the other constructor. 128 SkASSERT(fKey.count() > 1); 129 this->init(skBlob, skPaint); 130 } 131 key()132 const Key& key() const { return fKey; } 133 cpuMemorySize()134 size_t cpuMemorySize() const { return fCpuMemorySize; } 135 136 private: 137 void init(const SkTextBlob*, const SkPaint&); 138 139 const SkSTArray<1, uint32_t, true> fKey; 140 size_t fCpuMemorySize; 141 142 SK_DECLARE_INTERNAL_LLIST_INTERFACE(TextBlob); 143 }; 144 145 const TextBlob& findOrCreateTextBlob(const SkTextBlob*, const SkPaint&); 146 void purgeToFit(const TextBlob&); 147 148 GrAtlasTextContext* fFallbackTextContext; 149 SkTHashMap<uint32_t, TextBlob*> fBlobIdCache; 150 SkTHashTable<TextBlob*, const TextBlob::Key&, TextBlob> fBlobKeyCache; 151 SkTInternalLList<TextBlob> fLRUList; 152 size_t fCacheSize; 153 }; 154 155 #endif 156