| /* |
| * Copyright (C) 2021-2025 Apple Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
| * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| * DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS BE LIABLE FOR ANY |
| * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
| * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
| * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON |
| * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
| * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #pragma once |
| |
| #include "FloatRect.h" |
| #include "FloatRoundedRect.h" |
| #include "InlineIteratorInlineBox.h" |
| #include "InlineIteratorTextBox.h" |
| #include "PaintInfo.h" |
| #include "RenderObject.h" |
| #include "StylePrimitiveNumeric.h" |
| #include "TextBoxSelectableRange.h" |
| #include "TextDecorationPainter.h" |
| #include "TextRun.h" |
| |
| namespace WebCore { |
| |
| class Color; |
| class Document; |
| class RenderCombineText; |
| class RenderStyle; |
| class RenderText; |
| struct CompositionUnderline; |
| struct MarkedText; |
| struct StyledMarkedText; |
| class TextPainter; |
| |
| class TextBoxPainter { |
| public: |
| TextBoxPainter(const LayoutIntegration::InlineContent&, const InlineDisplay::Box&, const RenderStyle&, PaintInfo&, const LayoutPoint& paintOffset); |
| ~TextBoxPainter(); |
| |
| void paint(); |
| |
| static inline FloatSize rotateShadowOffset(const SpaceSeparatedPoint<Style::Length<CSS::AllUnzoomed>>& offset, WritingMode, const Style::ZoomFactor&); |
| |
| protected: |
| auto& textBox() const { return m_textBox; } |
| InlineIterator::TextBoxIterator makeIterator() const; |
| |
| void paintBackgroundFill(); |
| enum class BackgroundStyle : bool { Normal, Rounded }; |
| void paintBackgroundFillForRange(unsigned startOffset, unsigned endOffset, const Color&, BackgroundStyle); |
| |
| void paintForegroundAndDecorations(); |
| void paintCompositionUnderlines(); |
| void paintCompositionForeground(const StyledMarkedText&); |
| void paintPlatformDocumentMarkers(); |
| |
| void paintForeground(const StyledMarkedText&); |
| bool paintForegroundForShapeRange(TextPainter&); |
| TextDecorationPainter createDecorationPainter(const StyledMarkedText&, const FloatRect&); |
| void paintBackgroundDecorations(TextDecorationPainter&, const StyledMarkedText&, const FloatRect&); |
| void paintForegroundDecorations(TextDecorationPainter&, const StyledMarkedText&, const FloatRect&); |
| void paintCompositionUnderline(const CompositionUnderline&, const CornerRadii&, bool hasLiveConversion); |
| void fillCompositionUnderline(float start, float width, const CompositionUnderline&, const CornerRadii&, bool hasLiveConversion) const; |
| void paintPlatformDocumentMarker(const MarkedText&); |
| LayoutRect selectionRectForRange(unsigned startOffset, unsigned endOffset) const; |
| |
| float textPosition(); |
| FloatRect computePaintRect(const LayoutPoint& paintOffset); |
| bool computeHaveSelection() const; |
| std::pair<unsigned, unsigned> selectionStartEnd() const; |
| MarkedText createMarkedTextFromSelectionInBox(); |
| const FontCascade& fontCascade() const; |
| WritingMode writingMode() const { return m_style->writingMode(); } |
| FloatPoint textOriginFromPaintRect(const FloatRect&) const; |
| bool isInsideShapedContent() const; |
| |
| struct DecoratingBox { |
| InlineIterator::InlineBoxIterator inlineBox; |
| const CheckedRef<const RenderStyle> style; |
| TextDecorationPainter::Styles textDecorationStyles; |
| FloatPoint location; |
| }; |
| using DecoratingBoxList = Vector<DecoratingBox>; |
| void collectDecoratingBoxesForBackgroundPainting(DecoratingBoxList&, const InlineIterator::TextBoxIterator&, FloatPoint textBoxLocation, const TextDecorationPainter::Styles&); |
| |
| // FIXME: We could just talk to the display box directly. |
| const InlineIterator::BoxModernPath m_textBox; |
| const CheckedRef<const RenderText> m_renderer; |
| const CheckedRef<const Document> m_document; |
| const CheckedRef<const RenderStyle> m_style; |
| const FloatRect m_logicalRect; |
| const TextRun m_paintTextRun; |
| PaintInfo& m_paintInfo; |
| const TextBoxSelectableRange m_selectableRange; |
| const LayoutPoint m_paintOffset; |
| const FloatRect m_paintRect; |
| const bool m_isFirstLine; |
| const bool m_isCombinedText; |
| const bool m_isPrinting; |
| const bool m_haveSelection; |
| bool m_containsComposition { false }; |
| bool m_compositionWithCustomUnderlines { false }; |
| }; |
| |
| inline FloatSize TextBoxPainter::rotateShadowOffset(const SpaceSeparatedPoint<Style::Length<CSS::AllUnzoomed>>& offset, WritingMode writingMode, const Style::ZoomFactor& zoomFactor) |
| { |
| if (writingMode.isHorizontal()) { |
| return { |
| offset.x().resolveZoom(zoomFactor), |
| offset.y().resolveZoom(zoomFactor), |
| }; |
| } |
| |
| if (writingMode.isLineOverLeft()) { // sideways-lr |
| return { |
| -offset.y().resolveZoom(zoomFactor), |
| offset.x().resolveZoom(zoomFactor), |
| }; |
| } |
| |
| return { |
| offset.y().resolveZoom(zoomFactor), |
| -offset.x().resolveZoom(zoomFactor), |
| }; |
| } |
| |
| } // namespace WebCore |