| // Copyright 2019 Google LLC. |
| |
| #include "modules/skparagraph/src/TextLine.h" |
| |
| #include "include/core/SkBlurTypes.h" |
| #include "include/core/SkFont.h" |
| #include "include/core/SkFontMetrics.h" |
| #include "include/core/SkMaskFilter.h" |
| #include "include/core/SkPaint.h" |
| #include "include/core/SkSpan.h" |
| #include "include/core/SkString.h" |
| #include "include/core/SkTextBlob.h" |
| #include "include/core/SkTypes.h" |
| #include "include/private/base/SkTemplates.h" |
| #include "include/private/base/SkTo.h" |
| #include "modules/skparagraph/include/DartTypes.h" |
| #include "modules/skparagraph/include/Metrics.h" |
| #include "modules/skparagraph/include/ParagraphPainter.h" |
| #include "modules/skparagraph/include/ParagraphStyle.h" |
| #include "modules/skparagraph/include/TextShadow.h" |
| #include "modules/skparagraph/include/TextStyle.h" |
| #include "modules/skparagraph/src/Decorations.h" |
| #include "modules/skparagraph/src/ParagraphImpl.h" |
| #include "modules/skparagraph/src/ParagraphPainterImpl.h" |
| #include "modules/skshaper/include/SkShaper.h" |
| #include "modules/skshaper/include/SkShaper_harfbuzz.h" |
| #include "modules/skshaper/include/SkShaper_skunicode.h" |
| |
| #include <algorithm> |
| #include <iterator> |
| #include <limits> |
| #include <map> |
| #include <memory> |
| #include <tuple> |
| #include <type_traits> |
| #include <utility> |
| |
| using namespace skia_private; |
| |
| namespace skia { |
| namespace textlayout { |
| |
| namespace { |
| |
| // TODO: deal with all the intersection functionality |
| TextRange intersected(const TextRange& a, const TextRange& b) { |
| if (a.start == b.start && a.end == b.end) return a; |
| auto begin = std::max(a.start, b.start); |
| auto end = std::min(a.end, b.end); |
| return end >= begin ? TextRange(begin, end) : EMPTY_TEXT; |
| } |
| |
| SkScalar littleRound(SkScalar a) { |
| // This rounding is done to match Flutter tests. Must be removed.. |
| return SkScalarRoundToScalar(a * 100.0)/100.0; |
| } |
| |
| TextRange operator*(const TextRange& a, const TextRange& b) { |
| if (a.start == b.start && a.end == b.end) return a; |
| auto begin = std::max(a.start, b.start); |
| auto end = std::min(a.end, b.end); |
| return end > begin ? TextRange(begin, end) : EMPTY_TEXT; |
| } |
| |
| int compareRound(SkScalar a, SkScalar b, bool applyRoundingHack) { |
| // There is a rounding error that gets bigger when maxWidth gets bigger |
| // VERY long zalgo text (> 100000) on a VERY long line (> 10000) |
| // Canvas scaling affects it |
| // Letter spacing affects it |
| // It has to be relative to be useful |
| auto base = std::max(SkScalarAbs(a), SkScalarAbs(b)); |
| auto diff = SkScalarAbs(a - b); |
| if (nearlyZero(base) || diff / base < 0.001f) { |
| return 0; |
| } |
| |
| auto ra = a; |
| auto rb = b; |
| |
| if (applyRoundingHack) { |
| ra = littleRound(a); |
| rb = littleRound(b); |
| } |
| if (ra < rb) { |
| return -1; |
| } else { |
| return 1; |
| } |
| } |
| |
| } // namespace |
| |
| TextLine::TextLine(ParagraphImpl* owner, |
| SkVector offset, |
| SkVector advance, |
| BlockRange blocks, |
| TextRange textExcludingSpaces, |
| TextRange text, |
| TextRange textIncludingNewlines, |
| ClusterRange clusters, |
| ClusterRange clustersWithGhosts, |
| SkScalar widthWithSpaces, |
| InternalLineMetrics sizes) |
| : fOwner(owner) |
| , fBlockRange(blocks) |
| , fTextExcludingSpaces(textExcludingSpaces) |
| , fText(text) |
| , fTextIncludingNewlines(textIncludingNewlines) |
| , fClusterRange(clusters) |
| , fGhostClusterRange(clustersWithGhosts) |
| , fRunsInVisualOrder() |
| , fAdvance(advance) |
| , fOffset(offset) |
| , fShift(0.0) |
| , fWidthWithSpaces(widthWithSpaces) |
| , fEllipsis(nullptr) |
| , fSizes(sizes) |
| , fHasBackground(false) |
| , fHasShadows(false) |
| , fHasDecorations(false) |
| , fAscentStyle(LineMetricStyle::CSS) |
| , fDescentStyle(LineMetricStyle::CSS) |
| , fTextBlobCachePopulated(false) { |
| // Reorder visual runs |
| auto& start = owner->cluster(fGhostClusterRange.start); |
| auto& end = owner->cluster(fGhostClusterRange.end - 1); |
| size_t numRuns = end.runIndex() - start.runIndex() + 1; |
| |
| for (BlockIndex index = fBlockRange.start; index < fBlockRange.end; ++index) { |
| auto b = fOwner->styles().begin() + index; |
| if (b->fStyle.hasBackground()) { |
| fHasBackground = true; |
| } |
| if (b->fStyle.getDecorationType() != TextDecoration::kNoDecoration) { |
| fHasDecorations = true; |
| } |
| if (b->fStyle.getShadowNumber() > 0) { |
| fHasShadows = true; |
| } |
| } |
| |
| // Get the logical order |
| |
| // This is just chosen to catch the common/fast cases. Feel free to tweak. |
| constexpr int kPreallocCount = 4; |
| AutoSTArray<kPreallocCount, SkUnicode::BidiLevel> runLevels(numRuns); |
| std::vector<RunIndex> placeholdersInOriginalOrder; |
| size_t runLevelsIndex = 0; |
| // Placeholders must be laid out using the original order in which they were added |
| // in the input. The API does not provide a way to indicate that a placeholder |
| // position was moved due to bidi reordering. |
| for (auto runIndex = start.runIndex(); runIndex <= end.runIndex(); ++runIndex) { |
| auto& run = fOwner->run(runIndex); |
| runLevels[runLevelsIndex++] = run.fBidiLevel; |
| fMaxRunMetrics.add( |
| InternalLineMetrics(run.correctAscent(), run.correctDescent(), run.fFontMetrics.fLeading)); |
| if (run.isPlaceholder()) { |
| placeholdersInOriginalOrder.push_back(runIndex); |
| } |
| } |
| SkASSERT(runLevelsIndex == numRuns); |
| |
| AutoSTArray<kPreallocCount, int32_t> logicalOrder(numRuns); |
| |
| // TODO: hide all these logic in SkUnicode? |
| fOwner->getUnicode()->reorderVisual(runLevels.data(), numRuns, logicalOrder.data()); |
| auto firstRunIndex = start.runIndex(); |
| auto placeholderIter = placeholdersInOriginalOrder.begin(); |
| for (auto index : logicalOrder) { |
| auto runIndex = firstRunIndex + index; |
| if (fOwner->run(runIndex).isPlaceholder()) { |
| fRunsInVisualOrder.push_back(*placeholderIter++); |
| } else { |
| fRunsInVisualOrder.push_back(runIndex); |
| } |
| } |
| |
| // TODO: This is the fix for flutter. Must be removed... |
| for (auto cluster = &start; cluster <= &end; ++cluster) { |
| if (!cluster->run().isPlaceholder()) { |
| fShift += cluster->getHalfLetterSpacing(); |
| break; |
| } |
| } |
| } |
| |
| void TextLine::paint(ParagraphPainter* painter, SkScalar x, SkScalar y) { |
| if (fHasBackground) { |
| this->iterateThroughVisualRuns(false, |
| [painter, x, y, this] |
| (const Run* run, SkScalar runOffsetInLine, TextRange textRange, SkScalar* runWidthInLine) { |
| *runWidthInLine = this->iterateThroughSingleRunByStyles( |
| TextAdjustment::GlyphCluster, run, runOffsetInLine, textRange, StyleType::kBackground, |
| [painter, x, y, this](TextRange textRange, const TextStyle& style, const ClipContext& context) { |
| this->paintBackground(painter, x, y, textRange, style, context); |
| }); |
| return true; |
| }); |
| } |
| |
| if (fHasShadows) { |
| this->iterateThroughVisualRuns(false, |
| [painter, x, y, this] |
| (const Run* run, SkScalar runOffsetInLine, TextRange textRange, SkScalar* runWidthInLine) { |
| *runWidthInLine = this->iterateThroughSingleRunByStyles( |
| TextAdjustment::GlyphCluster, run, runOffsetInLine, textRange, StyleType::kShadow, |
| [painter, x, y, this] |
| (TextRange textRange, const TextStyle& style, const ClipContext& context) { |
| this->paintShadow(painter, x, y, textRange, style, context); |
| }); |
| return true; |
| }); |
| } |
| |
| this->ensureTextBlobCachePopulated(); |
| |
| for (auto& record : fTextBlobCache) { |
| record.paint(painter, x, y); |
| } |
| |
| if (fHasDecorations) { |
| this->iterateThroughVisualRuns(false, |
| [painter, x, y, this] |
| (const Run* run, SkScalar runOffsetInLine, TextRange textRange, SkScalar* runWidthInLine) { |
| *runWidthInLine = this->iterateThroughSingleRunByStyles( |
| TextAdjustment::GlyphCluster, run, runOffsetInLine, textRange, StyleType::kDecorations, |
| [painter, x, y, this] |
| (TextRange textRange, const TextStyle& style, const ClipContext& context) { |
| this->paintDecorations(painter, x, y, textRange, style, context); |
| }); |
| return true; |
| }); |
| } |
| } |
| |
| void TextLine::ensureTextBlobCachePopulated() { |
| if (fTextBlobCachePopulated) { |
| return; |
| } |
| if (fBlockRange.width() == 1 && |
| fRunsInVisualOrder.size() == 1 && |
| fEllipsis == nullptr && |
| fOwner->run(fRunsInVisualOrder[0]).placeholderStyle() == nullptr) { |
| if (fClusterRange.width() == 0) { |
| return; |
| } |
| // Most common and most simple case |
| const auto& style = fOwner->block(fBlockRange.start).fStyle; |
| const auto& run = fOwner->run(fRunsInVisualOrder[0]); |
| auto clip = SkRect::MakeXYWH(0.0f, this->sizes().runTop(&run, this->fAscentStyle), |
| fAdvance.fX, |
| run.calculateHeight(this->fAscentStyle, this->fDescentStyle)); |
| |
| auto& start = fOwner->cluster(fClusterRange.start); |
| auto& end = fOwner->cluster(fClusterRange.end - 1); |
| SkASSERT(start.runIndex() == end.runIndex()); |
| GlyphRange glyphs; |
| if (run.leftToRight()) { |
| glyphs = GlyphRange(start.startPos(), |
| end.isHardBreak() ? end.startPos() : end.endPos()); |
| } else { |
| glyphs = GlyphRange(end.startPos(), |
| start.isHardBreak() ? start.startPos() : start.endPos()); |
| } |
| ClipContext context = {/*run=*/&run, |
| /*pos=*/glyphs.start, |
| /*size=*/glyphs.width(), |
| /*fTextShift=*/-run.positionX(glyphs.start), // starting position |
| /*clip=*/clip, // entire line |
| /*fExcludedTrailingSpaces=*/0.0f, // no need for that |
| /*clippingNeeded=*/false}; // no need for that |
| this->buildTextBlob(fTextExcludingSpaces, style, context); |
| } else { |
| this->iterateThroughVisualRuns(false, |
| [this](const Run* run, |
| SkScalar runOffsetInLine, |
| TextRange textRange, |
| SkScalar* runWidthInLine) { |
| if (run->placeholderStyle() != nullptr) { |
| *runWidthInLine = run->advance().fX; |
| return true; |
| } |
| *runWidthInLine = this->iterateThroughSingleRunByStyles( |
| TextAdjustment::GlyphCluster, |
| run, |
| runOffsetInLine, |
| textRange, |
| StyleType::kForeground, |
| [this](TextRange textRange, const TextStyle& style, const ClipContext& context) { |
| this->buildTextBlob(textRange, style, context); |
| }); |
| return true; |
| }); |
| } |
| fTextBlobCachePopulated = true; |
| } |
| |
| void TextLine::format(TextAlign align, SkScalar maxWidth) { |
| SkScalar delta = maxWidth - this->width(); |
| if (delta <= 0) { |
| return; |
| } |
| |
| // We do nothing for left align |
| if (align == TextAlign::kJustify) { |
| if (!this->endsWithHardLineBreak()) { |
| this->justify(maxWidth); |
| } else if (fOwner->paragraphStyle().getTextDirection() == TextDirection::kRtl) { |
| // Justify -> Right align |
| fShift = delta; |
| } |
| } else if (align == TextAlign::kRight) { |
| fShift = delta; |
| } else if (align == TextAlign::kCenter) { |
| fShift = delta / 2; |
| } |
| } |
| |
| void TextLine::scanStyles(StyleType styleType, const RunStyleVisitor& visitor) { |
| if (this->empty()) { |
| return; |
| } |
| |
| this->iterateThroughVisualRuns( |
| false, |
| [this, visitor, styleType]( |
| const Run* run, SkScalar runOffset, TextRange textRange, SkScalar* width) { |
| *width = this->iterateThroughSingleRunByStyles( |
| TextAdjustment::GlyphCluster, |
| run, |
| runOffset, |
| textRange, |
| styleType, |
| [visitor](TextRange textRange, |
| const TextStyle& style, |
| const ClipContext& context) { |
| visitor(textRange, style, context); |
| }); |
| return true; |
| }); |
| } |
| |
| SkRect TextLine::extendHeight(const ClipContext& context) const { |
| SkRect result = context.clip; |
| result.fBottom += std::max(this->fMaxRunMetrics.height() - this->height(), 0.0f); |
| return result; |
| } |
| |
| void TextLine::buildTextBlob(TextRange textRange, const TextStyle& style, const ClipContext& context) { |
| if (context.run->placeholderStyle() != nullptr) { |
| return; |
| } |
| |
| fTextBlobCache.emplace_back(); |
| TextBlobRecord& record = fTextBlobCache.back(); |
| |
| if (style.hasForeground()) { |
| record.fPaint = style.getForegroundPaintOrID(); |
| } else { |
| std::get<SkPaint>(record.fPaint).setColor(style.getColor()); |
| } |
| record.fVisitor_Run = context.run; |
| record.fVisitor_Pos = context.pos; |
| |
| // TODO: This is the change for flutter, must be removed later |
| SkTextBlobBuilder builder; |
| context.run->copyTo(builder, SkToU32(context.pos), context.size); |
| record.fClippingNeeded = context.clippingNeeded; |
| if (context.clippingNeeded) { |
| record.fClipRect = extendHeight(context).makeOffset(this->offset()); |
| } else { |
| record.fClipRect = context.clip.makeOffset(this->offset()); |
| } |
| |
| SkASSERT(nearlyEqual(context.run->baselineShift(), style.getBaselineShift())); |
| SkScalar correctedBaseline = SkScalarFloorToScalar(this->baseline() + style.getBaselineShift() + 0.5); |
| record.fBlob = builder.make(); |
| if (record.fBlob != nullptr) { |
| record.fBounds.joinPossiblyEmptyRect(record.fBlob->bounds()); |
| } |
| |
| record.fOffset = SkPoint::Make(this->offset().fX + context.fTextShift, |
| this->offset().fY + correctedBaseline); |
| } |
| |
| void TextLine::TextBlobRecord::paint(ParagraphPainter* painter, SkScalar x, SkScalar y) { |
| if (fClippingNeeded) { |
| painter->save(); |
| painter->clipRect(fClipRect.makeOffset(x, y)); |
| } |
| painter->drawTextBlob(fBlob, x + fOffset.x(), y + fOffset.y(), fPaint); |
| if (fClippingNeeded) { |
| painter->restore(); |
| } |
| } |
| |
| void TextLine::paintBackground(ParagraphPainter* painter, |
| SkScalar x, |
| SkScalar y, |
| TextRange textRange, |
| const TextStyle& style, |
| const ClipContext& context) const { |
| if (style.hasBackground()) { |
| painter->drawRect(context.clip.makeOffset(this->offset() + SkPoint::Make(x, y)), |
| style.getBackgroundPaintOrID()); |
| } |
| } |
| |
| void TextLine::paintShadow(ParagraphPainter* painter, |
| SkScalar x, |
| SkScalar y, |
| TextRange textRange, |
| const TextStyle& style, |
| const ClipContext& context) const { |
| SkScalar correctedBaseline = SkScalarFloorToScalar(this->baseline() + style.getBaselineShift() + 0.5); |
| |
| for (TextShadow shadow : style.getShadows()) { |
| if (!shadow.hasShadow()) continue; |
| |
| SkTextBlobBuilder builder; |
| context.run->copyTo(builder, context.pos, context.size); |
| |
| if (context.clippingNeeded) { |
| painter->save(); |
| SkRect clip = extendHeight(context); |
| clip.offset(x, y); |
| clip.offset(this->offset()); |
| painter->clipRect(clip); |
| } |
| auto blob = builder.make(); |
| painter->drawTextShadow(blob, |
| x + this->offset().fX + shadow.fOffset.x() + context.fTextShift, |
| y + this->offset().fY + shadow.fOffset.y() + correctedBaseline, |
| shadow.fColor, |
| SkDoubleToScalar(shadow.fBlurSigma)); |
| if (context.clippingNeeded) { |
| painter->restore(); |
| } |
| } |
| } |
| |
| void TextLine::paintDecorations(ParagraphPainter* painter, SkScalar x, SkScalar y, TextRange textRange, const TextStyle& style, const ClipContext& context) const { |
| ParagraphPainterAutoRestore ppar(painter); |
| painter->translate(x + this->offset().fX, y + this->offset().fY + style.getBaselineShift()); |
| Decorations decorations; |
| SkScalar correctedBaseline = SkScalarFloorToScalar(-this->sizes().rawAscent() + style.getBaselineShift() + 0.5); |
| decorations.paint(painter, style, context, correctedBaseline); |
| } |
| |
| void TextLine::justify(SkScalar maxWidth) { |
| int whitespacePatches = 0; |
| SkScalar textLen = 0; |
| SkScalar whitespaceLen = 0; |
| bool whitespacePatch = false; |
| // Take leading whitespaces width but do not increment a whitespace patch number |
| bool leadingWhitespaces = false; |
| this->iterateThroughClustersInGlyphsOrder(false, false, |
| [&](const Cluster* cluster, ClusterIndex index, bool ghost) { |
| if (cluster->isWhitespaceBreak()) { |
| if (index == 0) { |
| leadingWhitespaces = true; |
| } else if (!whitespacePatch && !leadingWhitespaces) { |
| // We only count patches BETWEEN words, not before |
| ++whitespacePatches; |
| } |
| whitespacePatch = !leadingWhitespaces; |
| whitespaceLen += cluster->width(); |
| } else if (cluster->isIdeographic()) { |
| // Whitespace break before and after |
| if (!whitespacePatch && index != 0) { |
| // We only count patches BETWEEN words, not before |
| ++whitespacePatches; // before |
| } |
| whitespacePatch = true; |
| leadingWhitespaces = false; |
| ++whitespacePatches; // after |
| } else { |
| whitespacePatch = false; |
| leadingWhitespaces = false; |
| } |
| textLen += cluster->width(); |
| return true; |
| }); |
| |
| if (whitespacePatch) { |
| // We only count patches BETWEEN words, not after |
| --whitespacePatches; |
| } |
| if (whitespacePatches == 0) { |
| if (fOwner->paragraphStyle().getTextDirection() == TextDirection::kRtl) { |
| // Justify -> Right align |
| fShift = maxWidth - textLen; |
| } |
| return; |
| } |
| |
| SkScalar step = (maxWidth - textLen + whitespaceLen) / whitespacePatches; |
| SkScalar shift = 0.0f; |
| SkScalar prevShift = 0.0f; |
| |
| // Deal with the ghost spaces |
| auto ghostShift = maxWidth - this->fAdvance.fX; |
| // Spread the extra whitespaces |
| whitespacePatch = false; |
| // Do not break on leading whitespaces |
| leadingWhitespaces = false; |
| this->iterateThroughClustersInGlyphsOrder(false, true, [&](const Cluster* cluster, ClusterIndex index, bool ghost) { |
| |
| if (ghost) { |
| if (cluster->run().leftToRight()) { |
| this->shiftCluster(cluster, ghostShift, ghostShift); |
| } |
| return true; |
| } |
| |
| if (cluster->isWhitespaceBreak()) { |
| if (index == 0) { |
| leadingWhitespaces = true; |
| } else if (!whitespacePatch && !leadingWhitespaces) { |
| shift += step; |
| whitespacePatch = true; |
| --whitespacePatches; |
| } |
| shift -= cluster->width(); |
| } else if (cluster->isIdeographic()) { |
| if (!whitespacePatch && index != 0) { |
| shift += step; |
| --whitespacePatches; |
| } |
| whitespacePatch = false; |
| leadingWhitespaces = false; |
| } else { |
| whitespacePatch = false; |
| leadingWhitespaces = false; |
| } |
| this->shiftCluster(cluster, shift, prevShift); |
| prevShift = shift; |
| // We skip ideographic whitespaces |
| if (!cluster->isWhitespaceBreak() && cluster->isIdeographic()) { |
| shift += step; |
| whitespacePatch = true; |
| --whitespacePatches; |
| } |
| return true; |
| }); |
| |
| if (whitespacePatch && whitespacePatches < 0) { |
| whitespacePatches++; |
| shift -= step; |
| } |
| |
| SkAssertResult(nearlyEqual(shift, maxWidth - textLen)); |
| SkASSERT(whitespacePatches == 0); |
| |
| this->fWidthWithSpaces += ghostShift; |
| this->fAdvance.fX = maxWidth; |
| } |
| |
| void TextLine::shiftCluster(const Cluster* cluster, SkScalar shift, SkScalar prevShift) { |
| |
| auto& run = cluster->run(); |
| auto start = cluster->startPos(); |
| auto end = cluster->endPos(); |
| |
| if (end == run.size()) { |
| // Set the same shift for the fake last glyph (to avoid all extra checks) |
| ++end; |
| } |
| |
| if (run.fJustificationShifts.empty()) { |
| // Do not fill this array until needed |
| run.fJustificationShifts.push_back_n(run.size() + 1, { 0, 0 }); |
| } |
| |
| for (size_t pos = start; pos < end; ++pos) { |
| run.fJustificationShifts[pos] = { shift, prevShift }; |
| } |
| } |
| |
| void TextLine::createEllipsis(SkScalar maxWidth, const SkString& ellipsis, bool) { |
| // Replace some clusters with the ellipsis |
| // Go through the clusters in the reverse logical order |
| // taking off cluster by cluster until the ellipsis fits |
| SkScalar width = fAdvance.fX; |
| RunIndex lastRun = EMPTY_RUN; |
| std::unique_ptr<Run> ellipsisRun; |
| for (auto clusterIndex = fGhostClusterRange.end; clusterIndex > fGhostClusterRange.start; --clusterIndex) { |
| auto& cluster = fOwner->cluster(clusterIndex - 1); |
| // Shape the ellipsis if the run has changed |
| if (lastRun != cluster.runIndex()) { |
| ellipsisRun = this->shapeEllipsis(ellipsis, &cluster); |
| if (ellipsisRun->advance().fX > maxWidth) { |
| // Ellipsis is bigger than the entire line; no way we can add it at all |
| // BUT! We can keep scanning in case the next run will give us better results |
| lastRun = EMPTY_RUN; |
| continue; |
| } else { |
| // We may need to continue |
| lastRun = cluster.runIndex(); |
| } |
| } |
| // See if it fits |
| if (width + ellipsisRun->advance().fX > maxWidth) { |
| width -= cluster.width(); |
| // Continue if the ellipsis does not fit |
| continue; |
| } |
| // We found enough room for the ellipsis |
| fAdvance.fX = width; |
| fEllipsis = std::move(ellipsisRun); |
| fEllipsis->setOwner(fOwner); |
| |
| // Let's update the line |
| fClusterRange.end = clusterIndex; |
| fGhostClusterRange.end = fClusterRange.end; |
| fEllipsis->fClusterStart = cluster.textRange().start; |
| fText.end = cluster.textRange().end; |
| fTextIncludingNewlines.end = cluster.textRange().end; |
| fTextExcludingSpaces.end = cluster.textRange().end; |
| break; |
| } |
| |
| if (!fEllipsis) { |
| // Weird situation: ellipsis does not fit; no ellipsis then |
| fClusterRange.end = fClusterRange.start; |
| fGhostClusterRange.end = fClusterRange.start; |
| fText.end = fText.start; |
| fTextIncludingNewlines.end = fTextIncludingNewlines.start; |
| fTextExcludingSpaces.end = fTextExcludingSpaces.start; |
| fAdvance.fX = 0; |
| } |
| } |
| |
| std::unique_ptr<Run> TextLine::shapeEllipsis(const SkString& ellipsis, const Cluster* cluster) { |
| |
| class ShapeHandler final : public SkShaper::RunHandler { |
| public: |
| ShapeHandler(SkScalar lineHeight, bool useHalfLeading, SkScalar baselineShift, const SkString& ellipsis) |
| : fRun(nullptr), fLineHeight(lineHeight), fUseHalfLeading(useHalfLeading), fBaselineShift(baselineShift), fEllipsis(ellipsis) {} |
| std::unique_ptr<Run> run() & { return std::move(fRun); } |
| |
| private: |
| void beginLine() override {} |
| |
| void runInfo(const RunInfo&) override {} |
| |
| void commitRunInfo() override {} |
| |
| Buffer runBuffer(const RunInfo& info) override { |
| SkASSERT(!fRun); |
| fRun = std::make_unique<Run>(nullptr, info, 0, fLineHeight, fUseHalfLeading, fBaselineShift, 0, 0); |
| return fRun->newRunBuffer(); |
| } |
| |
| void commitRunBuffer(const RunInfo& info) override { |
| fRun->fAdvance.fX = info.fAdvance.fX; |
| fRun->fAdvance.fY = fRun->advance().fY; |
| fRun->fPlaceholderIndex = std::numeric_limits<size_t>::max(); |
| fRun->fEllipsis = true; |
| } |
| |
| void commitLine() override {} |
| |
| std::unique_ptr<Run> fRun; |
| SkScalar fLineHeight; |
| bool fUseHalfLeading; |
| SkScalar fBaselineShift; |
| SkString fEllipsis; |
| }; |
| |
| const Run& run = cluster->run(); |
| TextStyle textStyle = fOwner->paragraphStyle().getTextStyle(); |
| for (auto i = fBlockRange.start; i < fBlockRange.end; ++i) { |
| auto& block = fOwner->block(i); |
| if (run.leftToRight() && cluster->textRange().end <= block.fRange.end) { |
| textStyle = block.fStyle; |
| break; |
| } else if (!run.leftToRight() && cluster->textRange().start <= block.fRange.end) { |
| textStyle = block.fStyle; |
| break; |
| } |
| } |
| |
| auto shaped = [&](sk_sp<SkTypeface> typeface, sk_sp<SkFontMgr> fallback) -> std::unique_ptr<Run> { |
| ShapeHandler handler(run.heightMultiplier(), run.useHalfLeading(), run.baselineShift(), ellipsis); |
| SkFont font(std::move(typeface), textStyle.getFontSize()); |
| font.setEdging(SkFont::Edging::kAntiAlias); |
| font.setHinting(SkFontHinting::kSlight); |
| font.setSubpixel(true); |
| |
| std::unique_ptr<SkShaper> shaper = SkShapers::HB::ShapeDontWrapOrReorder( |
| fOwner->getUnicode(), fallback ? fallback : SkFontMgr::RefEmpty()); |
| |
| const SkBidiIterator::Level defaultLevel = SkBidiIterator::kLTR; |
| const char* utf8 = ellipsis.c_str(); |
| size_t utf8Bytes = ellipsis.size(); |
| |
| std::unique_ptr<SkShaper::BiDiRunIterator> bidi = SkShapers::unicode::BidiRunIterator( |
| fOwner->getUnicode(), utf8, utf8Bytes, defaultLevel); |
| SkASSERT(bidi); |
| |
| std::unique_ptr<SkShaper::LanguageRunIterator> language = |
| SkShaper::MakeStdLanguageRunIterator(utf8, utf8Bytes); |
| SkASSERT(language); |
| |
| std::unique_ptr<SkShaper::ScriptRunIterator> script = |
| SkShapers::HB::ScriptRunIterator(utf8, utf8Bytes); |
| SkASSERT(script); |
| |
| std::unique_ptr<SkShaper::FontRunIterator> fontRuns = SkShaper::MakeFontMgrRunIterator( |
| utf8, utf8Bytes, font, fallback ? fallback : SkFontMgr::RefEmpty()); |
| SkASSERT(fontRuns); |
| |
| shaper->shape(utf8, |
| utf8Bytes, |
| *fontRuns, |
| *bidi, |
| *script, |
| *language, |
| nullptr, |
| 0, |
| std::numeric_limits<SkScalar>::max(), |
| &handler); |
| auto ellipsisRun = handler.run(); |
| ellipsisRun->fTextRange = TextRange(0, ellipsis.size()); |
| ellipsisRun->fOwner = fOwner; |
| return ellipsisRun; |
| }; |
| |
| // Check the current font |
| auto ellipsisRun = shaped(run.fFont.refTypeface(), nullptr); |
| if (ellipsisRun->isResolved()) { |
| return ellipsisRun; |
| } |
| |
| // Check all allowed fonts |
| std::vector<sk_sp<SkTypeface>> typefaces = fOwner->fontCollection()->findTypefaces( |
| textStyle.getFontFamilies(), textStyle.getFontStyle(), textStyle.getFontArguments()); |
| for (const auto& typeface : typefaces) { |
| ellipsisRun = shaped(typeface, nullptr); |
| if (ellipsisRun->isResolved()) { |
| return ellipsisRun; |
| } |
| } |
| |
| // Try the fallback |
| if (fOwner->fontCollection()->fontFallbackEnabled()) { |
| const char* ch = ellipsis.c_str(); |
| SkUnichar unicode = SkUTF::NextUTF8WithReplacement(&ch, |
| ellipsis.c_str() |
| + ellipsis.size()); |
| // We do not expect emojis in ellipsis so if they appeat there |
| // they will not be resolved with the pretiest color emoji font |
| auto typeface = fOwner->fontCollection()->defaultFallback( |
| unicode, |
| textStyle.getFontStyle(), |
| textStyle.getLocale()); |
| if (typeface) { |
| ellipsisRun = shaped(typeface, fOwner->fontCollection()->getFallbackManager()); |
| if (ellipsisRun->isResolved()) { |
| return ellipsisRun; |
| } |
| } |
| } |
| return ellipsisRun; |
| } |
| |
| TextLine::ClipContext TextLine::measureTextInsideOneRun(TextRange textRange, |
| const Run* run, |
| SkScalar runOffsetInLine, |
| SkScalar textOffsetInRunInLine, |
| bool includeGhostSpaces, |
| TextAdjustment textAdjustment) const { |
| ClipContext result = { run, 0, run->size(), 0, SkRect::MakeEmpty(), 0, false }; |
| |
| if (run->fEllipsis) { |
| // Both ellipsis and placeholders can only be measured as one glyph |
| result.fTextShift = runOffsetInLine; |
| result.clip = SkRect::MakeXYWH(runOffsetInLine, |
| sizes().runTop(run, this->fAscentStyle), |
| run->advance().fX, |
| run->calculateHeight(this->fAscentStyle,this->fDescentStyle)); |
| return result; |
| } else if (run->isPlaceholder()) { |
| result.fTextShift = runOffsetInLine; |
| if (SkScalarIsFinite(run->fFontMetrics.fAscent)) { |
| result.clip = SkRect::MakeXYWH(runOffsetInLine, |
| sizes().runTop(run, this->fAscentStyle), |
| run->advance().fX, |
| run->calculateHeight(this->fAscentStyle,this->fDescentStyle)); |
| } else { |
| result.clip = SkRect::MakeXYWH(runOffsetInLine, run->fFontMetrics.fAscent, run->advance().fX, 0); |
| } |
| return result; |
| } else if (textRange.empty()) { |
| return result; |
| } |
| |
| TextRange originalTextRange(textRange); // We need it for proportional measurement |
| // Find [start:end] clusters for the text |
| while (true) { |
| // Update textRange by cluster edges (shift start up to the edge of the cluster) |
| // TODO: remove this limitation? |
| TextRange updatedTextRange; |
| bool found; |
| std::tie(found, updatedTextRange.start, updatedTextRange.end) = |
| run->findLimitingGlyphClusters(textRange); |
| if (!found) { |
| return result; |
| } |
| |
| if ((textAdjustment & TextAdjustment::Grapheme) == 0) { |
| textRange = updatedTextRange; |
| break; |
| } |
| |
| // Update text range by grapheme edges (shift start up to the edge of the grapheme) |
| std::tie(found, updatedTextRange.start, updatedTextRange.end) = |
| run->findLimitingGraphemes(updatedTextRange); |
| if (updatedTextRange == textRange) { |
| break; |
| } |
| |
| // Some clusters are inside graphemes and we need to adjust them |
| //SkDebugf("Correct range: [%d:%d) -> [%d:%d)\n", textRange.start, textRange.end, startIndex, endIndex); |
| textRange = updatedTextRange; |
| |
| // Move the start until it's on the grapheme edge (and glypheme, too) |
| } |
| Cluster* start = &fOwner->cluster(fOwner->clusterIndex(textRange.start)); |
| Cluster* end = &fOwner->cluster(fOwner->clusterIndex(textRange.end - (textRange.width() == 0 ? 0 : 1))); |
| |
| if (!run->leftToRight()) { |
| std::swap(start, end); |
| } |
| result.pos = start->startPos(); |
| result.size = (end->isHardBreak() ? end->startPos() : end->endPos()) - start->startPos(); |
| auto textStartInRun = run->positionX(start->startPos()); |
| auto textStartInLine = runOffsetInLine + textOffsetInRunInLine; |
| if (!run->leftToRight()) { |
| std::swap(start, end); |
| } |
| /* |
| if (!run->fJustificationShifts.empty()) { |
| SkDebugf("Justification for [%d:%d)\n", textRange.start, textRange.end); |
| for (auto i = result.pos; i < result.pos + result.size; ++i) { |
| auto j = run->fJustificationShifts[i]; |
| SkDebugf("[%d] = %f %f\n", i, j.fX, j.fY); |
| } |
| } |
| */ |
| // Calculate the clipping rectangle for the text with cluster edges |
| // There are 2 cases: |
| // EOL (when we expect the last cluster clipped without any spaces) |
| // Anything else (when we want the cluster width contain all the spaces - |
| // coming from letter spacing or word spacing or justification) |
| result.clip = |
| SkRect::MakeXYWH(0, |
| sizes().runTop(run, this->fAscentStyle), |
| run->calculateWidth(result.pos, result.pos + result.size, false), |
| run->calculateHeight(this->fAscentStyle,this->fDescentStyle)); |
| |
| // Correct the width in case the text edges don't match clusters |
| // TODO: This is where we get smart about selecting a part of a cluster |
| // by shaping each grapheme separately and then use the result sizes |
| // to calculate the proportions |
| auto leftCorrection = start->sizeToChar(originalTextRange.start); |
| auto rightCorrection = end->sizeFromChar(originalTextRange.end - 1); |
| /* |
| SkDebugf("[%d: %d) => [%d: %d), @%d, %d: [%f:%f) + [%f:%f) = ", // جَآَهُ |
| originalTextRange.start, originalTextRange.end, textRange.start, textRange.end, |
| result.pos, result.size, |
| result.clip.fLeft, result.clip.fRight, leftCorrection, rightCorrection); |
| */ |
| result.clippingNeeded = leftCorrection != 0 || rightCorrection != 0; |
| if (run->leftToRight()) { |
| result.clip.fLeft += leftCorrection; |
| result.clip.fRight -= rightCorrection; |
| textStartInLine -= leftCorrection; |
| } else { |
| result.clip.fRight -= leftCorrection; |
| result.clip.fLeft += rightCorrection; |
| textStartInLine -= rightCorrection; |
| } |
| |
| result.clip.offset(textStartInLine, 0); |
| //SkDebugf("@%f[%f:%f)\n", textStartInLine, result.clip.fLeft, result.clip.fRight); |
| |
| if (compareRound(result.clip.fRight, fAdvance.fX, fOwner->getApplyRoundingHack()) > 0 && !includeGhostSpaces) { |
| // There are few cases when we need it. |
| // The most important one: we measure the text with spaces at the end (or at the beginning in RTL) |
| // and we should ignore these spaces |
| if (fOwner->paragraphStyle().getTextDirection() == TextDirection::kLtr) { |
| // We only use this member for LTR |
| result.fExcludedTrailingSpaces = std::max(result.clip.fRight - fAdvance.fX, 0.0f); |
| result.clippingNeeded = true; |
| result.clip.fRight = fAdvance.fX; |
| } |
| } |
| |
| if (result.clip.width() < 0) { |
| // Weird situation when glyph offsets move the glyph to the left |
| // (happens with zalgo texts, for instance) |
| result.clip.fRight = result.clip.fLeft; |
| } |
| |
| // The text must be aligned with the lineOffset |
| result.fTextShift = textStartInLine - textStartInRun; |
| |
| return result; |
| } |
| |
| void TextLine::iterateThroughClustersInGlyphsOrder(bool reversed, |
| bool includeGhosts, |
| const ClustersVisitor& visitor) const { |
| // Walk through the clusters in the logical order (or reverse) |
| SkSpan<const size_t> runs(fRunsInVisualOrder.data(), fRunsInVisualOrder.size()); |
| bool ignore = false; |
| ClusterIndex index = 0; |
| directional_for_each(runs, !reversed, [&](decltype(runs[0]) r) { |
| if (ignore) return; |
| auto run = this->fOwner->run(r); |
| auto trimmedRange = fClusterRange.intersection(run.clusterRange()); |
| auto trailedRange = fGhostClusterRange.intersection(run.clusterRange()); |
| SkASSERT(trimmedRange.start == trailedRange.start); |
| |
| auto trailed = fOwner->clusters(trailedRange); |
| auto trimmed = fOwner->clusters(trimmedRange); |
| directional_for_each(trailed, reversed != run.leftToRight(), [&](Cluster& cluster) { |
| if (ignore) return; |
| bool ghost = &cluster >= trimmed.end(); |
| if (!includeGhosts && ghost) { |
| return; |
| } |
| if (!visitor(&cluster, index++, ghost)) { |
| |
| ignore = true; |
| return; |
| } |
| }); |
| }); |
| } |
| |
| SkScalar TextLine::iterateThroughSingleRunByStyles(TextAdjustment textAdjustment, |
| const Run* run, |
| SkScalar runOffset, |
| TextRange textRange, |
| StyleType styleType, |
| const RunStyleVisitor& visitor) const { |
| auto correctContext = [&](TextRange textRange, SkScalar textOffsetInRun) -> ClipContext { |
| auto result = this->measureTextInsideOneRun( |
| textRange, run, runOffset, textOffsetInRun, false, textAdjustment); |
| if (styleType == StyleType::kDecorations) { |
| // Decorations are drawn based on the real font metrics (regardless of styles and strut) |
| result.clip.fTop = this->sizes().runTop(run, LineMetricStyle::CSS); |
| result.clip.fBottom = result.clip.fTop + |
| run->calculateHeight(LineMetricStyle::CSS, LineMetricStyle::CSS); |
| } |
| return result; |
| }; |
| |
| if (run->fEllipsis) { |
| // Extra efforts to get the ellipsis text style |
| ClipContext clipContext = correctContext(run->textRange(), 0.0f); |
| TextRange testRange(run->fClusterStart, run->fClusterStart + run->textRange().width()); |
| for (BlockIndex index = fBlockRange.start; index < fBlockRange.end; ++index) { |
| auto block = fOwner->styles().begin() + index; |
| auto intersect = intersected(block->fRange, testRange); |
| if (intersect.width() > 0) { |
| visitor(testRange, block->fStyle, clipContext); |
| return run->advance().fX; |
| } |
| } |
| SkASSERT(false); |
| } |
| |
| if (styleType == StyleType::kNone) { |
| ClipContext clipContext = correctContext(textRange, 0.0f); |
| // The placehoder can have height=0 or (exclusively) width=0 and still be a thing |
| if (clipContext.clip.height() > 0.0f || clipContext.clip.width() > 0.0f) { |
| visitor(textRange, TextStyle(), clipContext); |
| return clipContext.clip.width(); |
| } else { |
| return 0; |
| } |
| } |
| |
| TextIndex start = EMPTY_INDEX; |
| size_t size = 0; |
| const TextStyle* prevStyle = nullptr; |
| SkScalar textOffsetInRun = 0; |
| |
| const BlockIndex blockRangeSize = fBlockRange.end - fBlockRange.start; |
| for (BlockIndex index = 0; index <= blockRangeSize; ++index) { |
| |
| TextRange intersect; |
| TextStyle* style = nullptr; |
| if (index < blockRangeSize) { |
| auto block = fOwner->styles().begin() + |
| (run->leftToRight() ? fBlockRange.start + index : fBlockRange.end - index - 1); |
| |
| // Get the text |
| intersect = intersected(block->fRange, textRange); |
| if (intersect.width() == 0) { |
| if (start == EMPTY_INDEX) { |
| // This style is not applicable to the text yet |
| continue; |
| } else { |
| // We have found all the good styles already |
| // but we need to process the last one of them |
| intersect = TextRange(start, start + size); |
| index = fBlockRange.end; |
| } |
| } else { |
| // Get the style |
| style = &block->fStyle; |
| if (start != EMPTY_INDEX && style->matchOneAttribute(styleType, *prevStyle)) { |
| size += intersect.width(); |
| // RTL text intervals move backward |
| start = std::min(intersect.start, start); |
| continue; |
| } else if (start == EMPTY_INDEX ) { |
| // First time only |
| prevStyle = style; |
| size = intersect.width(); |
| start = intersect.start; |
| continue; |
| } |
| } |
| } else if (prevStyle != nullptr) { |
| // This is the last style |
| } else { |
| break; |
| } |
| |
| // We have the style and the text |
| auto runStyleTextRange = TextRange(start, start + size); |
| ClipContext clipContext = correctContext(runStyleTextRange, textOffsetInRun); |
| textOffsetInRun += clipContext.clip.width(); |
| if (clipContext.clip.height() == 0) { |
| continue; |
| } |
| visitor(runStyleTextRange, *prevStyle, clipContext); |
| |
| // Start all over again |
| prevStyle = style; |
| start = intersect.start; |
| size = intersect.width(); |
| } |
| return textOffsetInRun; |
| } |
| |
| void TextLine::iterateThroughVisualRuns(bool includingGhostSpaces, const RunVisitor& visitor) const { |
| |
| // Walk through all the runs that intersect with the line in visual order |
| SkScalar width = 0; |
| SkScalar runOffset = 0; |
| SkScalar totalWidth = 0; |
| auto textRange = includingGhostSpaces ? this->textWithNewlines() : this->trimmedText(); |
| |
| if (this->ellipsis() != nullptr && fOwner->paragraphStyle().getTextDirection() == TextDirection::kRtl) { |
| runOffset = this->ellipsis()->offset().fX; |
| if (visitor(ellipsis(), runOffset, ellipsis()->textRange(), &width)) { |
| } |
| } |
| |
| for (auto& runIndex : fRunsInVisualOrder) { |
| |
| const auto run = &this->fOwner->run(runIndex); |
| auto lineIntersection = intersected(run->textRange(), textRange); |
| if (lineIntersection.width() == 0 && this->width() != 0) { |
| // TODO: deal with empty runs in a better way |
| continue; |
| } |
| if (!run->leftToRight() && runOffset == 0 && includingGhostSpaces) { |
| // runOffset does not take in account a possibility |
| // that RTL run could start before the line (trailing spaces) |
| // so we need to do runOffset -= "trailing whitespaces length" |
| TextRange whitespaces = intersected( |
| TextRange(fTextExcludingSpaces.end, fTextIncludingNewlines.end), run->fTextRange); |
| if (whitespaces.width() > 0) { |
| auto whitespacesLen = measureTextInsideOneRun(whitespaces, run, runOffset, 0, true, TextAdjustment::GlyphCluster).clip.width(); |
| runOffset -= whitespacesLen; |
| } |
| } |
| runOffset += width; |
| totalWidth += width; |
| if (!visitor(run, runOffset, lineIntersection, &width)) { |
| return; |
| } |
| } |
| |
| runOffset += width; |
| totalWidth += width; |
| |
| if (this->ellipsis() != nullptr && fOwner->paragraphStyle().getTextDirection() == TextDirection::kLtr) { |
| if (visitor(ellipsis(), runOffset, ellipsis()->textRange(), &width)) { |
| totalWidth += width; |
| } |
| } |
| |
| if (!includingGhostSpaces && compareRound(totalWidth, this->width(), fOwner->getApplyRoundingHack()) != 0) { |
| // This is a very important assert! |
| // It asserts that 2 different ways of calculation come with the same results |
| SkDEBUGFAILF("ASSERT: %f != %f\n", totalWidth, this->width()); |
| } |
| } |
| |
| SkVector TextLine::offset() const { |
| return fOffset + SkVector::Make(fShift, 0); |
| } |
| |
| LineMetrics TextLine::getMetrics() const { |
| LineMetrics result; |
| SkASSERT(fOwner); |
| |
| // Fill out the metrics |
| fOwner->ensureUTF16Mapping(); |
| result.fStartIndex = fOwner->getUTF16Index(fTextExcludingSpaces.start); |
| result.fEndExcludingWhitespaces = fOwner->getUTF16Index(fTextExcludingSpaces.end); |
| result.fEndIndex = fOwner->getUTF16Index(fText.end); |
| result.fEndIncludingNewline = fOwner->getUTF16Index(fTextIncludingNewlines.end); |
| result.fHardBreak = endsWithHardLineBreak(); |
| result.fAscent = - fMaxRunMetrics.ascent(); |
| result.fDescent = fMaxRunMetrics.descent(); |
| result.fUnscaledAscent = - fMaxRunMetrics.ascent(); // TODO: implement |
| result.fHeight = fAdvance.fY; |
| result.fWidth = fAdvance.fX; |
| if (fOwner->getApplyRoundingHack()) { |
| result.fHeight = littleRound(result.fHeight); |
| result.fWidth = littleRound(result.fWidth); |
| } |
| result.fLeft = this->offset().fX; |
| // This is Flutter definition of a baseline |
| result.fBaseline = this->offset().fY + this->height() - this->sizes().descent(); |
| result.fLineNumber = this - fOwner->lines().begin(); |
| |
| // Fill out the style parts |
| this->iterateThroughVisualRuns(false, |
| [this, &result] |
| (const Run* run, SkScalar runOffsetInLine, TextRange textRange, SkScalar* runWidthInLine) { |
| if (run->placeholderStyle() != nullptr) { |
| *runWidthInLine = run->advance().fX; |
| return true; |
| } |
| *runWidthInLine = this->iterateThroughSingleRunByStyles( |
| TextAdjustment::GlyphCluster, run, runOffsetInLine, textRange, StyleType::kForeground, |
| [&result, &run](TextRange textRange, const TextStyle& style, const ClipContext& context) { |
| SkFontMetrics fontMetrics; |
| run->fFont.getMetrics(&fontMetrics); |
| StyleMetrics styleMetrics(&style, fontMetrics); |
| result.fLineMetrics.emplace(textRange.start, styleMetrics); |
| }); |
| return true; |
| }); |
| |
| return result; |
| } |
| |
| bool TextLine::isFirstLine() const { |
| return this == &fOwner->lines().front(); |
| } |
| |
| bool TextLine::isLastLine() const { |
| return this == &fOwner->lines().back(); |
| } |
| |
| bool TextLine::endsWithHardLineBreak() const { |
| // TODO: For some reason Flutter imagines a hard line break at the end of the last line. |
| // To be removed... |
| return (fGhostClusterRange.width() > 0 && fOwner->cluster(fGhostClusterRange.end - 1).isHardBreak()) || |
| fEllipsis != nullptr || |
| fGhostClusterRange.end == fOwner->clusters().size() - 1; |
| } |
| |
| void TextLine::getRectsForRange(TextRange textRange0, |
| RectHeightStyle rectHeightStyle, |
| RectWidthStyle rectWidthStyle, |
| std::vector<TextBox>& boxes) const |
| { |
| const Run* lastRun = nullptr; |
| auto startBox = boxes.size(); |
| this->iterateThroughVisualRuns(true, |
| [textRange0, rectHeightStyle, rectWidthStyle, &boxes, &lastRun, startBox, this] |
| (const Run* run, SkScalar runOffsetInLine, TextRange textRange, SkScalar* runWidthInLine) { |
| *runWidthInLine = this->iterateThroughSingleRunByStyles( |
| TextAdjustment::GraphemeGluster, run, runOffsetInLine, textRange, StyleType::kNone, |
| [run, runOffsetInLine, textRange0, rectHeightStyle, rectWidthStyle, &boxes, &lastRun, startBox, this] |
| (TextRange textRange, const TextStyle& style, const TextLine::ClipContext& lineContext) { |
| |
| auto intersect = textRange * textRange0; |
| if (intersect.empty()) { |
| return true; |
| } |
| |
| auto paragraphStyle = fOwner->paragraphStyle(); |
| |
| // Found a run that intersects with the text |
| auto context = this->measureTextInsideOneRun( |
| intersect, run, runOffsetInLine, 0, true, TextAdjustment::GraphemeGluster); |
| SkRect clip = context.clip; |
| clip.offset(lineContext.fTextShift - context.fTextShift, 0); |
| |
| switch (rectHeightStyle) { |
| case RectHeightStyle::kMax: |
| // TODO: Change it once flutter rolls into google3 |
| // (probably will break things if changed before) |
| clip.fBottom = this->height(); |
| clip.fTop = this->sizes().delta(); |
| break; |
| case RectHeightStyle::kIncludeLineSpacingTop: { |
| clip.fBottom = this->height(); |
| clip.fTop = this->sizes().delta(); |
| auto verticalShift = this->sizes().rawAscent() - this->sizes().ascent(); |
| if (isFirstLine()) { |
| clip.fTop += verticalShift; |
| } |
| break; |
| } |
| case RectHeightStyle::kIncludeLineSpacingMiddle: { |
| clip.fBottom = this->height(); |
| clip.fTop = this->sizes().delta(); |
| auto verticalShift = this->sizes().rawAscent() - this->sizes().ascent(); |
| clip.offset(0, verticalShift / 2.0); |
| if (isFirstLine()) { |
| clip.fTop += verticalShift / 2.0; |
| } |
| if (isLastLine()) { |
| clip.fBottom -= verticalShift / 2.0; |
| } |
| break; |
| } |
| case RectHeightStyle::kIncludeLineSpacingBottom: { |
| clip.fBottom = this->height(); |
| clip.fTop = this->sizes().delta(); |
| auto verticalShift = this->sizes().rawAscent() - this->sizes().ascent(); |
| clip.offset(0, verticalShift); |
| if (isLastLine()) { |
| clip.fBottom -= verticalShift; |
| } |
| break; |
| } |
| case RectHeightStyle::kStrut: { |
| const auto& strutStyle = paragraphStyle.getStrutStyle(); |
| if (strutStyle.getStrutEnabled() |
| && strutStyle.getFontSize() > 0) { |
| auto strutMetrics = fOwner->strutMetrics(); |
| auto top = this->baseline(); |
| clip.fTop = top + strutMetrics.ascent(); |
| clip.fBottom = top + strutMetrics.descent(); |
| } |
| } |
| break; |
| case RectHeightStyle::kTight: { |
| if (run->fHeightMultiplier <= 0) { |
| break; |
| } |
| const auto effectiveBaseline = this->baseline() + this->sizes().delta(); |
| clip.fTop = effectiveBaseline + run->ascent(); |
| clip.fBottom = effectiveBaseline + run->descent(); |
| } |
| break; |
| default: |
| SkASSERT(false); |
| break; |
| } |
| |
| // Separate trailing spaces and move them in the default order of the paragraph |
| // in case the run order and the paragraph order don't match |
| SkRect trailingSpaces = SkRect::MakeEmpty(); |
| if (this->trimmedText().end <this->textWithNewlines().end && // Line has trailing space |
| this->textWithNewlines().end == intersect.end && // Range is at the end of the line |
| this->trimmedText().end > intersect.start) // Range has more than just spaces |
| { |
| auto delta = this->spacesWidth(); |
| trailingSpaces = SkRect::MakeXYWH(0, 0, 0, 0); |
| // There are trailing spaces in this run |
| if (paragraphStyle.getTextAlign() == TextAlign::kJustify && isLastLine()) |
| { |
| // TODO: this is just a patch. Make it right later (when it's clear what and how) |
| trailingSpaces = clip; |
| if(run->leftToRight()) { |
| trailingSpaces.fLeft = this->width(); |
| clip.fRight = this->width(); |
| } else { |
| trailingSpaces.fRight = 0; |
| clip.fLeft = 0; |
| } |
| } else if (paragraphStyle.getTextDirection() == TextDirection::kRtl && |
| !run->leftToRight()) |
| { |
| // Split |
| trailingSpaces = clip; |
| trailingSpaces.fLeft = - delta; |
| trailingSpaces.fRight = 0; |
| clip.fLeft += delta; |
| } else if (paragraphStyle.getTextDirection() == TextDirection::kLtr && |
| run->leftToRight()) |
| { |
| // Split |
| trailingSpaces = clip; |
| trailingSpaces.fLeft = this->width(); |
| trailingSpaces.fRight = trailingSpaces.fLeft + delta; |
| clip.fRight -= delta; |
| } |
| } |
| |
| clip.offset(this->offset()); |
| if (trailingSpaces.width() > 0) { |
| trailingSpaces.offset(this->offset()); |
| } |
| |
| // Check if we can merge two boxes instead of adding a new one |
| auto merge = [&lastRun, &context, &boxes](SkRect clip) { |
| bool mergedBoxes = false; |
| if (!boxes.empty() && |
| lastRun != nullptr && |
| context.run->leftToRight() == lastRun->leftToRight() && |
| lastRun->placeholderStyle() == nullptr && |
| context.run->placeholderStyle() == nullptr && |
| nearlyEqual(lastRun->heightMultiplier(), |
| context.run->heightMultiplier()) && |
| lastRun->font() == context.run->font()) |
| { |
| auto& lastBox = boxes.back(); |
| if (nearlyEqual(lastBox.rect.fTop, clip.fTop) && |
| nearlyEqual(lastBox.rect.fBottom, clip.fBottom) && |
| (nearlyEqual(lastBox.rect.fLeft, clip.fRight) || |
| nearlyEqual(lastBox.rect.fRight, clip.fLeft))) |
| { |
| lastBox.rect.fLeft = std::min(lastBox.rect.fLeft, clip.fLeft); |
| lastBox.rect.fRight = std::max(lastBox.rect.fRight, clip.fRight); |
| mergedBoxes = true; |
| } |
| } |
| lastRun = context.run; |
| return mergedBoxes; |
| }; |
| |
| if (!merge(clip)) { |
| boxes.emplace_back(clip, context.run->getTextDirection()); |
| } |
| if (!nearlyZero(trailingSpaces.width()) && !merge(trailingSpaces)) { |
| boxes.emplace_back(trailingSpaces, paragraphStyle.getTextDirection()); |
| } |
| |
| if (rectWidthStyle == RectWidthStyle::kMax && !isLastLine()) { |
| // Align the very left/right box horizontally |
| auto lineStart = this->offset().fX; |
| auto lineEnd = this->offset().fX + this->width(); |
| auto left = boxes[startBox]; |
| auto right = boxes.back(); |
| if (left.rect.fLeft > lineStart && left.direction == TextDirection::kRtl) { |
| left.rect.fRight = left.rect.fLeft; |
| left.rect.fLeft = 0; |
| boxes.insert(boxes.begin() + startBox + 1, left); |
| } |
| if (right.direction == TextDirection::kLtr && |
| right.rect.fRight >= lineEnd && |
| right.rect.fRight < fOwner->widthWithTrailingSpaces()) { |
| right.rect.fLeft = right.rect.fRight; |
| right.rect.fRight = fOwner->widthWithTrailingSpaces(); |
| boxes.emplace_back(right); |
| } |
| } |
| |
| return true; |
| }); |
| return true; |
| }); |
| if (fOwner->getApplyRoundingHack()) { |
| for (auto& r : boxes) { |
| r.rect.fLeft = littleRound(r.rect.fLeft); |
| r.rect.fRight = littleRound(r.rect.fRight); |
| r.rect.fTop = littleRound(r.rect.fTop); |
| r.rect.fBottom = littleRound(r.rect.fBottom); |
| } |
| } |
| } |
| |
| PositionWithAffinity TextLine::getGlyphPositionAtCoordinate(SkScalar dx) { |
| |
| if (SkScalarNearlyZero(this->width()) && SkScalarNearlyZero(this->spacesWidth())) { |
| // TODO: this is one of the flutter changes that have to go away eventually |
| // Empty line is a special case in txtlib (but only when there are no spaces, too) |
| auto utf16Index = fOwner->getUTF16Index(this->fTextExcludingSpaces.end); |
| return { SkToS32(utf16Index) , kDownstream }; |
| } |
| |
| PositionWithAffinity result(0, Affinity::kDownstream); |
| this->iterateThroughVisualRuns(true, |
| [this, dx, &result] |
| (const Run* run, SkScalar runOffsetInLine, TextRange textRange, SkScalar* runWidthInLine) { |
| bool keepLooking = true; |
| *runWidthInLine = this->iterateThroughSingleRunByStyles( |
| TextAdjustment::GraphemeGluster, run, runOffsetInLine, textRange, StyleType::kNone, |
| [this, run, dx, &result, &keepLooking] |
| (TextRange textRange, const TextStyle& style, const TextLine::ClipContext& context0) { |
| |
| SkScalar offsetX = this->offset().fX; |
| ClipContext context = context0; |
| |
| // Correct the clip size because libtxt counts trailing spaces |
| if (run->leftToRight()) { |
| context.clip.fRight += context.fExcludedTrailingSpaces; // extending clip to the right |
| } else { |
| // Clip starts from 0; we cannot extend it to the left from that |
| } |
| // However, we need to offset the clip |
| context.clip.offset(offsetX, 0.0f); |
| |
| // This patch will help us to avoid a floating point error |
| if (SkScalarNearlyEqual(context.clip.fRight, dx, 0.01f)) { |
| context.clip.fRight = dx; |
| } |
| |
| if (dx <= context.clip.fLeft) { |
| // All the other runs are placed right of this one |
| auto utf16Index = fOwner->getUTF16Index(context.run->globalClusterIndex(context.pos)); |
| if (run->leftToRight()) { |
| result = { SkToS32(utf16Index), kDownstream}; |
| keepLooking = false; |
| } else { |
| result = { SkToS32(utf16Index + 1), kUpstream}; |
| // If we haven't reached the end of the run we need to keep looking |
| keepLooking = context.pos != 0; |
| } |
| // For RTL we go another way |
| return !run->leftToRight(); |
| } |
| |
| if (dx >= context.clip.fRight) { |
| // We have to keep looking ; just in case keep the last one as the closest |
| auto utf16Index = fOwner->getUTF16Index(context.run->globalClusterIndex(context.pos + context.size)); |
| if (run->leftToRight()) { |
| result = {SkToS32(utf16Index), kUpstream}; |
| } else { |
| result = {SkToS32(utf16Index), kDownstream}; |
| } |
| // For RTL we go another way |
| return run->leftToRight(); |
| } |
| |
| // So we found the run that contains our coordinates |
| // Find the glyph position in the run that is the closest left of our point |
| // TODO: binary search |
| size_t found = context.pos; |
| for (size_t index = context.pos; index < context.pos + context.size; ++index) { |
| // TODO: this rounding is done to match Flutter tests. Must be removed.. |
| auto end = context.run->positionX(index) + context.fTextShift + offsetX; |
| if (fOwner->getApplyRoundingHack()) { |
| end = littleRound(end); |
| } |
| if (end > dx) { |
| break; |
| } else if (end == dx && !context.run->leftToRight()) { |
| // When we move RTL variable end points to the beginning of the code point which is included |
| found = index; |
| break; |
| } |
| found = index; |
| } |
| |
| SkScalar glyphemePosLeft = context.run->positionX(found) + context.fTextShift + offsetX; |
| SkScalar glyphemesWidth = context.run->positionX(found + 1) - context.run->positionX(found); |
| |
| // Find the grapheme range that contains the point |
| auto clusterIndex8 = context.run->globalClusterIndex(found); |
| auto clusterEnd8 = context.run->globalClusterIndex(found + 1); |
| auto graphemes = fOwner->countSurroundingGraphemes({clusterIndex8, clusterEnd8}); |
| |
| SkScalar center = glyphemePosLeft + glyphemesWidth / 2; |
| if (graphemes.size() > 1) { |
| // Calculate the position proportionally based on grapheme count |
| SkScalar averageGraphemeWidth = glyphemesWidth / graphemes.size(); |
| SkScalar delta = dx - glyphemePosLeft; |
| int graphemeIndex = SkScalarNearlyZero(averageGraphemeWidth) |
| ? 0 |
| : SkScalarFloorToInt(delta / averageGraphemeWidth); |
| auto graphemeCenter = glyphemePosLeft + graphemeIndex * averageGraphemeWidth + |
| averageGraphemeWidth / 2; |
| auto graphemeUtf8Index = graphemes[graphemeIndex]; |
| if ((dx < graphemeCenter) == context.run->leftToRight()) { |
| size_t utf16Index = fOwner->getUTF16Index(graphemeUtf8Index); |
| result = { SkToS32(utf16Index), kDownstream }; |
| } else { |
| size_t utf16Index = fOwner->getUTF16Index(graphemeUtf8Index + 1); |
| result = { SkToS32(utf16Index), kUpstream }; |
| } |
| // Keep UTF16 index as is |
| } else if ((dx < center) == context.run->leftToRight()) { |
| size_t utf16Index = fOwner->getUTF16Index(clusterIndex8); |
| result = { SkToS32(utf16Index), kDownstream }; |
| } else { |
| size_t utf16Index = context.run->leftToRight() |
| ? fOwner->getUTF16Index(clusterEnd8) |
| : fOwner->getUTF16Index(clusterIndex8) + 1; |
| result = { SkToS32(utf16Index), kUpstream }; |
| } |
| |
| return keepLooking = false; |
| |
| }); |
| return keepLooking; |
| } |
| ); |
| return result; |
| } |
| |
| void TextLine::getRectsForPlaceholders(std::vector<TextBox>& boxes) { |
| this->iterateThroughVisualRuns( |
| true, |
| [&boxes, this](const Run* run, SkScalar runOffset, TextRange textRange, |
| SkScalar* width) { |
| auto context = this->measureTextInsideOneRun( |
| textRange, run, runOffset, 0, true, TextAdjustment::GraphemeGluster); |
| *width = context.clip.width(); |
| |
| if (textRange.width() == 0) { |
| return true; |
| } |
| if (!run->isPlaceholder()) { |
| return true; |
| } |
| |
| SkRect clip = context.clip; |
| clip.offset(this->offset()); |
| |
| if (fOwner->getApplyRoundingHack()) { |
| clip.fLeft = littleRound(clip.fLeft); |
| clip.fRight = littleRound(clip.fRight); |
| clip.fTop = littleRound(clip.fTop); |
| clip.fBottom = littleRound(clip.fBottom); |
| } |
| boxes.emplace_back(clip, run->getTextDirection()); |
| return true; |
| }); |
| } |
| } // namespace textlayout |
| } // namespace skia |