blob: 2eae03b8d97f420c5fb5ea4478376e16d025a432 [file] [log] [blame]
/*
* Copyright 2020 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "src/text/gpu/SDFTControl.h"
#include "include/core/SkFont.h"
#include "include/core/SkGraphics.h"
#include "include/core/SkMatrix.h"
#include "include/core/SkPaint.h"
#include "include/core/SkScalar.h"
#include "include/core/SkSurfaceProps.h"
#include "src/core/SkFontPriv.h"
#include "src/core/SkGlyphRunPainter.h"
#include "src/core/SkReadBuffer.h"
#include <tuple>
namespace sktext::gpu {
#if !defined(SK_DISABLE_SDF_TEXT)
// DF sizes and thresholds for usage of the small and medium sizes. For example, above
// kSmallDFFontLimit we will use the medium size. The large size is used up until the size at
// which we switch over to drawing as paths as controlled by Control.
static const int kSmallDFFontLimit = 32;
static const int kMediumDFFontLimit = 72;
static const int kLargeDFFontLimit = 162;
#ifdef SK_BUILD_FOR_MAC
static const int kExtraLargeDFFontLimit = 256;
#endif
SkScalar SDFTControl::MinSDFTRange(bool useSDFTForSmallText, SkScalar min) {
if (!useSDFTForSmallText) {
return kLargeDFFontLimit;
}
return min;
}
SDFTControl::SDFTControl(
bool ableToUseSDFT, bool useSDFTForSmallText, bool useSDFTForPerspectiveText,
SkScalar min, SkScalar max)
: fMinDistanceFieldFontSize{MinSDFTRange(useSDFTForSmallText, min)}
, fMaxDistanceFieldFontSize{max}
, fAbleToUseSDFT{ableToUseSDFT}
, fAbleToUsePerspectiveSDFT{useSDFTForPerspectiveText} {
SkASSERT_RELEASE(0 < min && min <= max);
}
#endif // !defined(SK_DISABLE_SDF_TEXT)
bool SDFTControl::isDirect(SkScalar approximateDeviceTextSize, const SkPaint& paint,
const SkMatrix& matrix) const {
#if !defined(SK_DISABLE_SDF_TEXT)
const bool isSDFT = this->isSDFT(approximateDeviceTextSize, paint, matrix);
#else
const bool isSDFT = false;
#endif
return !isSDFT &&
!matrix.hasPerspective() &&
0 < approximateDeviceTextSize &&
approximateDeviceTextSize < SkGlyphDigest::kSkSideTooBigForAtlas;
}
#if !defined(SK_DISABLE_SDF_TEXT)
bool SDFTControl::isSDFT(SkScalar approximateDeviceTextSize, const SkPaint& paint,
const SkMatrix& matrix) const {
return fAbleToUseSDFT &&
paint.getMaskFilter() == nullptr &&
paint.getStyle() == SkPaint::kFill_Style &&
0 < approximateDeviceTextSize &&
(fAbleToUsePerspectiveSDFT || !matrix.hasPerspective()) &&
(fMinDistanceFieldFontSize <= approximateDeviceTextSize || matrix.hasPerspective()) &&
approximateDeviceTextSize <= fMaxDistanceFieldFontSize;
}
std::tuple<SkFont, SkScalar, SDFTMatrixRange>
SDFTControl::getSDFFont(const SkFont& font, const SkMatrix& viewMatrix,
const SkPoint& textLoc) const {
SkScalar textSize = font.getSize();
SkScalar scaledTextSize = SkFontPriv::ApproximateTransformedTextSize(font, viewMatrix, textLoc);
if (scaledTextSize <= 0 || SkScalarNearlyEqual(textSize, scaledTextSize)) {
scaledTextSize = textSize;
}
SkFont dfFont{font};
SkScalar dfMaskScaleFloor;
SkScalar dfMaskScaleCeil;
if (scaledTextSize <= kSmallDFFontLimit) {
dfMaskScaleFloor = fMinDistanceFieldFontSize;
dfMaskScaleCeil = kSmallDFFontLimit;
} else if (scaledTextSize <= kMediumDFFontLimit) {
dfMaskScaleFloor = kSmallDFFontLimit;
dfMaskScaleCeil = kMediumDFFontLimit;
#ifdef SK_BUILD_FOR_MAC
} else if (scaledTextSize <= kLargeDFFontLimit) {
dfMaskScaleFloor = kMediumDFFontLimit;
dfMaskScaleCeil = kLargeDFFontLimit;
} else {
dfMaskScaleFloor = kLargeDFFontLimit;
dfMaskScaleCeil = kExtraLargeDFFontLimit;
}
#else
} else {
dfMaskScaleFloor = kMediumDFFontLimit;
dfMaskScaleCeil = kLargeDFFontLimit;
}
#endif
dfFont.setSize(SkIntToScalar(dfMaskScaleCeil));
dfFont.setEdging(SkFont::Edging::kAntiAlias);
dfFont.setForceAutoHinting(false);
dfFont.setHinting(SkFontHinting::kNormal);
// The sub-pixel position will always happen when transforming to the screen.
dfFont.setSubpixel(false);
SkScalar minMatrixScale = dfMaskScaleFloor / textSize,
maxMatrixScale = dfMaskScaleCeil / textSize;
return {dfFont, textSize / dfMaskScaleCeil, {minMatrixScale, maxMatrixScale}};
}
bool SDFTMatrixRange::matrixInRange(const SkMatrix& matrix) const {
SkScalar maxScale = matrix.getMaxScale();
return fMatrixMin < maxScale && maxScale <= fMatrixMax;
}
void SDFTMatrixRange::flatten(SkWriteBuffer& buffer) const {
buffer.writeScalar(fMatrixMin);
buffer.writeScalar(fMatrixMax);
}
SDFTMatrixRange SDFTMatrixRange::MakeFromBuffer(SkReadBuffer& buffer) {
SkScalar min = buffer.readScalar();
SkScalar max = buffer.readScalar();
return SDFTMatrixRange{min, max};
}
#endif // !defined(SK_DISABLE_SDF_TEXT)
} // namespace sktext::gpu