1 // Copyright 2019 Google LLC.
2 #include "modules/skparagraph/src/ParagraphImpl.h"
3 #include "modules/skparagraph/src/TextWrapper.h"
4
5 namespace skia {
6 namespace textlayout {
7
8 namespace {
9 struct LineBreakerWithLittleRounding {
LineBreakerWithLittleRoundingskia::textlayout::__anon9f9d95f20111::LineBreakerWithLittleRounding10 LineBreakerWithLittleRounding(SkScalar maxWidth)
11 : fLower(maxWidth - 0.25f)
12 , fMaxWidth(maxWidth)
13 , fUpper(maxWidth + 0.25f) {}
14
breakLineskia::textlayout::__anon9f9d95f20111::LineBreakerWithLittleRounding15 bool breakLine(SkScalar width) const {
16 if (width < fLower) {
17 return false;
18 } else if (width > fUpper) {
19 return true;
20 }
21
22 auto val = std::fabs(width);
23 SkScalar roundedWidth;
24 if (val < 10000) {
25 roundedWidth = SkScalarRoundToScalar(width * 100) * (1.0f/100);
26 } else if (val < 100000) {
27 roundedWidth = SkScalarRoundToScalar(width * 10) * (1.0f/10);
28 } else {
29 roundedWidth = SkScalarFloorToScalar(width);
30 }
31 return roundedWidth > fMaxWidth;
32 }
33
34 const SkScalar fLower, fMaxWidth, fUpper;
35 };
36 } // namespace
37
38 // Since we allow cluster clipping when they don't fit
39 // we have to work with stretches - parts of clusters
lookAhead(SkScalar maxWidth,Cluster * endOfClusters)40 void TextWrapper::lookAhead(SkScalar maxWidth, Cluster* endOfClusters) {
41
42 reset();
43 fEndLine.metrics().clean();
44 fWords.startFrom(fEndLine.startCluster(), fEndLine.startPos());
45 fClusters.startFrom(fEndLine.startCluster(), fEndLine.startPos());
46 fClip.startFrom(fEndLine.startCluster(), fEndLine.startPos());
47
48 LineBreakerWithLittleRounding breaker(maxWidth);
49 Cluster* nextNonBreakingSpace = nullptr;
50 for (auto cluster = fEndLine.endCluster(); cluster < endOfClusters; ++cluster) {
51 if (cluster->isHardBreak()) {
52 } else if (
53 // TODO: Trying to deal with flutter rounding problem. Must be removed...
54 SkScalar width = fWords.width() + fClusters.width() + cluster->width();
55 breaker.breakLine(width)) {
56 if (cluster->isWhitespaceBreak()) {
57 // It's the end of the word
58 fClusters.extend(cluster);
59 fMinIntrinsicWidth = std::max(fMinIntrinsicWidth, getClustersTrimmedWidth());
60 fWords.extend(fClusters);
61 continue;
62 } else if (cluster->run().isPlaceholder()) {
63 if (!fClusters.empty()) {
64 // Placeholder ends the previous word
65 fMinIntrinsicWidth = std::max(fMinIntrinsicWidth, getClustersTrimmedWidth());
66 fWords.extend(fClusters);
67 }
68
69 if (cluster->width() > maxWidth && fWords.empty()) {
70 // Placeholder is the only text and it's longer than the line;
71 // it does not count in fMinIntrinsicWidth
72 fClusters.extend(cluster);
73 fTooLongCluster = true;
74 fTooLongWord = true;
75 } else {
76 // Placeholder does not fit the line; it will be considered again on the next line
77 }
78 break;
79 }
80
81 // Walk further to see if there is a too long word, cluster or glyph
82 SkScalar nextWordLength = fClusters.width();
83 SkScalar nextShortWordLength = nextWordLength;
84 for (auto further = cluster; further != endOfClusters; ++further) {
85 if (further->isSoftBreak() || further->isHardBreak() || further->isWhitespaceBreak()) {
86 break;
87 }
88 if (further->run().isPlaceholder()) {
89 // Placeholder ends the word
90 break;
91 }
92
93 if (nextWordLength > 0 && nextWordLength <= maxWidth && further->isIntraWordBreak()) {
94 // The cluster is spaces but not the end of the word in a normal sense
95 nextNonBreakingSpace = further;
96 nextShortWordLength = nextWordLength;
97 }
98
99 if (maxWidth == 0) {
100 // This is a tricky flutter case: layout(width:0) places 1 cluster on each line
101 nextWordLength = std::max(nextWordLength, further->width());
102 } else {
103 nextWordLength += further->width();
104 }
105 }
106 if (nextWordLength > maxWidth) {
107 if (nextNonBreakingSpace != nullptr) {
108 // We only get here if the non-breaking space improves our situation
109 // (allows us to break the text to fit the word)
110 if (SkScalar shortLength = fWords.width() + nextShortWordLength;
111 !breaker.breakLine(shortLength)) {
112 // We can add the short word to the existing line
113 fClusters = TextStretch(fClusters.startCluster(), nextNonBreakingSpace, fClusters.metrics().getForceStrut());
114 fMinIntrinsicWidth = std::max(fMinIntrinsicWidth, nextShortWordLength);
115 fWords.extend(fClusters);
116 } else {
117 // We can place the short word on the next line
118 fClusters.clean();
119 }
120 // Either way we are not in "word is too long" situation anymore
121 break;
122 }
123 // If the word is too long we can break it right now and hope it's enough
124 fMinIntrinsicWidth = std::max(fMinIntrinsicWidth, nextWordLength);
125 if (fClusters.endPos() - fClusters.startPos() > 1 ||
126 fWords.empty()) {
127 fTooLongWord = true;
128 } else {
129 // Even if the word is too long there is a very little space on this line.
130 // let's deal with it on the next line.
131 }
132 }
133
134 if (cluster->width() > maxWidth) {
135 fClusters.extend(cluster);
136 fTooLongCluster = true;
137 fTooLongWord = true;
138 }
139 break;
140 }
141
142 if (cluster->run().isPlaceholder()) {
143 if (!fClusters.empty()) {
144 // Placeholder ends the previous word (placeholders are ignored in trimming)
145 fMinIntrinsicWidth = std::max(fMinIntrinsicWidth, getClustersTrimmedWidth());
146 fWords.extend(fClusters);
147 }
148
149 // Placeholder is separate word and its width now is counted in minIntrinsicWidth
150 fMinIntrinsicWidth = std::max(fMinIntrinsicWidth, cluster->width());
151 fWords.extend(cluster);
152 } else {
153 fClusters.extend(cluster);
154
155 // Keep adding clusters/words
156 if (fClusters.endOfWord()) {
157 fMinIntrinsicWidth = std::max(fMinIntrinsicWidth, getClustersTrimmedWidth());
158 fWords.extend(fClusters);
159 }
160 }
161
162 if ((fHardLineBreak = cluster->isHardBreak())) {
163 // Stop at the hard line break
164 break;
165 }
166 }
167 }
168
moveForward(bool hasEllipsis)169 void TextWrapper::moveForward(bool hasEllipsis) {
170
171 // We normally break lines by words.
172 // The only way we may go to clusters is if the word is too long or
173 // it's the first word and it has an ellipsis attached to it.
174 // If nothing fits we show the clipping.
175 if (!fWords.empty()) {
176 fEndLine.extend(fWords);
177 if (!fTooLongWord || hasEllipsis) {
178 return;
179 }
180 }
181 if (!fClusters.empty()) {
182 fEndLine.extend(fClusters);
183 if (!fTooLongCluster) {
184 return;
185 }
186 }
187
188 if (!fClip.empty()) {
189 // Flutter: forget the clipped cluster but keep the metrics
190 fEndLine.metrics().add(fClip.metrics());
191 }
192 }
193
194 // Special case for start/end cluster since they can be clipped
trimEndSpaces(TextAlign align)195 void TextWrapper::trimEndSpaces(TextAlign align) {
196 // Remember the breaking position
197 fEndLine.saveBreak();
198 // Skip all space cluster at the end
199 for (auto cluster = fEndLine.endCluster();
200 cluster >= fEndLine.startCluster() && cluster->isWhitespaceBreak();
201 --cluster) {
202 fEndLine.trim(cluster);
203 }
204 fEndLine.trim();
205 }
206
getClustersTrimmedWidth()207 SkScalar TextWrapper::getClustersTrimmedWidth() {
208 // Move the end of the line to the left
209 SkScalar width = 0;
210 bool trailingSpaces = true;
211 for (auto cluster = fClusters.endCluster(); cluster >= fClusters.startCluster(); --cluster) {
212 if (cluster->run().isPlaceholder()) {
213 continue;
214 }
215 if (trailingSpaces) {
216 if (!cluster->isWhitespaceBreak()) {
217 width += cluster->trimmedWidth(cluster->endPos());
218 trailingSpaces = false;
219 }
220 continue;
221 }
222 width += cluster->width();
223 }
224 return width;
225 }
226
227 // Trim the beginning spaces in case of soft line break
trimStartSpaces(Cluster * endOfClusters)228 std::tuple<Cluster*, size_t, SkScalar> TextWrapper::trimStartSpaces(Cluster* endOfClusters) {
229
230 if (fHardLineBreak) {
231 // End of line is always end of cluster, but need to skip \n
232 auto width = fEndLine.width();
233 auto cluster = fEndLine.endCluster() + 1;
234 while (cluster < fEndLine.breakCluster() && cluster->isWhitespaceBreak()) {
235 width += cluster->width();
236 ++cluster;
237 }
238 return std::make_tuple(fEndLine.breakCluster() + 1, 0, width);
239 }
240
241 // breakCluster points to the end of the line;
242 // It's a soft line break so we need to move lineStart forward skipping all the spaces
243 auto width = fEndLine.widthWithGhostSpaces();
244 auto cluster = fEndLine.breakCluster() + 1;
245 while (cluster < endOfClusters && cluster->isWhitespaceBreak()) {
246 width += cluster->width();
247 ++cluster;
248 }
249
250 if (fEndLine.breakCluster()->isWhitespaceBreak() && fEndLine.breakCluster() < endOfClusters) {
251 // In case of a soft line break by the whitespace
252 // fBreak should point to the beginning of the next line
253 // (it only matters when there are trailing spaces)
254 fEndLine.shiftBreak();
255 }
256
257 return std::make_tuple(cluster, 0, width);
258 }
259
260 // TODO: refactor the code for line ending (with/without ellipsis)
breakTextIntoLines(ParagraphImpl * parent,SkScalar maxWidth,const AddLineToParagraph & addLine)261 void TextWrapper::breakTextIntoLines(ParagraphImpl* parent,
262 SkScalar maxWidth,
263 const AddLineToParagraph& addLine) {
264 fHeight = 0;
265 fMinIntrinsicWidth = std::numeric_limits<SkScalar>::min();
266 fMaxIntrinsicWidth = std::numeric_limits<SkScalar>::min();
267
268 auto span = parent->clusters();
269 if (span.size() == 0) {
270 return;
271 }
272 auto maxLines = parent->paragraphStyle().getMaxLines();
273 auto align = parent->paragraphStyle().effective_align();
274 auto unlimitedLines = maxLines == std::numeric_limits<size_t>::max();
275 auto endlessLine = !SkScalarIsFinite(maxWidth);
276 auto hasEllipsis = parent->paragraphStyle().ellipsized();
277
278 auto disableFirstAscent = parent->paragraphStyle().getTextHeightBehavior() & TextHeightBehavior::kDisableFirstAscent;
279 auto disableLastDescent = parent->paragraphStyle().getTextHeightBehavior() & TextHeightBehavior::kDisableLastDescent;
280 bool firstLine = true; // We only interested in fist line if we have to disable the first ascent
281
282 SkScalar softLineMaxIntrinsicWidth = 0;
283 fEndLine = TextStretch(span.begin(), span.begin(), parent->strutForceHeight());
284 auto end = span.end() - 1;
285 auto start = span.begin();
286 InternalLineMetrics maxRunMetrics;
287 bool needEllipsis = false;
288 while (fEndLine.endCluster() != end) {
289
290 lookAhead(maxWidth, end);
291
292 auto lastLine = (hasEllipsis && unlimitedLines) || fLineNumber >= maxLines;
293 needEllipsis = hasEllipsis && !endlessLine && lastLine;
294
295 moveForward(needEllipsis);
296 needEllipsis &= fEndLine.endCluster() < end - 1; // Only if we have some text to ellipsize
297
298 // Do not trim end spaces on the naturally last line of the left aligned text
299 trimEndSpaces(align);
300
301 // For soft line breaks add to the line all the spaces next to it
302 Cluster* startLine;
303 size_t pos;
304 SkScalar widthWithSpaces;
305 std::tie(startLine, pos, widthWithSpaces) = trimStartSpaces(end);
306
307 if (needEllipsis && !fHardLineBreak) {
308 // This is what we need to do to preserve a space before the ellipsis
309 fEndLine.restoreBreak();
310 widthWithSpaces = fEndLine.widthWithGhostSpaces();
311 }
312
313 // If the line is empty with the hard line break, let's take the paragraph font (flutter???)
314 if (fHardLineBreak && fEndLine.width() == 0) {
315 fEndLine.setMetrics(parent->getEmptyMetrics());
316 }
317
318 // Deal with placeholder clusters == runs[@size==1]
319 Run* lastRun = nullptr;
320 for (auto cluster = fEndLine.startCluster(); cluster <= fEndLine.endCluster(); ++cluster) {
321 auto r = cluster->runOrNull();
322 if (r == lastRun) {
323 continue;
324 }
325 lastRun = r;
326 if (lastRun->placeholderStyle() != nullptr) {
327 SkASSERT(lastRun->size() == 1);
328 // Update the placeholder metrics so we can get the placeholder positions later
329 // and the line metrics (to make sure the placeholder fits)
330 lastRun->updateMetrics(&fEndLine.metrics());
331 }
332 }
333
334 // Before we update the line metrics with struts,
335 // let's save it for GetRectsForRange(RectHeightStyle::kMax)
336 maxRunMetrics = fEndLine.metrics();
337 maxRunMetrics.fForceStrut = false;
338
339 if (parent->strutEnabled()) {
340 // Make sure font metrics are not less than the strut
341 parent->strutMetrics().updateLineMetrics(fEndLine.metrics());
342 }
343
344 // TODO: keep start/end/break info for text and runs but in a better way that below
345 TextRange textExcludingSpaces(fEndLine.startCluster()->textRange().start, fEndLine.endCluster()->textRange().end);
346 TextRange text(fEndLine.startCluster()->textRange().start, fEndLine.breakCluster()->textRange().start);
347 TextRange textIncludingNewlines(fEndLine.startCluster()->textRange().start, startLine->textRange().start);
348 if (startLine == end) {
349 textIncludingNewlines.end = parent->text().size();
350 text.end = parent->text().size();
351 }
352 ClusterRange clusters(fEndLine.startCluster() - start, fEndLine.endCluster() - start + 1);
353 ClusterRange clustersWithGhosts(fEndLine.startCluster() - start, startLine - start);
354
355 if (disableFirstAscent && firstLine) {
356 fEndLine.metrics().fAscent = fEndLine.metrics().fRawAscent;
357 }
358 if (disableLastDescent && (lastLine || (startLine == end && !fHardLineBreak ))) {
359 fEndLine.metrics().fDescent = fEndLine.metrics().fRawDescent;
360 }
361
362 SkScalar lineHeight = fEndLine.metrics().height();
363 firstLine = false;
364
365 if (fEndLine.empty()) {
366 // Correct text and clusters (make it empty for an empty line)
367 textExcludingSpaces.end = textExcludingSpaces.start;
368 clusters.end = clusters.start;
369 }
370
371 // In case of a force wrapping we don't have a break cluster and have to use the end cluster
372 text.end = std::max(text.end, textExcludingSpaces.end);
373
374 addLine(textExcludingSpaces,
375 text,
376 textIncludingNewlines, clusters, clustersWithGhosts, widthWithSpaces,
377 fEndLine.startPos(),
378 fEndLine.endPos(),
379 SkVector::Make(0, fHeight),
380 SkVector::Make(fEndLine.width(), lineHeight),
381 fEndLine.metrics(),
382 needEllipsis && !fHardLineBreak);
383
384 softLineMaxIntrinsicWidth += widthWithSpaces;
385
386 fMaxIntrinsicWidth = std::max(fMaxIntrinsicWidth, softLineMaxIntrinsicWidth);
387 if (fHardLineBreak) {
388 softLineMaxIntrinsicWidth = 0;
389 }
390 // Start a new line
391 fHeight += lineHeight;
392 if (!fHardLineBreak || startLine != end) {
393 fEndLine.clean();
394 }
395 fEndLine.startFrom(startLine, pos);
396 parent->fMaxWidthWithTrailingSpaces = std::max(parent->fMaxWidthWithTrailingSpaces, widthWithSpaces);
397
398 if (hasEllipsis && unlimitedLines) {
399 // There is one case when we need an ellipsis on a separate line
400 // after a line break when width is infinite
401 if (!fHardLineBreak) {
402 break;
403 }
404 } else if (lastLine) {
405 // There is nothing more to draw
406 fHardLineBreak = false;
407 break;
408 }
409
410 ++fLineNumber;
411 }
412
413 // We finished formatting the text but we need to scan the rest for some numbers
414 // TODO: make it a case of a normal flow
415 if (fEndLine.endCluster() != nullptr) {
416 auto lastWordLength = 0.0f;
417 auto cluster = fEndLine.endCluster();
418 while (cluster != end || cluster->endPos() < end->endPos()) {
419 fExceededMaxLines = true;
420 if (cluster->isHardBreak()) {
421 // Hard line break ends the word and the line
422 fMaxIntrinsicWidth = std::max(fMaxIntrinsicWidth, softLineMaxIntrinsicWidth);
423 softLineMaxIntrinsicWidth = 0;
424 fMinIntrinsicWidth = std::max(fMinIntrinsicWidth, lastWordLength);
425 lastWordLength = 0;
426 } else if (cluster->isWhitespaceBreak()) {
427 // Whitespaces end the word
428 softLineMaxIntrinsicWidth += cluster->width();
429 fMinIntrinsicWidth = std::max(fMinIntrinsicWidth, lastWordLength);
430 lastWordLength = 0;
431 } else if (cluster->run().isPlaceholder()) {
432 // Placeholder ends the previous word and creates a separate one
433 fMinIntrinsicWidth = std::max(fMinIntrinsicWidth, lastWordLength);
434 // Placeholder width now counts in fMinIntrinsicWidth
435 softLineMaxIntrinsicWidth += cluster->width();
436 fMinIntrinsicWidth = std::max(fMinIntrinsicWidth, cluster->width());
437 lastWordLength = 0;
438 } else {
439 // Nothing out of ordinary - just add this cluster to the word and to the line
440 softLineMaxIntrinsicWidth += cluster->width();
441 lastWordLength += cluster->width();
442 }
443 ++cluster;
444 }
445 fMinIntrinsicWidth = std::max(fMinIntrinsicWidth, lastWordLength);
446 fMaxIntrinsicWidth = std::max(fMaxIntrinsicWidth, softLineMaxIntrinsicWidth);
447
448 if (parent->lines().empty()) {
449 // In case we could not place even a single cluster on the line
450 if (disableFirstAscent) {
451 fEndLine.metrics().fAscent = fEndLine.metrics().fRawAscent;
452 }
453 if (disableLastDescent && !fHardLineBreak) {
454 fEndLine.metrics().fDescent = fEndLine.metrics().fRawDescent;
455 }
456 fHeight = std::max(fHeight, fEndLine.metrics().height());
457 }
458 }
459
460 if (fHardLineBreak) {
461 // Last character is a line break
462 if (parent->strutEnabled()) {
463 // Make sure font metrics are not less than the strut
464 parent->strutMetrics().updateLineMetrics(fEndLine.metrics());
465 }
466
467 if (disableLastDescent) {
468 fEndLine.metrics().fDescent = fEndLine.metrics().fRawDescent;
469 }
470
471 ClusterRange clusters(fEndLine.breakCluster() - start, fEndLine.endCluster() - start);
472 addLine(fEndLine.breakCluster()->textRange(),
473 fEndLine.breakCluster()->textRange(),
474 fEndLine.endCluster()->textRange(),
475 clusters,
476 clusters,
477 0,
478 0,
479 0,
480 SkVector::Make(0, fHeight),
481 SkVector::Make(0, fEndLine.metrics().height()),
482 fEndLine.metrics(),
483 needEllipsis);
484 fHeight += fEndLine.metrics().height();
485 parent->lines().back().setMaxRunMetrics(maxRunMetrics);
486 }
487 }
488
489 } // namespace textlayout
490 } // namespace skia
491