joshualitt | 1d89e8d | 2015-04-01 12:40:54 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2015 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 GrAtlasTextContext_DEFINED |
| 9 | #define GrAtlasTextContext_DEFINED |
| 10 | |
joshualitt | 374b2f7 | 2015-07-21 08:05:03 -0700 | [diff] [blame] | 11 | #include "GrAtlasTextBlob.h" |
joshualitt | 1acabf3 | 2015-12-10 09:10:10 -0800 | [diff] [blame] | 12 | #include "GrDistanceFieldAdjustTable.h" |
joshualitt | 1d89e8d | 2015-04-01 12:40:54 -0700 | [diff] [blame] | 13 | #include "GrGeometryProcessor.h" |
Brian Salomon | 6f1d36c | 2017-01-13 12:02:17 -0500 | [diff] [blame] | 14 | #include "GrTextUtils.h" |
halcanary | 3377975 | 2015-10-27 14:01:05 -0700 | [diff] [blame] | 15 | #include "SkTextBlobRunIterator.h" |
joshualitt | 1d89e8d | 2015-04-01 12:40:54 -0700 | [diff] [blame] | 16 | |
Hal Canary | 6f6961e | 2017-01-31 13:50:44 -0500 | [diff] [blame] | 17 | #if GR_TEST_UTILS |
Brian Salomon | 5ec9def | 2016-12-20 15:34:05 -0500 | [diff] [blame] | 18 | #include "GrDrawOpTest.h" |
joshualitt | 79dfb2b | 2015-05-11 08:58:08 -0700 | [diff] [blame] | 19 | #endif |
| 20 | |
Brian Salomon | 9afd371 | 2016-12-01 10:59:09 -0500 | [diff] [blame] | 21 | class GrDrawOp; |
joshualitt | b7133be | 2015-04-08 09:08:31 -0700 | [diff] [blame] | 22 | class GrTextBlobCache; |
joshualitt | 6c2c2b0 | 2015-07-24 10:37:00 -0700 | [diff] [blame] | 23 | class SkGlyph; |
joshualitt | 1d89e8d | 2015-04-01 12:40:54 -0700 | [diff] [blame] | 24 | |
| 25 | /* |
joshualitt | 8e84a1e | 2016-02-16 11:09:25 -0800 | [diff] [blame] | 26 | * Renders text using some kind of an atlas, ie BitmapText or DistanceField text |
joshualitt | 1d89e8d | 2015-04-01 12:40:54 -0700 | [diff] [blame] | 27 | */ |
Herb Derby | 26cbe51 | 2018-05-24 14:39:01 -0400 | [diff] [blame^] | 28 | class GrTextContext { |
joshualitt | 1d89e8d | 2015-04-01 12:40:54 -0700 | [diff] [blame] | 29 | public: |
Brian Salomon | af59748 | 2017-11-07 16:23:34 -0500 | [diff] [blame] | 30 | struct Options { |
| 31 | /** |
| 32 | * Below this size (in device space) distance field text will not be used. Negative means |
| 33 | * use a default value. |
| 34 | */ |
| 35 | SkScalar fMinDistanceFieldFontSize = -1.f; |
| 36 | /** |
| 37 | * Above this size (in device space) distance field text will not be used and glyphs will |
| 38 | * be rendered from outline as individual paths. Negative means use a default value. |
| 39 | */ |
| 40 | SkScalar fMaxDistanceFieldFontSize = -1.f; |
Brian Salomon | b508696 | 2017-12-13 10:59:33 -0500 | [diff] [blame] | 41 | /** Forces all distance field vertices to use 3 components, not just when in perspective. */ |
| 42 | bool fDistanceFieldVerticesAlwaysHaveW = false; |
Brian Salomon | af59748 | 2017-11-07 16:23:34 -0500 | [diff] [blame] | 43 | }; |
| 44 | |
Herb Derby | 26cbe51 | 2018-05-24 14:39:01 -0400 | [diff] [blame^] | 45 | static std::unique_ptr<GrTextContext> Make(const Options& options); |
joshualitt | 1d89e8d | 2015-04-01 12:40:54 -0700 | [diff] [blame] | 46 | |
Brian Salomon | f18b1d8 | 2017-10-27 11:30:49 -0400 | [diff] [blame] | 47 | void drawText(GrContext*, GrTextUtils::Target*, const GrClip&, const SkPaint&, |
joshualitt | 2c89bc1 | 2016-02-11 05:42:30 -0800 | [diff] [blame] | 48 | const SkMatrix& viewMatrix, const SkSurfaceProps&, const char text[], |
Brian Salomon | 82f4431 | 2017-01-11 13:42:54 -0500 | [diff] [blame] | 49 | size_t byteLength, SkScalar x, SkScalar y, const SkIRect& regionClipBounds); |
Brian Salomon | f18b1d8 | 2017-10-27 11:30:49 -0400 | [diff] [blame] | 50 | void drawPosText(GrContext*, GrTextUtils::Target*, const GrClip&, const SkPaint&, |
Brian Salomon | 82f4431 | 2017-01-11 13:42:54 -0500 | [diff] [blame] | 51 | const SkMatrix& viewMatrix, const SkSurfaceProps&, const char text[], |
| 52 | size_t byteLength, const SkScalar pos[], int scalarsPerPosition, |
joshualitt | 8e84a1e | 2016-02-16 11:09:25 -0800 | [diff] [blame] | 53 | const SkPoint& offset, const SkIRect& regionClipBounds); |
Brian Salomon | f18b1d8 | 2017-10-27 11:30:49 -0400 | [diff] [blame] | 54 | void drawTextBlob(GrContext*, GrTextUtils::Target*, const GrClip&, const SkPaint&, |
joshualitt | 2c89bc1 | 2016-02-11 05:42:30 -0800 | [diff] [blame] | 55 | const SkMatrix& viewMatrix, const SkSurfaceProps&, const SkTextBlob*, |
Brian Salomon | f18b1d8 | 2017-10-27 11:30:49 -0400 | [diff] [blame] | 56 | SkScalar x, SkScalar y, SkDrawFilter*, const SkIRect& clipBounds); |
joshualitt | 1d89e8d | 2015-04-01 12:40:54 -0700 | [diff] [blame] | 57 | |
Herb Derby | 26cbe51 | 2018-05-24 14:39:01 -0400 | [diff] [blame^] | 58 | std::unique_ptr<GrDrawOp> createOp_TestingOnly(GrContext*, GrTextContext*, |
Robert Phillips | d2e9f76 | 2018-03-07 11:54:37 -0500 | [diff] [blame] | 59 | GrRenderTargetContext*, const SkPaint&, |
| 60 | const SkMatrix& viewMatrix, const char* text, |
| 61 | int x, int y); |
| 62 | |
Khushal | 3e7548c | 2018-05-23 15:45:01 -0700 | [diff] [blame] | 63 | static void SanitizeOptions(Options* options); |
| 64 | static bool CanDrawAsDistanceFields(const SkPaint& skPaint, const SkMatrix& viewMatrix, |
| 65 | const SkSurfaceProps& props, |
| 66 | bool contextSupportsDistanceFieldText, |
| 67 | const Options& options); |
| 68 | static void InitDistanceFieldPaint(GrAtlasTextBlob* blob, |
| 69 | SkPaint* skPaint, |
| 70 | const SkMatrix& viewMatrix, |
| 71 | const Options& options, |
| 72 | SkScalar* textRatio, |
| 73 | SkScalerContextFlags* flags); |
| 74 | |
joshualitt | e55750e | 2016-02-10 12:52:21 -0800 | [diff] [blame] | 75 | private: |
Herb Derby | 26cbe51 | 2018-05-24 14:39:01 -0400 | [diff] [blame^] | 76 | GrTextContext(const Options& options); |
joshualitt | e55750e | 2016-02-10 12:52:21 -0800 | [diff] [blame] | 77 | |
Jim Van Verth | c401bb9 | 2018-02-15 14:05:24 -0500 | [diff] [blame] | 78 | class FallbackTextHelper { |
| 79 | public: |
| 80 | FallbackTextHelper(const SkMatrix& viewMatrix, |
Jim Van Verth | 080a928 | 2018-03-02 10:41:43 -0500 | [diff] [blame] | 81 | const SkPaint& pathPaint, |
| 82 | const GrGlyphCache* glyphCache, |
Jim Van Verth | c401bb9 | 2018-02-15 14:05:24 -0500 | [diff] [blame] | 83 | SkScalar textRatio) |
| 84 | : fViewMatrix(viewMatrix) |
Jim Van Verth | 080a928 | 2018-03-02 10:41:43 -0500 | [diff] [blame] | 85 | , fTextSize(pathPaint.getTextSize()) |
| 86 | , fMaxTextSize(glyphCache->getGlyphSizeLimit()) |
Jim Van Verth | c401bb9 | 2018-02-15 14:05:24 -0500 | [diff] [blame] | 87 | , fTextRatio(textRatio) |
Jim Van Verth | b515ae7 | 2018-05-23 16:44:55 -0400 | [diff] [blame] | 88 | , fTransformedFallbackTextSize(fMaxTextSize) |
| 89 | , fUseTransformedFallback(false) { |
Jim Van Verth | c401bb9 | 2018-02-15 14:05:24 -0500 | [diff] [blame] | 90 | fMaxScale = viewMatrix.getMaxScale(); |
| 91 | } |
| 92 | |
| 93 | void appendText(const SkGlyph& glyph, int count, const char* text, SkPoint glyphPos); |
Robert Phillips | c4039ea | 2018-03-01 11:36:45 -0500 | [diff] [blame] | 94 | void drawText(GrAtlasTextBlob* blob, int runIndex, GrGlyphCache*, const SkSurfaceProps&, |
| 95 | const GrTextUtils::Paint&, SkScalerContextFlags); |
Jim Van Verth | c401bb9 | 2018-02-15 14:05:24 -0500 | [diff] [blame] | 96 | |
| 97 | private: |
| 98 | SkTDArray<char> fFallbackTxt; |
| 99 | SkTDArray<SkPoint> fFallbackPos; |
| 100 | |
| 101 | const SkMatrix& fViewMatrix; |
Jim Van Verth | c401bb9 | 2018-02-15 14:05:24 -0500 | [diff] [blame] | 102 | SkScalar fTextSize; |
| 103 | SkScalar fMaxTextSize; |
Jim Van Verth | 080a928 | 2018-03-02 10:41:43 -0500 | [diff] [blame] | 104 | SkScalar fTextRatio; |
Jim Van Verth | b515ae7 | 2018-05-23 16:44:55 -0400 | [diff] [blame] | 105 | SkScalar fTransformedFallbackTextSize; |
Jim Van Verth | c401bb9 | 2018-02-15 14:05:24 -0500 | [diff] [blame] | 106 | SkScalar fMaxScale; |
Jim Van Verth | b515ae7 | 2018-05-23 16:44:55 -0400 | [diff] [blame] | 107 | bool fUseTransformedFallback; |
Jim Van Verth | c401bb9 | 2018-02-15 14:05:24 -0500 | [diff] [blame] | 108 | }; |
| 109 | |
joshualitt | 1d89e8d | 2015-04-01 12:40:54 -0700 | [diff] [blame] | 110 | // sets up the descriptor on the blob and returns a detached cache. Client must attach |
Herb Derby | d8327a8 | 2018-01-22 14:39:27 -0500 | [diff] [blame] | 111 | static SkColor ComputeCanonicalColor(const SkPaint&, bool lcd); |
brianosman | a1e8f8d | 2016-04-08 06:47:54 -0700 | [diff] [blame] | 112 | // Determines if we need to use fake gamma (and contrast boost): |
Herb Derby | d8327a8 | 2018-01-22 14:39:27 -0500 | [diff] [blame] | 113 | static SkScalerContextFlags ComputeScalerContextFlags(const GrColorSpaceInfo&); |
Brian Salomon | af59748 | 2017-11-07 16:23:34 -0500 | [diff] [blame] | 114 | void regenerateTextBlob(GrAtlasTextBlob* bmp, |
Robert Phillips | c4039ea | 2018-03-01 11:36:45 -0500 | [diff] [blame] | 115 | GrGlyphCache*, |
Brian Salomon | af59748 | 2017-11-07 16:23:34 -0500 | [diff] [blame] | 116 | const GrShaderCaps&, |
| 117 | const GrTextUtils::Paint&, |
Herb Derby | d8327a8 | 2018-01-22 14:39:27 -0500 | [diff] [blame] | 118 | SkScalerContextFlags scalerContextFlags, |
Brian Salomon | af59748 | 2017-11-07 16:23:34 -0500 | [diff] [blame] | 119 | const SkMatrix& viewMatrix, |
| 120 | const SkSurfaceProps&, |
| 121 | const SkTextBlob* blob, SkScalar x, SkScalar y, |
| 122 | SkDrawFilter* drawFilter) const; |
| 123 | |
Herb Derby | d8327a8 | 2018-01-22 14:39:27 -0500 | [diff] [blame] | 124 | static bool HasLCD(const SkTextBlob*); |
joshualitt | 9bd2daf | 2015-04-17 09:30:06 -0700 | [diff] [blame] | 125 | |
Robert Phillips | c4039ea | 2018-03-01 11:36:45 -0500 | [diff] [blame] | 126 | sk_sp<GrAtlasTextBlob> makeDrawTextBlob(GrTextBlobCache*, GrGlyphCache*, |
Herb Derby | d8327a8 | 2018-01-22 14:39:27 -0500 | [diff] [blame] | 127 | const GrShaderCaps&, |
| 128 | const GrTextUtils::Paint&, |
| 129 | SkScalerContextFlags scalerContextFlags, |
| 130 | const SkMatrix& viewMatrix, |
| 131 | const SkSurfaceProps&, |
| 132 | const char text[], size_t byteLength, |
| 133 | SkScalar x, SkScalar y) const; |
Brian Salomon | af59748 | 2017-11-07 16:23:34 -0500 | [diff] [blame] | 134 | |
Robert Phillips | c4039ea | 2018-03-01 11:36:45 -0500 | [diff] [blame] | 135 | sk_sp<GrAtlasTextBlob> makeDrawPosTextBlob(GrTextBlobCache*, GrGlyphCache*, |
Herb Derby | d8327a8 | 2018-01-22 14:39:27 -0500 | [diff] [blame] | 136 | const GrShaderCaps&, |
| 137 | const GrTextUtils::Paint&, |
| 138 | SkScalerContextFlags scalerContextFlags, |
| 139 | const SkMatrix& viewMatrix, |
| 140 | const SkSurfaceProps&, |
| 141 | const char text[], size_t byteLength, |
| 142 | const SkScalar pos[], |
| 143 | int scalarsPerPosition, |
| 144 | const SkPoint& offset) const; |
Brian Salomon | 52db940 | 2017-11-07 14:58:55 -0500 | [diff] [blame] | 145 | |
| 146 | // Functions for appending BMP text to GrAtlasTextBlob |
Robert Phillips | c4039ea | 2018-03-01 11:36:45 -0500 | [diff] [blame] | 147 | static void DrawBmpText(GrAtlasTextBlob*, int runIndex, GrGlyphCache*, |
Brian Salomon | 52db940 | 2017-11-07 14:58:55 -0500 | [diff] [blame] | 148 | const SkSurfaceProps&, const GrTextUtils::Paint& paint, |
Herb Derby | d8327a8 | 2018-01-22 14:39:27 -0500 | [diff] [blame] | 149 | SkScalerContextFlags scalerContextFlags, const SkMatrix& viewMatrix, |
Brian Salomon | 52db940 | 2017-11-07 14:58:55 -0500 | [diff] [blame] | 150 | const char text[], size_t byteLength, SkScalar x, SkScalar y); |
| 151 | |
Robert Phillips | c4039ea | 2018-03-01 11:36:45 -0500 | [diff] [blame] | 152 | static void DrawBmpPosText(GrAtlasTextBlob*, int runIndex, GrGlyphCache*, |
Brian Salomon | 52db940 | 2017-11-07 14:58:55 -0500 | [diff] [blame] | 153 | const SkSurfaceProps&, const GrTextUtils::Paint& paint, |
Herb Derby | d8327a8 | 2018-01-22 14:39:27 -0500 | [diff] [blame] | 154 | SkScalerContextFlags scalerContextFlags, const SkMatrix& viewMatrix, |
Brian Salomon | 52db940 | 2017-11-07 14:58:55 -0500 | [diff] [blame] | 155 | const char text[], size_t byteLength, const SkScalar pos[], |
Jim Van Verth | c401bb9 | 2018-02-15 14:05:24 -0500 | [diff] [blame] | 156 | int scalarsPerPosition, const SkPoint& offset); |
Brian Salomon | 52db940 | 2017-11-07 14:58:55 -0500 | [diff] [blame] | 157 | |
Robert Phillips | c4039ea | 2018-03-01 11:36:45 -0500 | [diff] [blame] | 158 | static void DrawBmpTextAsPaths(GrAtlasTextBlob*, int runIndex, GrGlyphCache*, |
Jim Van Verth | c401bb9 | 2018-02-15 14:05:24 -0500 | [diff] [blame] | 159 | const SkSurfaceProps&, const GrTextUtils::Paint& paint, |
| 160 | SkScalerContextFlags scalerContextFlags, |
| 161 | const SkMatrix& viewMatrix, const char text[], |
Jim Van Verth | 54d9c88 | 2018-02-08 16:14:48 -0500 | [diff] [blame] | 162 | size_t byteLength, SkScalar x, SkScalar y); |
| 163 | |
Robert Phillips | c4039ea | 2018-03-01 11:36:45 -0500 | [diff] [blame] | 164 | static void DrawBmpPosTextAsPaths(GrAtlasTextBlob*, int runIndex, GrGlyphCache*, |
Jim Van Verth | c401bb9 | 2018-02-15 14:05:24 -0500 | [diff] [blame] | 165 | const SkSurfaceProps&, const GrTextUtils::Paint& paint, |
| 166 | SkScalerContextFlags scalerContextFlags, |
| 167 | const SkMatrix& viewMatrix, |
Jim Van Verth | 54d9c88 | 2018-02-08 16:14:48 -0500 | [diff] [blame] | 168 | const char text[], size_t byteLength, |
| 169 | const SkScalar pos[], int scalarsPerPosition, |
Jim Van Verth | c401bb9 | 2018-02-15 14:05:24 -0500 | [diff] [blame] | 170 | const SkPoint& offset); |
Jim Van Verth | 54d9c88 | 2018-02-08 16:14:48 -0500 | [diff] [blame] | 171 | |
Brian Salomon | 52db940 | 2017-11-07 14:58:55 -0500 | [diff] [blame] | 172 | // functions for appending distance field text |
Robert Phillips | c4039ea | 2018-03-01 11:36:45 -0500 | [diff] [blame] | 173 | void drawDFText(GrAtlasTextBlob* blob, int runIndex, GrGlyphCache*, const SkSurfaceProps&, |
Herb Derby | d8327a8 | 2018-01-22 14:39:27 -0500 | [diff] [blame] | 174 | const GrTextUtils::Paint& paint, SkScalerContextFlags scalerContextFlags, |
Brian Salomon | af59748 | 2017-11-07 16:23:34 -0500 | [diff] [blame] | 175 | const SkMatrix& viewMatrix, const char text[], size_t byteLength, SkScalar x, |
| 176 | SkScalar y) const; |
Brian Salomon | 52db940 | 2017-11-07 14:58:55 -0500 | [diff] [blame] | 177 | |
Robert Phillips | c4039ea | 2018-03-01 11:36:45 -0500 | [diff] [blame] | 178 | void drawDFPosText(GrAtlasTextBlob* blob, int runIndex, GrGlyphCache*, |
Brian Salomon | af59748 | 2017-11-07 16:23:34 -0500 | [diff] [blame] | 179 | const SkSurfaceProps&, const GrTextUtils::Paint& paint, |
Herb Derby | d8327a8 | 2018-01-22 14:39:27 -0500 | [diff] [blame] | 180 | SkScalerContextFlags scalerContextFlags, |
| 181 | const SkMatrix& viewMatrix, const char text[], |
Brian Salomon | af59748 | 2017-11-07 16:23:34 -0500 | [diff] [blame] | 182 | size_t byteLength, const SkScalar pos[], int scalarsPerPosition, |
| 183 | const SkPoint& offset) const; |
Brian Salomon | 52db940 | 2017-11-07 14:58:55 -0500 | [diff] [blame] | 184 | |
Robert Phillips | c4039ea | 2018-03-01 11:36:45 -0500 | [diff] [blame] | 185 | static void BmpAppendGlyph(GrAtlasTextBlob*, int runIndex, GrGlyphCache*, |
Robert Phillips | caf1ebb | 2018-03-01 14:28:44 -0500 | [diff] [blame] | 186 | sk_sp<GrTextStrike>*, const SkGlyph&, SkScalar sx, SkScalar sy, |
Jim Van Verth | b515ae7 | 2018-05-23 16:44:55 -0400 | [diff] [blame] | 187 | GrColor color, SkGlyphCache*, SkScalar textRatio, bool needsXform); |
Brian Salomon | 52db940 | 2017-11-07 14:58:55 -0500 | [diff] [blame] | 188 | |
Robert Phillips | c4039ea | 2018-03-01 11:36:45 -0500 | [diff] [blame] | 189 | static void DfAppendGlyph(GrAtlasTextBlob*, int runIndex, GrGlyphCache*, |
Robert Phillips | caf1ebb | 2018-03-01 14:28:44 -0500 | [diff] [blame] | 190 | sk_sp<GrTextStrike>*, const SkGlyph&, SkScalar sx, SkScalar sy, |
Jim Van Verth | f4c1316 | 2018-01-11 16:40:24 -0500 | [diff] [blame] | 191 | GrColor color, SkGlyphCache* cache, SkScalar textRatio); |
Brian Salomon | 52db940 | 2017-11-07 14:58:55 -0500 | [diff] [blame] | 192 | |
Hal Canary | 144caf5 | 2016-11-07 17:57:18 -0500 | [diff] [blame] | 193 | const GrDistanceFieldAdjustTable* dfAdjustTable() const { return fDistanceAdjustTable.get(); } |
joshualitt | 79dfb2b | 2015-05-11 08:58:08 -0700 | [diff] [blame] | 194 | |
Hal Canary | 144caf5 | 2016-11-07 17:57:18 -0500 | [diff] [blame] | 195 | sk_sp<const GrDistanceFieldAdjustTable> fDistanceAdjustTable; |
joshualitt | 1d89e8d | 2015-04-01 12:40:54 -0700 | [diff] [blame] | 196 | |
Khushal | 3e7548c | 2018-05-23 15:45:01 -0700 | [diff] [blame] | 197 | Options fOptions; |
Brian Salomon | af59748 | 2017-11-07 16:23:34 -0500 | [diff] [blame] | 198 | |
Hal Canary | 6f6961e | 2017-01-31 13:50:44 -0500 | [diff] [blame] | 199 | #if GR_TEST_UTILS |
Herb Derby | d8327a8 | 2018-01-22 14:39:27 -0500 | [diff] [blame] | 200 | static const SkScalerContextFlags kTextBlobOpScalerContextFlags = |
| 201 | SkScalerContextFlags::kFakeGammaAndBoostContrast; |
Brian Salomon | 44acb5b | 2017-07-18 19:59:24 -0400 | [diff] [blame] | 202 | GR_DRAW_OP_TEST_FRIEND(GrAtlasTextOp); |
joshualitt | 79dfb2b | 2015-05-11 08:58:08 -0700 | [diff] [blame] | 203 | #endif |
joshualitt | 1d89e8d | 2015-04-01 12:40:54 -0700 | [diff] [blame] | 204 | }; |
| 205 | |
| 206 | #endif |