blob: 4e9b42b08821065c8abb93aa79d5cc5ccd2e236f [file] [log] [blame]
joshualitt1d89e8d2015-04-01 12:40:54 -07001/*
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 */
Hal Canaryc640d0d2018-06-13 09:59:02 -04007
Herb Derby26cbe512018-05-24 14:39:01 -04008#include "GrTextContext.h"
Hal Canaryc640d0d2018-06-13 09:59:02 -04009
Brian Salomonc7fe0f72018-05-11 10:14:21 -040010#include "GrCaps.h"
robertphillips73c4e642016-03-02 11:36:59 -080011#include "GrContext.h"
Robert Phillipsf35fd8d2018-01-22 10:48:15 -050012#include "GrContextPriv.h"
Jim Van Verthd401da62018-05-03 10:40:30 -040013#include "GrSDFMaskFilter.h"
joshualittb7133be2015-04-08 09:08:31 -070014#include "GrTextBlobCache.h"
Brian Salomon52db9402017-11-07 14:58:55 -050015#include "SkDistanceFieldGen.h"
joshualitt1d89e8d2015-04-01 12:40:54 -070016#include "SkDraw.h"
Jim Van Verth54d9c882018-02-08 16:14:48 -050017#include "SkDrawProcs.h"
Hal Canaryc640d0d2018-06-13 09:59:02 -040018#include "SkGlyphRun.h"
Brian Osman3b655982017-03-07 16:58:08 -050019#include "SkGr.h"
Jim Van Verthc65b65d2018-01-16 16:26:35 -050020#include "SkGraphics.h"
Brian Salomonaf597482017-11-07 16:23:34 -050021#include "SkMakeUnique.h"
Mike Reed80747ef2018-01-23 15:29:32 -050022#include "SkMaskFilterBase.h"
Herb Derbyeb3f6742018-03-05 14:36:45 -050023#include "SkPaintPriv.h"
Hal Canaryc640d0d2018-06-13 09:59:02 -040024#include "SkTo.h"
Brian Salomon649a3412017-03-09 13:50:43 -050025#include "ops/GrMeshDrawOp.h"
joshualitt1d89e8d2015-04-01 12:40:54 -070026
Brian Salomonaf597482017-11-07 16:23:34 -050027// DF sizes and thresholds for usage of the small and medium sizes. For example, above
28// kSmallDFFontLimit we will use the medium size. The large size is used up until the size at
29// which we switch over to drawing as paths as controlled by Options.
30static const int kSmallDFFontSize = 32;
31static const int kSmallDFFontLimit = 32;
Jim Van Verth825d4d72018-01-30 20:37:03 +000032static const int kMediumDFFontSize = 72;
33static const int kMediumDFFontLimit = 72;
34static const int kLargeDFFontSize = 162;
Brian Salomonaf597482017-11-07 16:23:34 -050035
36static const int kDefaultMinDistanceFieldFontSize = 18;
37#ifdef SK_BUILD_FOR_ANDROID
38static const int kDefaultMaxDistanceFieldFontSize = 384;
39#else
40static const int kDefaultMaxDistanceFieldFontSize = 2 * kLargeDFFontSize;
41#endif
42
Herb Derby26cbe512018-05-24 14:39:01 -040043GrTextContext::GrTextContext(const Options& options)
Khushal3e7548c2018-05-23 15:45:01 -070044 : fDistanceAdjustTable(new GrDistanceFieldAdjustTable), fOptions(options) {
45 SanitizeOptions(&fOptions);
joshualitt9bd2daf2015-04-17 09:30:06 -070046}
47
Herb Derby26cbe512018-05-24 14:39:01 -040048std::unique_ptr<GrTextContext> GrTextContext::Make(const Options& options) {
49 return std::unique_ptr<GrTextContext>(new GrTextContext(options));
joshualitt1d89e8d2015-04-01 12:40:54 -070050}
51
Herb Derby26cbe512018-05-24 14:39:01 -040052SkColor GrTextContext::ComputeCanonicalColor(const SkPaint& paint, bool lcd) {
Brian Salomon6f1d36c2017-01-13 12:02:17 -050053 SkColor canonicalColor = paint.computeLuminanceColor();
joshualitt9e36c1a2015-04-14 12:17:27 -070054 if (lcd) {
55 // This is the correct computation, but there are tons of cases where LCD can be overridden.
56 // For now we just regenerate if any run in a textblob has LCD.
57 // TODO figure out where all of these overrides are and see if we can incorporate that logic
58 // at a higher level *OR* use sRGB
59 SkASSERT(false);
60 //canonicalColor = SkMaskGamma::CanonicalColor(canonicalColor);
61 } else {
62 // A8, though can have mixed BMP text but it shouldn't matter because BMP text won't have
63 // gamma corrected masks anyways, nor color
64 U8CPU lum = SkComputeLuminance(SkColorGetR(canonicalColor),
65 SkColorGetG(canonicalColor),
66 SkColorGetB(canonicalColor));
67 // reduce to our finite number of bits
68 canonicalColor = SkMaskGamma::CanonicalColor(SkColorSetRGB(lum, lum, lum));
69 }
70 return canonicalColor;
71}
72
Herb Derby26cbe512018-05-24 14:39:01 -040073SkScalerContextFlags GrTextContext::ComputeScalerContextFlags(
Herb Derbyd8327a82018-01-22 14:39:27 -050074 const GrColorSpaceInfo& colorSpaceInfo) {
Brian Osman34ec3742018-07-03 10:40:57 -040075 // If we're doing linear blending, then we can disable the gamma hacks.
brianosman5280dcb2016-04-14 06:02:59 -070076 // Otherwise, leave them on. In either case, we still want the contrast boost:
Brian Osman34ec3742018-07-03 10:40:57 -040077 // TODO: Can we be even smarter about mask gamma based on the dest transfer function?
78 if (colorSpaceInfo.isLinearlyBlended()) {
Herb Derbyd8327a82018-01-22 14:39:27 -050079 return SkScalerContextFlags::kBoostContrast;
brianosman32f77822016-04-07 06:25:45 -070080 } else {
Herb Derbyd8327a82018-01-22 14:39:27 -050081 return SkScalerContextFlags::kFakeGammaAndBoostContrast;
brianosman32f77822016-04-07 06:25:45 -070082 }
83}
84
Herb Derby26cbe512018-05-24 14:39:01 -040085void GrTextContext::SanitizeOptions(Options* options) {
Khushal3e7548c2018-05-23 15:45:01 -070086 if (options->fMaxDistanceFieldFontSize < 0.f) {
87 options->fMaxDistanceFieldFontSize = kDefaultMaxDistanceFieldFontSize;
88 }
89 if (options->fMinDistanceFieldFontSize < 0.f) {
90 options->fMinDistanceFieldFontSize = kDefaultMinDistanceFieldFontSize;
91 }
92}
93
Mike Reedf2b074e2018-12-03 16:52:59 -050094bool GrTextContext::CanDrawAsDistanceFields(const SkPaint& paint, const SkFont& font,
95 const SkMatrix& viewMatrix,
Herb Derby26cbe512018-05-24 14:39:01 -040096 const SkSurfaceProps& props,
97 bool contextSupportsDistanceFieldText,
98 const Options& options) {
Brian Salomon52db9402017-11-07 14:58:55 -050099 if (!viewMatrix.hasPerspective()) {
100 SkScalar maxScale = viewMatrix.getMaxScale();
Mike Reedf2b074e2018-12-03 16:52:59 -0500101 SkScalar scaledTextSize = maxScale * font.getSize();
Brian Salomon52db9402017-11-07 14:58:55 -0500102 // Hinted text looks far better at small resolutions
103 // Scaling up beyond 2x yields undesireable artifacts
Khushal3e7548c2018-05-23 15:45:01 -0700104 if (scaledTextSize < options.fMinDistanceFieldFontSize ||
105 scaledTextSize > options.fMaxDistanceFieldFontSize) {
Brian Salomon52db9402017-11-07 14:58:55 -0500106 return false;
107 }
108
109 bool useDFT = props.isUseDeviceIndependentFonts();
110#if SK_FORCE_DISTANCE_FIELD_TEXT
111 useDFT = true;
112#endif
113
114 if (!useDFT && scaledTextSize < kLargeDFFontSize) {
115 return false;
116 }
117 }
118
Mike Reed8ad91a92018-01-19 19:09:32 -0500119 // mask filters modify alpha, which doesn't translate well to distance
Mike Reedf2b074e2018-12-03 16:52:59 -0500120 if (paint.getMaskFilter() || !contextSupportsDistanceFieldText) {
Brian Salomon52db9402017-11-07 14:58:55 -0500121 return false;
122 }
123
124 // TODO: add some stroking support
Mike Reedf2b074e2018-12-03 16:52:59 -0500125 if (paint.getStyle() != SkPaint::kFill_Style) {
Brian Salomon52db9402017-11-07 14:58:55 -0500126 return false;
127 }
128
129 return true;
130}
131
Herb Derby86240592018-05-24 16:12:31 -0400132void GrTextContext::InitDistanceFieldPaint(GrTextBlob* blob,
Herb Derby26cbe512018-05-24 14:39:01 -0400133 SkPaint* skPaint,
134 const SkMatrix& viewMatrix,
135 const Options& options,
136 SkScalar* textRatio,
137 SkScalerContextFlags* flags) {
Brian Salomon52db9402017-11-07 14:58:55 -0500138 SkScalar textSize = skPaint->getTextSize();
139 SkScalar scaledTextSize = textSize;
140
141 if (viewMatrix.hasPerspective()) {
142 // for perspective, we simply force to the medium size
143 // TODO: compute a size based on approximate screen area
144 scaledTextSize = kMediumDFFontLimit;
145 } else {
146 SkScalar maxScale = viewMatrix.getMaxScale();
147 // if we have non-unity scale, we need to choose our base text size
148 // based on the SkPaint's text size multiplied by the max scale factor
149 // TODO: do we need to do this if we're scaling down (i.e. maxScale < 1)?
150 if (maxScale > 0 && !SkScalarNearlyEqual(maxScale, SK_Scalar1)) {
151 scaledTextSize *= maxScale;
152 }
153 }
154
155 // We have three sizes of distance field text, and within each size 'bucket' there is a floor
156 // and ceiling. A scale outside of this range would require regenerating the distance fields
157 SkScalar dfMaskScaleFloor;
158 SkScalar dfMaskScaleCeil;
159 if (scaledTextSize <= kSmallDFFontLimit) {
Khushal3e7548c2018-05-23 15:45:01 -0700160 dfMaskScaleFloor = options.fMinDistanceFieldFontSize;
Brian Salomon52db9402017-11-07 14:58:55 -0500161 dfMaskScaleCeil = kSmallDFFontLimit;
162 *textRatio = textSize / kSmallDFFontSize;
163 skPaint->setTextSize(SkIntToScalar(kSmallDFFontSize));
164 } else if (scaledTextSize <= kMediumDFFontLimit) {
165 dfMaskScaleFloor = kSmallDFFontLimit;
166 dfMaskScaleCeil = kMediumDFFontLimit;
167 *textRatio = textSize / kMediumDFFontSize;
168 skPaint->setTextSize(SkIntToScalar(kMediumDFFontSize));
169 } else {
170 dfMaskScaleFloor = kMediumDFFontLimit;
Khushal3e7548c2018-05-23 15:45:01 -0700171 dfMaskScaleCeil = options.fMaxDistanceFieldFontSize;
Brian Salomon52db9402017-11-07 14:58:55 -0500172 *textRatio = textSize / kLargeDFFontSize;
173 skPaint->setTextSize(SkIntToScalar(kLargeDFFontSize));
174 }
175
176 // Because there can be multiple runs in the blob, we want the overall maxMinScale, and
177 // minMaxScale to make regeneration decisions. Specifically, we want the maximum minimum scale
178 // we can tolerate before we'd drop to a lower mip size, and the minimum maximum scale we can
179 // tolerate before we'd have to move to a large mip size. When we actually test these values
180 // we look at the delta in scale between the new viewmatrix and the old viewmatrix, and test
181 // against these values to decide if we can reuse or not(ie, will a given scale change our mip
182 // level)
183 SkASSERT(dfMaskScaleFloor <= scaledTextSize && scaledTextSize <= dfMaskScaleCeil);
Khushal3e7548c2018-05-23 15:45:01 -0700184 if (blob) {
185 blob->setMinAndMaxScale(dfMaskScaleFloor / scaledTextSize,
186 dfMaskScaleCeil / scaledTextSize);
187 }
Brian Salomon52db9402017-11-07 14:58:55 -0500188
189 skPaint->setAntiAlias(true);
190 skPaint->setLCDRenderText(false);
191 skPaint->setAutohinted(false);
Mike Reed9edbf422018-11-07 19:54:33 -0500192 skPaint->setHinting(kNormal_SkFontHinting);
Brian Salomon52db9402017-11-07 14:58:55 -0500193 skPaint->setSubpixelText(true);
Jim Van Verthd401da62018-05-03 10:40:30 -0400194
195 skPaint->setMaskFilter(GrSDFMaskFilter::Make());
Khushal3e7548c2018-05-23 15:45:01 -0700196
197 // We apply the fake-gamma by altering the distance in the shader, so we ignore the
198 // passed-in scaler context flags. (It's only used when we fall-back to bitmap text).
199 *flags = SkScalerContextFlags::kNone;
Brian Salomon52db9402017-11-07 14:58:55 -0500200}
201
joshualitt79dfb2b2015-05-11 08:58:08 -0700202///////////////////////////////////////////////////////////////////////////////////////////////////
Herb Derbyf4f6bbf2018-07-27 11:58:37 -0400203
Hal Canary6f6961e2017-01-31 13:50:44 -0500204#if GR_TEST_UTILS
joshualitt79dfb2b2015-05-11 08:58:08 -0700205
Brian Salomonf18b1d82017-10-27 11:30:49 -0400206#include "GrRenderTargetContext.h"
207
Brian Salomon44acb5b2017-07-18 19:59:24 -0400208GR_DRAW_OP_TEST_DEFINE(GrAtlasTextOp) {
joshualitt79dfb2b2015-05-11 08:58:08 -0700209 static uint32_t gContextID = SK_InvalidGenID;
Herb Derby26cbe512018-05-24 14:39:01 -0400210 static std::unique_ptr<GrTextContext> gTextContext;
robertphillipsfcf78292015-06-19 11:49:52 -0700211 static SkSurfaceProps gSurfaceProps(SkSurfaceProps::kLegacyFontHost_InitType);
joshualitt79dfb2b2015-05-11 08:58:08 -0700212
213 if (context->uniqueID() != gContextID) {
214 gContextID = context->uniqueID();
Herb Derby26cbe512018-05-24 14:39:01 -0400215 gTextContext = GrTextContext::Make(GrTextContext::Options());
joshualitt79dfb2b2015-05-11 08:58:08 -0700216 }
217
Greg Daniel4065d452018-11-16 15:43:41 -0500218 const GrBackendFormat format =
219 context->contextPriv().caps()->getBackendFormatFromColorType(kRGBA_8888_SkColorType);
220
Brian Osman11052242016-10-27 14:47:55 -0400221 // Setup dummy SkPaint / GrPaint / GrRenderTargetContext
Robert Phillips0c4b7b12018-03-06 08:20:37 -0500222 sk_sp<GrRenderTargetContext> rtc(context->contextPriv().makeDeferredRenderTargetContext(
Greg Daniel4065d452018-11-16 15:43:41 -0500223 format, SkBackingFit::kApprox, 1024, 1024, kRGBA_8888_GrPixelConfig, nullptr));
brianosman8fe485b2016-07-25 12:31:51 -0700224
joshualitt6c891102015-05-13 08:51:49 -0700225 SkMatrix viewMatrix = GrTest::TestMatrixInvertible(random);
Brian Salomon44acb5b2017-07-18 19:59:24 -0400226
joshualitt79dfb2b2015-05-11 08:58:08 -0700227 SkPaint skPaint;
Brian Salomon6f1d36c2017-01-13 12:02:17 -0500228 skPaint.setColor(random->nextU());
joshualitt79dfb2b2015-05-11 08:58:08 -0700229 skPaint.setLCDRenderText(random->nextBool());
230 skPaint.setAntiAlias(skPaint.isLCDRenderText() ? true : random->nextBool());
231 skPaint.setSubpixelText(random->nextBool());
232
joshualitt79dfb2b2015-05-11 08:58:08 -0700233 const char* text = "The quick brown fox jumps over the lazy dog.";
joshualitt79dfb2b2015-05-11 08:58:08 -0700234
joshualittb8d86492016-02-24 09:23:03 -0800235 // create some random x/y offsets, including negative offsets
236 static const int kMaxTrans = 1024;
237 int xPos = (random->nextU() % 2) * 2 - 1;
238 int yPos = (random->nextU() % 2) * 2 - 1;
239 int xInt = (random->nextU() % kMaxTrans) * xPos;
240 int yInt = (random->nextU() % kMaxTrans) * yPos;
halcanary9d524f22016-03-29 09:03:52 -0700241
Robert Phillipsd2e9f762018-03-07 11:54:37 -0500242 return gTextContext->createOp_TestingOnly(context, gTextContext.get(), rtc.get(),
243 skPaint, viewMatrix, text, xInt, yInt);
joshualitt79dfb2b2015-05-11 08:58:08 -0700244}
245
246#endif