1*c8dee2aaSAndroid Build Coastguard Worker // Copyright 2019 Google LLC.
2*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkColor.h"
3*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkFontStyle.h"
4*c8dee2aaSAndroid Build Coastguard Worker #include "modules/skparagraph/include/TextStyle.h"
5*c8dee2aaSAndroid Build Coastguard Worker
6*c8dee2aaSAndroid Build Coastguard Worker namespace skia {
7*c8dee2aaSAndroid Build Coastguard Worker namespace textlayout {
8*c8dee2aaSAndroid Build Coastguard Worker
9*c8dee2aaSAndroid Build Coastguard Worker const std::vector<SkString>* TextStyle::kDefaultFontFamilies =
10*c8dee2aaSAndroid Build Coastguard Worker new std::vector<SkString>{SkString(DEFAULT_FONT_FAMILY)};
11*c8dee2aaSAndroid Build Coastguard Worker
cloneForPlaceholder()12*c8dee2aaSAndroid Build Coastguard Worker TextStyle TextStyle::cloneForPlaceholder() {
13*c8dee2aaSAndroid Build Coastguard Worker TextStyle result;
14*c8dee2aaSAndroid Build Coastguard Worker result.fColor = fColor;
15*c8dee2aaSAndroid Build Coastguard Worker result.fFontSize = fFontSize;
16*c8dee2aaSAndroid Build Coastguard Worker result.fFontFamilies = fFontFamilies;
17*c8dee2aaSAndroid Build Coastguard Worker result.fDecoration = fDecoration;
18*c8dee2aaSAndroid Build Coastguard Worker result.fHasBackground = fHasBackground;
19*c8dee2aaSAndroid Build Coastguard Worker result.fHasForeground = fHasForeground;
20*c8dee2aaSAndroid Build Coastguard Worker result.fBackground = fBackground;
21*c8dee2aaSAndroid Build Coastguard Worker result.fForeground = fForeground;
22*c8dee2aaSAndroid Build Coastguard Worker result.fHeightOverride = fHeightOverride;
23*c8dee2aaSAndroid Build Coastguard Worker result.fIsPlaceholder = true;
24*c8dee2aaSAndroid Build Coastguard Worker result.fFontFeatures = fFontFeatures;
25*c8dee2aaSAndroid Build Coastguard Worker result.fHalfLeading = fHalfLeading;
26*c8dee2aaSAndroid Build Coastguard Worker result.fBaselineShift = fBaselineShift;
27*c8dee2aaSAndroid Build Coastguard Worker result.fFontArguments = fFontArguments;
28*c8dee2aaSAndroid Build Coastguard Worker return result;
29*c8dee2aaSAndroid Build Coastguard Worker }
30*c8dee2aaSAndroid Build Coastguard Worker
equals(const TextStyle & other) const31*c8dee2aaSAndroid Build Coastguard Worker bool TextStyle::equals(const TextStyle& other) const {
32*c8dee2aaSAndroid Build Coastguard Worker
33*c8dee2aaSAndroid Build Coastguard Worker if (fIsPlaceholder || other.fIsPlaceholder) {
34*c8dee2aaSAndroid Build Coastguard Worker return false;
35*c8dee2aaSAndroid Build Coastguard Worker }
36*c8dee2aaSAndroid Build Coastguard Worker
37*c8dee2aaSAndroid Build Coastguard Worker if (fColor != other.fColor) {
38*c8dee2aaSAndroid Build Coastguard Worker return false;
39*c8dee2aaSAndroid Build Coastguard Worker }
40*c8dee2aaSAndroid Build Coastguard Worker if (!(fDecoration == other.fDecoration)) {
41*c8dee2aaSAndroid Build Coastguard Worker return false;
42*c8dee2aaSAndroid Build Coastguard Worker }
43*c8dee2aaSAndroid Build Coastguard Worker if (!(fFontStyle == other.fFontStyle)) {
44*c8dee2aaSAndroid Build Coastguard Worker return false;
45*c8dee2aaSAndroid Build Coastguard Worker }
46*c8dee2aaSAndroid Build Coastguard Worker if (fFontFamilies != other.fFontFamilies) {
47*c8dee2aaSAndroid Build Coastguard Worker return false;
48*c8dee2aaSAndroid Build Coastguard Worker }
49*c8dee2aaSAndroid Build Coastguard Worker if (fLetterSpacing != other.fLetterSpacing) {
50*c8dee2aaSAndroid Build Coastguard Worker return false;
51*c8dee2aaSAndroid Build Coastguard Worker }
52*c8dee2aaSAndroid Build Coastguard Worker if (fWordSpacing != other.fWordSpacing) {
53*c8dee2aaSAndroid Build Coastguard Worker return false;
54*c8dee2aaSAndroid Build Coastguard Worker }
55*c8dee2aaSAndroid Build Coastguard Worker if (fHeight != other.fHeight) {
56*c8dee2aaSAndroid Build Coastguard Worker return false;
57*c8dee2aaSAndroid Build Coastguard Worker }
58*c8dee2aaSAndroid Build Coastguard Worker if (fHeightOverride != other.fHeightOverride) {
59*c8dee2aaSAndroid Build Coastguard Worker return false;
60*c8dee2aaSAndroid Build Coastguard Worker }
61*c8dee2aaSAndroid Build Coastguard Worker if (fHalfLeading != other.fHalfLeading) {
62*c8dee2aaSAndroid Build Coastguard Worker return false;
63*c8dee2aaSAndroid Build Coastguard Worker }
64*c8dee2aaSAndroid Build Coastguard Worker if (fBaselineShift != other.fBaselineShift) {
65*c8dee2aaSAndroid Build Coastguard Worker return false;
66*c8dee2aaSAndroid Build Coastguard Worker }
67*c8dee2aaSAndroid Build Coastguard Worker if (fFontSize != other.fFontSize) {
68*c8dee2aaSAndroid Build Coastguard Worker return false;
69*c8dee2aaSAndroid Build Coastguard Worker }
70*c8dee2aaSAndroid Build Coastguard Worker if (fLocale != other.fLocale) {
71*c8dee2aaSAndroid Build Coastguard Worker return false;
72*c8dee2aaSAndroid Build Coastguard Worker }
73*c8dee2aaSAndroid Build Coastguard Worker if (fHasForeground != other.fHasForeground || fForeground != other.fForeground) {
74*c8dee2aaSAndroid Build Coastguard Worker return false;
75*c8dee2aaSAndroid Build Coastguard Worker }
76*c8dee2aaSAndroid Build Coastguard Worker if (fHasBackground != other.fHasBackground || fBackground != other.fBackground) {
77*c8dee2aaSAndroid Build Coastguard Worker return false;
78*c8dee2aaSAndroid Build Coastguard Worker }
79*c8dee2aaSAndroid Build Coastguard Worker if (fTextShadows.size() != other.fTextShadows.size()) {
80*c8dee2aaSAndroid Build Coastguard Worker return false;
81*c8dee2aaSAndroid Build Coastguard Worker }
82*c8dee2aaSAndroid Build Coastguard Worker for (size_t i = 0; i < fTextShadows.size(); ++i) {
83*c8dee2aaSAndroid Build Coastguard Worker if (fTextShadows[i] != other.fTextShadows[i]) {
84*c8dee2aaSAndroid Build Coastguard Worker return false;
85*c8dee2aaSAndroid Build Coastguard Worker }
86*c8dee2aaSAndroid Build Coastguard Worker }
87*c8dee2aaSAndroid Build Coastguard Worker if (fFontFeatures.size() != other.fFontFeatures.size()) {
88*c8dee2aaSAndroid Build Coastguard Worker return false;
89*c8dee2aaSAndroid Build Coastguard Worker }
90*c8dee2aaSAndroid Build Coastguard Worker for (size_t i = 0; i < fFontFeatures.size(); ++i) {
91*c8dee2aaSAndroid Build Coastguard Worker if (!(fFontFeatures[i] == other.fFontFeatures[i])) {
92*c8dee2aaSAndroid Build Coastguard Worker return false;
93*c8dee2aaSAndroid Build Coastguard Worker }
94*c8dee2aaSAndroid Build Coastguard Worker }
95*c8dee2aaSAndroid Build Coastguard Worker if (fFontArguments != other.fFontArguments) {
96*c8dee2aaSAndroid Build Coastguard Worker return false;
97*c8dee2aaSAndroid Build Coastguard Worker }
98*c8dee2aaSAndroid Build Coastguard Worker
99*c8dee2aaSAndroid Build Coastguard Worker return true;
100*c8dee2aaSAndroid Build Coastguard Worker }
101*c8dee2aaSAndroid Build Coastguard Worker
equalsByFonts(const TextStyle & that) const102*c8dee2aaSAndroid Build Coastguard Worker bool TextStyle::equalsByFonts(const TextStyle& that) const {
103*c8dee2aaSAndroid Build Coastguard Worker
104*c8dee2aaSAndroid Build Coastguard Worker return !fIsPlaceholder && !that.fIsPlaceholder &&
105*c8dee2aaSAndroid Build Coastguard Worker fFontStyle == that.fFontStyle &&
106*c8dee2aaSAndroid Build Coastguard Worker fFontFamilies == that.fFontFamilies &&
107*c8dee2aaSAndroid Build Coastguard Worker fFontFeatures == that.fFontFeatures &&
108*c8dee2aaSAndroid Build Coastguard Worker fFontArguments == that.getFontArguments() &&
109*c8dee2aaSAndroid Build Coastguard Worker nearlyEqual(fLetterSpacing, that.fLetterSpacing) &&
110*c8dee2aaSAndroid Build Coastguard Worker nearlyEqual(fWordSpacing, that.fWordSpacing) &&
111*c8dee2aaSAndroid Build Coastguard Worker nearlyEqual(fHeight, that.fHeight) &&
112*c8dee2aaSAndroid Build Coastguard Worker nearlyEqual(fBaselineShift, that.fBaselineShift) &&
113*c8dee2aaSAndroid Build Coastguard Worker nearlyEqual(fFontSize, that.fFontSize) &&
114*c8dee2aaSAndroid Build Coastguard Worker fLocale == that.fLocale;
115*c8dee2aaSAndroid Build Coastguard Worker }
116*c8dee2aaSAndroid Build Coastguard Worker
matchOneAttribute(StyleType styleType,const TextStyle & other) const117*c8dee2aaSAndroid Build Coastguard Worker bool TextStyle::matchOneAttribute(StyleType styleType, const TextStyle& other) const {
118*c8dee2aaSAndroid Build Coastguard Worker switch (styleType) {
119*c8dee2aaSAndroid Build Coastguard Worker case kForeground:
120*c8dee2aaSAndroid Build Coastguard Worker return (!fHasForeground && !other.fHasForeground && fColor == other.fColor) ||
121*c8dee2aaSAndroid Build Coastguard Worker ( fHasForeground && other.fHasForeground && fForeground == other.fForeground);
122*c8dee2aaSAndroid Build Coastguard Worker
123*c8dee2aaSAndroid Build Coastguard Worker case kBackground:
124*c8dee2aaSAndroid Build Coastguard Worker return (!fHasBackground && !other.fHasBackground) ||
125*c8dee2aaSAndroid Build Coastguard Worker ( fHasBackground && other.fHasBackground && fBackground == other.fBackground);
126*c8dee2aaSAndroid Build Coastguard Worker
127*c8dee2aaSAndroid Build Coastguard Worker case kShadow:
128*c8dee2aaSAndroid Build Coastguard Worker if (fTextShadows.size() != other.fTextShadows.size()) {
129*c8dee2aaSAndroid Build Coastguard Worker return false;
130*c8dee2aaSAndroid Build Coastguard Worker }
131*c8dee2aaSAndroid Build Coastguard Worker
132*c8dee2aaSAndroid Build Coastguard Worker for (int32_t i = 0; i < SkToInt(fTextShadows.size()); ++i) {
133*c8dee2aaSAndroid Build Coastguard Worker if (fTextShadows[i] != other.fTextShadows[i]) {
134*c8dee2aaSAndroid Build Coastguard Worker return false;
135*c8dee2aaSAndroid Build Coastguard Worker }
136*c8dee2aaSAndroid Build Coastguard Worker }
137*c8dee2aaSAndroid Build Coastguard Worker return true;
138*c8dee2aaSAndroid Build Coastguard Worker
139*c8dee2aaSAndroid Build Coastguard Worker case kDecorations:
140*c8dee2aaSAndroid Build Coastguard Worker return this->fDecoration == other.fDecoration;
141*c8dee2aaSAndroid Build Coastguard Worker
142*c8dee2aaSAndroid Build Coastguard Worker case kLetterSpacing:
143*c8dee2aaSAndroid Build Coastguard Worker return fLetterSpacing == other.fLetterSpacing;
144*c8dee2aaSAndroid Build Coastguard Worker
145*c8dee2aaSAndroid Build Coastguard Worker case kWordSpacing:
146*c8dee2aaSAndroid Build Coastguard Worker return fWordSpacing == other.fWordSpacing;
147*c8dee2aaSAndroid Build Coastguard Worker
148*c8dee2aaSAndroid Build Coastguard Worker case kAllAttributes:
149*c8dee2aaSAndroid Build Coastguard Worker return this->equals(other);
150*c8dee2aaSAndroid Build Coastguard Worker
151*c8dee2aaSAndroid Build Coastguard Worker case kFont:
152*c8dee2aaSAndroid Build Coastguard Worker // TODO: should not we take typefaces in account?
153*c8dee2aaSAndroid Build Coastguard Worker return fFontStyle == other.fFontStyle &&
154*c8dee2aaSAndroid Build Coastguard Worker fLocale == other.fLocale &&
155*c8dee2aaSAndroid Build Coastguard Worker fFontFamilies == other.fFontFamilies &&
156*c8dee2aaSAndroid Build Coastguard Worker fFontSize == other.fFontSize &&
157*c8dee2aaSAndroid Build Coastguard Worker fHeight == other.fHeight &&
158*c8dee2aaSAndroid Build Coastguard Worker fHalfLeading == other.fHalfLeading &&
159*c8dee2aaSAndroid Build Coastguard Worker fBaselineShift == other.fBaselineShift &&
160*c8dee2aaSAndroid Build Coastguard Worker fFontArguments == other.fFontArguments;
161*c8dee2aaSAndroid Build Coastguard Worker default:
162*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(false);
163*c8dee2aaSAndroid Build Coastguard Worker return false;
164*c8dee2aaSAndroid Build Coastguard Worker }
165*c8dee2aaSAndroid Build Coastguard Worker }
166*c8dee2aaSAndroid Build Coastguard Worker
getFontMetrics(SkFontMetrics * metrics) const167*c8dee2aaSAndroid Build Coastguard Worker void TextStyle::getFontMetrics(SkFontMetrics* metrics) const {
168*c8dee2aaSAndroid Build Coastguard Worker SkFont font(fTypeface, fFontSize);
169*c8dee2aaSAndroid Build Coastguard Worker font.setEdging(SkFont::Edging::kAntiAlias);
170*c8dee2aaSAndroid Build Coastguard Worker font.setSubpixel(true);
171*c8dee2aaSAndroid Build Coastguard Worker font.setHinting(SkFontHinting::kSlight);
172*c8dee2aaSAndroid Build Coastguard Worker font.getMetrics(metrics);
173*c8dee2aaSAndroid Build Coastguard Worker if (fHeightOverride) {
174*c8dee2aaSAndroid Build Coastguard Worker auto multiplier = fHeight * fFontSize;
175*c8dee2aaSAndroid Build Coastguard Worker auto height = metrics->fDescent - metrics->fAscent + metrics->fLeading;
176*c8dee2aaSAndroid Build Coastguard Worker metrics->fAscent = (metrics->fAscent - metrics->fLeading / 2) * multiplier / height;
177*c8dee2aaSAndroid Build Coastguard Worker metrics->fDescent = (metrics->fDescent + metrics->fLeading / 2) * multiplier / height;
178*c8dee2aaSAndroid Build Coastguard Worker
179*c8dee2aaSAndroid Build Coastguard Worker } else {
180*c8dee2aaSAndroid Build Coastguard Worker metrics->fAscent = (metrics->fAscent - metrics->fLeading / 2);
181*c8dee2aaSAndroid Build Coastguard Worker metrics->fDescent = (metrics->fDescent + metrics->fLeading / 2);
182*c8dee2aaSAndroid Build Coastguard Worker }
183*c8dee2aaSAndroid Build Coastguard Worker // If we shift the baseline we need to make sure the shifted text fits the line
184*c8dee2aaSAndroid Build Coastguard Worker metrics->fAscent += fBaselineShift;
185*c8dee2aaSAndroid Build Coastguard Worker metrics->fDescent += fBaselineShift;
186*c8dee2aaSAndroid Build Coastguard Worker }
187*c8dee2aaSAndroid Build Coastguard Worker
setFontArguments(const std::optional<SkFontArguments> & args)188*c8dee2aaSAndroid Build Coastguard Worker void TextStyle::setFontArguments(const std::optional<SkFontArguments>& args) {
189*c8dee2aaSAndroid Build Coastguard Worker if (!args) {
190*c8dee2aaSAndroid Build Coastguard Worker fFontArguments.reset();
191*c8dee2aaSAndroid Build Coastguard Worker return;
192*c8dee2aaSAndroid Build Coastguard Worker }
193*c8dee2aaSAndroid Build Coastguard Worker
194*c8dee2aaSAndroid Build Coastguard Worker fFontArguments.emplace(*args);
195*c8dee2aaSAndroid Build Coastguard Worker }
196*c8dee2aaSAndroid Build Coastguard Worker
equals(const PlaceholderStyle & other) const197*c8dee2aaSAndroid Build Coastguard Worker bool PlaceholderStyle::equals(const PlaceholderStyle& other) const {
198*c8dee2aaSAndroid Build Coastguard Worker return nearlyEqual(fWidth, other.fWidth) &&
199*c8dee2aaSAndroid Build Coastguard Worker nearlyEqual(fHeight, other.fHeight) &&
200*c8dee2aaSAndroid Build Coastguard Worker fAlignment == other.fAlignment &&
201*c8dee2aaSAndroid Build Coastguard Worker fBaseline == other.fBaseline &&
202*c8dee2aaSAndroid Build Coastguard Worker (fAlignment != PlaceholderAlignment::kBaseline ||
203*c8dee2aaSAndroid Build Coastguard Worker nearlyEqual(fBaselineOffset, other.fBaselineOffset));
204*c8dee2aaSAndroid Build Coastguard Worker }
205*c8dee2aaSAndroid Build Coastguard Worker
206*c8dee2aaSAndroid Build Coastguard Worker } // namespace textlayout
207*c8dee2aaSAndroid Build Coastguard Worker } // namespace skia
208