blob: acc60466c28e34542e75955201926771a9b1ff25 [file] [log] [blame]
/*
* Copyright (C) 2000 Lars Knoll ([email protected])
* (C) 2000 Antti Koivisto ([email protected])
* (C) 2000 Dirk Mueller ([email protected])
* (C) 2004 Allan Sandfeld Jensen ([email protected])
* Copyright (C) 2003-2022 Apple Inc. All rights reserved.
* Copyright (C) 2009 Google Inc. All rights reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*
*/
#pragma once
#include <WebCore/CachedImageClient.h>
#include <WebCore/LayoutRect.h>
#include <WebCore/PlatformLayerIdentifier.h>
#include <WebCore/RenderObjectEnums.h>
#include <WebCore/RenderStyleConstants.h>
#include <WebCore/RepaintRectCalculation.h>
#include <wtf/CheckedPtr.h>
#include <wtf/EnumSet.h>
#include <wtf/Platform.h>
#include <wtf/TZoneMalloc.h>
namespace WTF {
class TextStream;
}
namespace WebCore {
class AffineTransform;
class Color;
class ControlPart;
class Cursor;
class Document;
class FloatQuad;
class HitTestLocation;
class HitTestRequest;
class HitTestResult;
class HostWindow;
class LegacyInlineBox;
class LocalFrame;
class LocalFrameViewLayoutContext;
class Node;
class Page;
class Path;
class Position;
class PositionWithAffinity;
class ReferencedSVGResources;
class RenderBox;
class RenderBoxModelObject;
class RenderInline;
class RenderBlock;
class RenderBlockFlow;
class RenderElement;
class RenderFragmentedFlow;
class RenderGeometryMap;
class RenderLayer;
class RenderLayerModelObject;
class RenderFragmentContainer;
class RenderStyle;
class RenderTheme;
class RenderTreeBuilder;
class RenderView;
class RenderHighlight;
class ScrollAnchoringController;
class SelectionGeometry;
class Settings;
class TransformState;
class TreeScope;
class VisiblePosition;
class WeakPtrImplWithEventTargetData;
struct InlineBoxAndOffset;
struct PaintInfo;
struct ScrollRectToVisibleOptions;
struct SimpleRange;
struct VisibleRectContext;
namespace Layout {
class Box;
}
namespace Style {
class PseudoElementRequest;
enum class MarginTrimSide : uint8_t;
}
enum class Affinity : bool;
enum class HitTestSource : bool;
enum class RepaintOutlineBounds : bool { No, Yes };
enum class PointerEvents : uint8_t;
enum class RequiresFullRepaint : bool { No, Yes };
enum class StyleColorOptions : uint8_t;
typedef const void* WrappedImagePtr;
// Base class for all rendering tree objects.
class RenderObject : public CanMakeSingleThreadWeakPtr<RenderObject>, public CanMakeCheckedPtr<RenderObject> {
WTF_MAKE_PREFERABLY_COMPACT_TZONE_ALLOCATED(RenderObject);
WTF_OVERRIDE_DELETE_FOR_CHECKED_PTR(RenderObject);
friend class RenderBlock;
friend class RenderBlockFlow;
friend class RenderBox;
friend class RenderElement;
friend class RenderLayer;
friend class RenderLayerScrollableArea;
friend class RenderTreeBuilder;
public:
enum class Type : uint8_t {
#if ENABLE(ATTACHMENT_ELEMENT)
Attachment,
#endif
BlockFlow,
Button,
CombineText,
Counter,
DeprecatedFlexibleBox,
DetailsMarker,
EmbeddedObject,
FileUploadControl,
FlexibleBox,
Frame,
FrameSet,
Grid,
HTMLCanvas,
IFrame,
Image,
Inline,
LineBreak,
ListBox,
ListItem,
ListMarker,
Media,
MenuList,
Meter,
#if ENABLE(MODEL_ELEMENT)
Model,
#endif
MultiColumnFlow,
MultiColumnSet,
MultiColumnSpannerPlaceholder,
Progress,
Quote,
Replica,
ScrollbarPart,
SearchField,
Slider,
SliderContainer,
Table,
TableCaption,
TableCell,
TableCol,
TableRow,
TableSection,
Text,
TextControlInnerBlock,
TextControlInnerContainer,
TextControlMultiLine,
TextControlSingleLine,
TextFragment,
VTTCue,
Video,
View,
ViewTransitionCapture,
#if ENABLE(MATHML)
MathMLBlock,
MathMLFenced,
MathMLFencedOperator,
MathMLFraction,
MathMLMath,
MathMLMenclose,
MathMLOperator,
MathMLPadded,
MathMLRoot,
MathMLRow,
MathMLScripts,
MathMLSpace,
MathMLTable,
MathMLToken,
MathMLUnderOver,
#endif
SVGEllipse,
SVGForeignObject,
SVGGradientStop,
SVGHiddenContainer,
SVGImage,
SVGInline,
SVGInlineText,
SVGPath,
SVGRect,
SVGResourceClipper,
SVGResourceFilter,
SVGResourceFilterPrimitive,
SVGResourceLinearGradient,
SVGResourceMarker,
SVGResourceMasker,
SVGResourcePattern,
SVGResourceRadialGradient,
SVGRoot,
SVGTSpan,
SVGText,
SVGTextPath,
SVGTransformableContainer,
SVGViewportContainer,
LegacySVGEllipse,
LegacySVGForeignObject,
LegacySVGHiddenContainer,
LegacySVGImage,
LegacySVGPath,
LegacySVGRect,
LegacySVGResourceClipper,
LegacySVGResourceFilter,
LegacySVGResourceFilterPrimitive,
LegacySVGResourceLinearGradient,
LegacySVGResourceMarker,
LegacySVGResourceMasker,
LegacySVGResourcePattern,
LegacySVGResourceRadialGradient,
LegacySVGRoot,
LegacySVGTransformableContainer,
LegacySVGViewportContainer
};
enum class TypeFlag : uint8_t {
IsAnonymous = 1 << 0,
IsText = 1 << 1,
IsBox = 1 << 2,
IsBoxModelObject = 1 << 3,
IsLayerModelObject = 1 << 4,
IsRenderInline = 1 << 5,
IsRenderBlock = 1 << 6,
IsFlexibleBox = 1 << 7,
};
// Type Specific Flags
enum class BlockFlowFlag : uint8_t {
IsFragmentContainer = 1 << 0,
IsFragmentedFlow = 1 << 1,
IsTextControl = 1 << 2,
IsSVGBlock = 1 << 3,
IsViewTransitionContainingBlock = 1 << 4,
};
enum class LineBreakFlag : uint8_t {
IsWBR = 1 << 0,
};
enum class ReplacedFlag : uint8_t {
IsImage = 1 << 0,
IsMedia = 1 << 1,
IsWidget = 1 << 2,
IsViewTransitionCapture = 1 << 3,
UsesBoundaryCaching = 1 << 5,
};
enum class SVGModelObjectFlag : uint8_t {
IsLegacy = 1 << 0,
IsContainer = 1 << 1,
IsHiddenContainer = 1 << 2,
IsResourceContainer = 1 << 3,
IsShape = 1 << 4,
UsesBoundaryCaching = 1 << 5,
};
class TypeSpecificFlags {
public:
enum class Kind : uint8_t {
Invalid = 0,
BlockFlow,
LineBreak,
Replaced,
SVGModelObject,
};
TypeSpecificFlags() = default;
TypeSpecificFlags(OptionSet<BlockFlowFlag> flags)
: m_kind(enumToUnderlyingType(Kind::BlockFlow))
, m_flags(flags.toRaw())
{
ASSERT(blockFlowFlags() == flags);
}
TypeSpecificFlags(OptionSet<LineBreakFlag> flags)
: m_kind(enumToUnderlyingType(Kind::LineBreak))
, m_flags(flags.toRaw())
{
ASSERT(lineBreakFlags() == flags);
}
TypeSpecificFlags(OptionSet<ReplacedFlag> flags)
: m_kind(enumToUnderlyingType(Kind::Replaced))
, m_flags(flags.toRaw())
{
ASSERT(replacedFlags() == flags);
}
TypeSpecificFlags(OptionSet<SVGModelObjectFlag> flags)
: m_kind(enumToUnderlyingType(Kind::SVGModelObject))
, m_flags(flags.toRaw())
{
ASSERT(svgFlags() == flags);
}
Kind kind() const { return static_cast<Kind>(m_kind); }
OptionSet<BlockFlowFlag> blockFlowFlags() const { return OptionSet<BlockFlowFlag>::fromRaw(valueForKind(Kind::BlockFlow)); }
OptionSet<LineBreakFlag> lineBreakFlags() const { return OptionSet<LineBreakFlag>::fromRaw(valueForKind(Kind::LineBreak)); }
OptionSet<ReplacedFlag> replacedFlags() const { return OptionSet<ReplacedFlag>::fromRaw(valueForKind(Kind::Replaced)); }
OptionSet<SVGModelObjectFlag> svgFlags() const { return OptionSet<SVGModelObjectFlag>::fromRaw(valueForKind(Kind::SVGModelObject)); }
private:
uint8_t valueForKind(Kind kind) const
{
ASSERT(this->kind() == kind);
return this->kind() == kind ? m_flags : 0;
}
const uint8_t m_kind : 3 { enumToUnderlyingType(Kind::Invalid) }; // Security hardening to store the type.
const uint8_t m_flags : 6 { 0 };
// 7 bits free.
};
// Anonymous objects should pass the document as their node, and they will then automatically be
// marked as anonymous in the constructor.
RenderObject(Type, Node&, OptionSet<TypeFlag>, TypeSpecificFlags);
virtual ~RenderObject();
Type type() const { return m_type; }
Layout::Box* layoutBox() { return m_layoutBox.get(); }
const Layout::Box* layoutBox() const { return m_layoutBox.get(); }
void setLayoutBox(Layout::Box&);
void clearLayoutBox();
WEBCORE_EXPORT RenderTheme& theme() const;
virtual ASCIILiteral renderName() const = 0;
inline RenderElement* parent() const; // Defined in RenderElement.h.
inline CheckedPtr<RenderElement> checkedParent() const; // Defined in RenderElement.h.
bool isDescendantOf(const RenderObject*) const;
RenderObject* previousSibling() const { return m_previous.get(); }
RenderObject* nextSibling() const { return m_next.get(); }
RenderObject* previousInFlowSibling() const;
RenderObject* nextInFlowSibling() const;
// Use RenderElement versions instead.
virtual RenderObject* firstChildSlow() const { return nullptr; }
virtual RenderObject* lastChildSlow() const { return nullptr; }
RenderObject* nextInPreOrder() const;
RenderObject* nextInPreOrder(const RenderObject* stayWithin) const;
RenderObject* nextInPreOrderAfterChildren() const;
RenderObject* nextInPreOrderAfterChildren(const RenderObject* stayWithin) const;
RenderObject* previousInPreOrder() const;
RenderObject* previousInPreOrder(const RenderObject* stayWithin) const;
WEBCORE_EXPORT RenderObject* childAt(unsigned) const;
RenderObject* firstLeafChild() const;
RenderObject* lastLeafChild() const;
RenderElement* firstNonAnonymousAncestor() const;
#if ENABLE(TEXT_AUTOSIZING)
// Minimal distance between the block with fixed height and overflowing content and the text block to apply text autosizing.
// The greater this constant is the more potential places we have where autosizing is turned off.
// So it should be as low as possible. There are sites that break at 2.
static const int TextAutoSizingFixedHeightDepth = 3;
enum BlockContentHeightType {
FixedHeight,
FlexibleHeight,
OverflowHeight
};
typedef BlockContentHeightType (*HeightTypeTraverseNextInclusionFunction)(const RenderObject&);
RenderObject* traverseNext(const RenderObject* stayWithin, HeightTypeTraverseNextInclusionFunction, int& currentDepth, int& newFixedDepth) const;
#endif
WEBCORE_EXPORT RenderLayer* enclosingLayer() const;
WEBCORE_EXPORT CheckedPtr<RenderLayer> checkedEnclosingLayer() const;
WEBCORE_EXPORT RenderBox& enclosingBox() const;
RenderBoxModelObject& enclosingBoxModelObject() const;
RenderBox* enclosingScrollableContainer() const;
// Return our enclosing flow thread if we are contained inside one. Follows the containing block chain.
RenderFragmentedFlow* enclosingFragmentedFlow() const;
WEBCORE_EXPORT bool useDarkAppearance() const;
WEBCORE_EXPORT OptionSet<StyleColorOptions> styleColorOptions() const;
// Creates a scope where this object will assert on calls to setNeedsLayout().
class SetLayoutNeededForbiddenScope;
// RenderObject tree manipulation
//////////////////////////////////////////
virtual bool canHaveChildren() const = 0;
virtual bool canHaveGeneratedChildren() const;
virtual bool createsAnonymousWrapper() const { return false; }
//////////////////////////////////////////
#if ENABLE(TREE_DEBUGGING)
void showNodeTreeForThis() const;
void showRenderTreeForThis() const;
void showLineTreeForThis() const;
void outputRenderObject(WTF::TextStream&, bool mark, int depth) const;
void outputRenderSubTreeAndMark(WTF::TextStream&, const RenderObject* markedObject, int depth) const;
void outputRegionsInformation(WTF::TextStream&) const;
#endif
inline bool isPseudoElement() const; // Defined in RenderObjectNode.h
bool isRenderElement() const { return !isRenderText(); }
bool isRenderReplaced() const { return m_typeSpecificFlags.kind() == TypeSpecificFlags::Kind::Replaced; }
bool isRenderBoxModelObject() const { return m_typeFlags.contains(TypeFlag::IsBoxModelObject); }
bool isRenderBlock() const { return m_typeFlags.contains(TypeFlag::IsRenderBlock); }
bool isRenderBlockFlow() const { return m_typeSpecificFlags.kind() == TypeSpecificFlags::Kind::BlockFlow; }
bool isRenderInline() const { return m_typeFlags.contains(TypeFlag::IsRenderInline); }
bool isRenderLayerModelObject() const { return m_typeFlags.contains(TypeFlag::IsLayerModelObject); }
inline bool isAtomicInlineLevelBox() const; // Defined in RenderObjectStyle.h
inline bool isNonReplacedAtomicInlineLevelBox() const;
bool isRenderCounter() const { return type() == Type::Counter; }
bool isRenderQuote() const { return type() == Type::Quote; }
bool isRenderDetailsMarker() const { return type() == Type::DetailsMarker; }
bool isRenderEmbeddedObject() const { return type() == Type::EmbeddedObject; }
bool isFieldset() const;
bool isRenderFileUploadControl() const { return type() == Type::FileUploadControl; }
bool isRenderFrame() const { return type() == Type::Frame; }
bool isRenderFrameSet() const { return type() == Type::FrameSet; }
virtual bool isImage() const { return false; }
bool isRenderListBox() const { return type() == Type::ListBox; }
bool isRenderListItem() const { return type() == Type::ListItem; }
bool isRenderListMarker() const { return type() == Type::ListMarker; }
bool isRenderMedia() const { return isRenderReplaced() && m_typeSpecificFlags.replacedFlags().contains(ReplacedFlag::IsMedia); }
bool isRenderMenuList() const { return type() == Type::MenuList; }
bool isRenderMeter() const { return type() == Type::Meter; }
bool isRenderProgress() const { return type() == Type::Progress; }
bool isRenderButton() const { return type() == Type::Button; }
bool isRenderIFrame() const { return type() == Type::IFrame; }
bool isRenderImage() const { return isRenderReplaced() && m_typeSpecificFlags.replacedFlags().contains(ReplacedFlag::IsImage); }
bool isRenderTextFragment() const { return type() == Type::TextFragment; }
#if ENABLE(MODEL_ELEMENT)
bool isRenderModel() const { return type() == Type::Model; }
#endif
bool isRenderFragmentContainer() const { return isRenderBlockFlow() && m_typeSpecificFlags.blockFlowFlags().contains(BlockFlowFlag::IsFragmentContainer); }
bool isRenderReplica() const { return type() == Type::Replica; }
bool isRenderSlider() const { return type() == Type::Slider; }
bool isRenderSliderContainer() const { return type() == Type::SliderContainer; }
bool isRenderTable() const;
bool isRenderTableCell() const { return type() == Type::TableCell; }
bool isRenderTableCol() const { return type() == Type::TableCol; }
bool isRenderTableCaption() const { return type() == Type::TableCaption; }
bool isRenderTableSection() const { return type() == Type::TableSection; }
bool isRenderTextControl() const { return isRenderBlockFlow() && m_typeSpecificFlags.blockFlowFlags().contains(BlockFlowFlag::IsTextControl); }
bool isRenderTextControlMultiLine() const { return type() == Type::TextControlMultiLine; }
bool isRenderTextControlSingleLine() const { return isRenderTextControl() && !isRenderTextControlMultiLine(); }
bool isRenderSearchField() const { return type() == Type::SearchField; }
bool isRenderTextControlInnerBlock() const { return type() == Type::TextControlInnerBlock; }
bool isRenderTextControlInnerContainer() const { return type() == Type::TextControlInnerContainer; }
bool isRenderVideo() const { return type() == Type::Video; }
bool isRenderViewTransitionCapture() const { return isRenderReplaced() && m_typeSpecificFlags.replacedFlags().contains(ReplacedFlag::IsViewTransitionCapture); }
bool isRenderWidget() const { return isRenderReplaced() && m_typeSpecificFlags.replacedFlags().contains(ReplacedFlag::IsWidget); }
bool isRenderHTMLCanvas() const { return type() == Type::HTMLCanvas; }
#if ENABLE(ATTACHMENT_ELEMENT)
bool isRenderAttachment() const { return type() == Type::Attachment; }
#endif
bool isRenderGrid() const { return type() == Type::Grid; }
bool isRenderMultiColumnSet() const { return type() == Type::MultiColumnSet; }
bool isRenderMultiColumnFlow() const { return type() == Type::MultiColumnFlow; }
bool isRenderMultiColumnSpannerPlaceholder() const { return type() == Type::MultiColumnSpannerPlaceholder; }
bool isRenderScrollbarPart() const { return type() == Type::ScrollbarPart; }
bool isRenderVTTCue() const { return type() == Type::VTTCue; }
bool isViewTransitionContainingBlock() const { return isRenderBlockFlow() && m_typeSpecificFlags.blockFlowFlags().contains(BlockFlowFlag::IsViewTransitionContainingBlock); }
inline bool isDocumentElementRenderer() const; // Defined in RenderObjectInlines.h
inline bool isBody() const; // Defined in RenderObjectNode.h
inline bool isHR() const; // Defined in RenderObjectNode.h
bool isLegend() const;
bool isHTMLMarquee() const;
bool isTablePart() const { return isRenderTableCell() || isRenderTableCol() || isRenderTableCaption() || isRenderTableRow() || isRenderTableSection(); }
bool beingDestroyed() const { return m_stateBitfields.hasFlag(StateFlag::BeingDestroyed); }
bool everHadLayout() const { return m_stateBitfields.hasFlag(StateFlag::EverHadLayout); }
std::optional<bool> wasSkippedDuringLastLayoutDueToContentVisibility() const { return everHadLayout() ? std::make_optional(m_stateBitfields.hasFlag(StateFlag::WasSkippedDuringLastLayoutDueToContentVisibility)) : std::nullopt; }
static ScrollAnchoringController* searchParentChainForScrollAnchoringController(const RenderObject&);
bool childrenInline() const { return m_stateBitfields.hasFlag(StateFlag::ChildrenInline); }
virtual void setChildrenInline(bool b) { m_stateBitfields.setFlag(StateFlag::ChildrenInline, b); }
enum class FragmentedFlowState : bool {
NotInsideFlow = 0,
InsideFlow = 1,
};
enum class SkipDescendentFragmentedFlow : bool { No, Yes };
void setFragmentedFlowStateIncludingDescendants(FragmentedFlowState, SkipDescendentFragmentedFlow = SkipDescendentFragmentedFlow::Yes);
FragmentedFlowState fragmentedFlowState() const { return m_stateBitfields.fragmentedFlowState(); }
void setFragmentedFlowState(FragmentedFlowState state) { m_stateBitfields.setFragmentedFlowState(state); }
#if ENABLE(MATHML)
virtual bool isRenderMathMLBlock() const { return false; }
bool isRenderMathMLTable() const { return type() == Type::MathMLTable; }
virtual bool isRenderMathMLOperator() const { return false; }
bool isRenderMathMLRow() const;
bool isRenderMathMLMath() const { return type() == Type::MathMLMath; }
bool isRenderMathMLMenclose() const { return type() == Type::MathMLMenclose; }
bool isRenderMathMLFenced() const { return type() == Type::MathMLFenced; }
bool isRenderMathMLFencedOperator() const { return type() == Type::MathMLFencedOperator; }
bool isRenderMathMLFraction() const { return type() == Type::MathMLFraction; }
bool isRenderMathMLPadded() const { return type() == Type::MathMLPadded; }
bool isRenderMathMLRoot() const { return type() == Type::MathMLRoot; }
bool isRenderMathMLSpace() const { return type() == Type::MathMLSpace; }
virtual bool isRenderMathMLSquareRoot() const { return false; }
virtual bool isRenderMathMLScripts() const { return false; }
virtual bool isRenderMathMLToken() const { return false; }
bool isRenderMathMLUnderOver() const { return type() == Type::MathMLUnderOver; }
#endif // ENABLE(MATHML)
bool isLegacyRenderSVGModelObject() const { return m_typeSpecificFlags.kind() == TypeSpecificFlags::Kind::SVGModelObject && m_typeSpecificFlags.svgFlags().contains(SVGModelObjectFlag::IsLegacy); }
bool isRenderSVGModelObject() const { return m_typeSpecificFlags.kind() == TypeSpecificFlags::Kind::SVGModelObject && !m_typeSpecificFlags.svgFlags().contains(SVGModelObjectFlag::IsLegacy); }
bool isRenderSVGBlock() const { return isRenderBlockFlow() && m_typeSpecificFlags.blockFlowFlags().contains(BlockFlowFlag::IsSVGBlock); }
bool isLegacyRenderSVGRoot() const { return type() == Type::LegacySVGRoot; }
bool isRenderSVGRoot() const { return type() == Type::SVGRoot; }
bool isRenderSVGContainer() const { return isRenderSVGModelObject() && m_typeSpecificFlags.svgFlags().contains(SVGModelObjectFlag::IsContainer); }
bool isLegacyRenderSVGContainer() const { return isLegacyRenderSVGModelObject() && m_typeSpecificFlags.svgFlags().contains(SVGModelObjectFlag::IsContainer); }
bool isRenderSVGTransformableContainer() const { return type() == Type::SVGTransformableContainer; }
bool isLegacyRenderSVGTransformableContainer() const { return type() == Type::LegacySVGTransformableContainer; }
bool isRenderSVGViewportContainer() const { return type() == Type::SVGViewportContainer; }
bool isLegacyRenderSVGViewportContainer() const { return type() == Type::LegacySVGViewportContainer; }
bool isRenderSVGGradientStop() const { return type() == Type::SVGGradientStop; }
bool isLegacyRenderSVGHiddenContainer() const { return isLegacyRenderSVGModelObject() && m_typeSpecificFlags.svgFlags().contains(SVGModelObjectFlag::IsHiddenContainer); }
bool isRenderSVGHiddenContainer() const { return isRenderSVGModelObject() && m_typeSpecificFlags.svgFlags().contains(SVGModelObjectFlag::IsHiddenContainer); }
bool isLegacyRenderSVGPath() const { return type() == Type::LegacySVGPath; }
bool isRenderSVGPath() const { return type() == Type::SVGPath; }
bool isRenderSVGShape() const { return isRenderSVGModelObject() && m_typeSpecificFlags.svgFlags().contains(SVGModelObjectFlag::IsShape); }
bool isLegacyRenderSVGShape() const { return isLegacyRenderSVGModelObject() && m_typeSpecificFlags.svgFlags().contains(SVGModelObjectFlag::IsShape); }
bool isLegacyRenderSVGRect() const { return type() == Type::LegacySVGRect; }
bool isRenderSVGText() const { return type() == Type::SVGText; }
bool isRenderSVGTextPath() const { return type() == Type::SVGTextPath; }
bool isRenderSVGTSpan() const { return type() == Type::SVGTSpan; }
bool isRenderSVGInline() const { return type() == Type::SVGInline || type() == Type::SVGTSpan || type() == Type::SVGTextPath; }
bool isRenderSVGInlineText() const { return type() == Type::SVGInlineText; }
bool isLegacyRenderSVGImage() const { return type() == Type::LegacySVGImage; }
bool isRenderSVGImage() const { return type() == Type::SVGImage; }
bool isLegacyRenderSVGForeignObject() const { return type() == Type::LegacySVGForeignObject; }
bool isRenderSVGForeignObject() const { return type() == Type::SVGForeignObject; }
bool isLegacyRenderSVGResourceContainer() const { return isLegacyRenderSVGModelObject() && m_typeSpecificFlags.svgFlags().contains(SVGModelObjectFlag::IsResourceContainer); }
bool isRenderSVGResourceContainer() const { return isRenderSVGModelObject() && m_typeSpecificFlags.svgFlags().contains(SVGModelObjectFlag::IsResourceContainer); }
bool isLegacyRenderSVGResourceFilter() const { return type() == Type::LegacySVGResourceFilter; }
bool isRenderSVGResourceFilter() const { return type() == Type::SVGResourceFilter; }
bool isLegacyRenderSVGResourceClipper() const { return type() == Type::LegacySVGResourceClipper; }
bool isLegacyRenderSVGResourceMarker() const { return type() == Type::LegacySVGResourceMarker; }
bool isLegacyRenderSVGResourceMasker() const { return type() == Type::LegacySVGResourceMasker; }
bool isRenderSVGResourceGradient() const { return type() == Type::SVGResourceLinearGradient || type() == Type::SVGResourceRadialGradient; }
bool isRenderSVGResourcePaintServer() const { return isRenderSVGResourceFilter() || isRenderSVGResourceGradient() || isRenderSVGResourcePattern(); }
bool isRenderSVGResourcePattern() const { return type() == Type::SVGResourcePattern; }
bool isRenderSVGResourceClipper() const { return type() == Type::SVGResourceClipper; }
bool isLegacyRenderSVGResourceFilterPrimitive() const { return type() == Type::LegacySVGResourceFilterPrimitive; }
bool isRenderSVGResourceFilterPrimitive() const { return type() == Type::SVGResourceFilterPrimitive; }
bool isRenderSVGResourceLinearGradient() const { return type() == Type::SVGResourceLinearGradient; }
bool isRenderSVGResourceMarker() const { return type() == Type::SVGResourceMarker; }
bool isRenderSVGResourceMasker() const { return type() == Type::SVGResourceMasker; }
bool isRenderSVGResourceRadialGradient() const { return type() == Type::SVGResourceRadialGradient; }
bool isRenderOrLegacyRenderSVGRoot() const { return isRenderSVGRoot() || isLegacyRenderSVGRoot(); }
bool isRenderOrLegacyRenderSVGShape() const { return isRenderSVGShape() || isLegacyRenderSVGShape(); }
bool isRenderOrLegacyRenderSVGPath() const { return isRenderSVGPath() || isLegacyRenderSVGPath(); }
bool isRenderOrLegacyRenderSVGImage() const { return isRenderSVGImage() || isLegacyRenderSVGImage(); }
bool isRenderOrLegacyRenderSVGForeignObject() const { return isRenderSVGForeignObject() || isLegacyRenderSVGForeignObject(); }
bool isRenderOrLegacyRenderSVGModelObject() const { return isRenderSVGModelObject() || isLegacyRenderSVGModelObject(); }
bool isRenderOrLegacyRenderSVGResourceFilterPrimitive() const { return isRenderSVGResourceFilterPrimitive() || isLegacyRenderSVGResourceFilterPrimitive(); }
bool isSVGLayerAwareRenderer() const { return isRenderSVGRoot() || isRenderSVGModelObject() || isRenderSVGText() || isRenderSVGInline() || isRenderSVGForeignObject(); }
bool isSVGRenderer() const { return isRenderOrLegacyRenderSVGRoot() || isRenderOrLegacyRenderSVGModelObject() || isRenderSVGBlock() || isRenderSVGInline(); }
// FIXME: Those belong into a SVG specific base-class for all renderers (see above)
// Unfortunately we don't have such a class yet, because it's not possible for all renderers
// to inherit from RenderSVGObject -> RenderObject (some need RenderBlock inheritance for instance)
void invalidateCachedBoundaries();
bool usesBoundaryCaching() const;
virtual void setNeedsBoundariesUpdate();
virtual void setNeedsTransformUpdate() { }
// Per SVG 1.1 objectBoundingBox ignores clipping, masking, filter effects, opacity and stroke-width.
// This is used for all computation of objectBoundingBox relative units and by SVGLocatable::getBBox().
// NOTE: Markers are not specifically ignored here by SVG 1.1 spec, but we ignore them
// since stroke-width is ignored (and marker size can depend on stroke-width).
// objectBoundingBox is returned local coordinates.
// The name objectBoundingBox is taken from the SVG 1.1 spec.
virtual FloatRect objectBoundingBox() const;
virtual FloatRect strokeBoundingBox() const;
// The objectBoundingBox of a SVG container is affected by the transformations applied on its children -- the container
// bounding box is a union of all child bounding boxes, mapped through their transformation matrices.
//
// This method ignores all transformations and computes the objectBoundingBox, without mapping through the child
// transformation matrices. The SVG render tree is constructed in such a way, that it can be mapped to CSS equivalents:
// The SVG render tree underneath the outermost <svg> behaves as a set of absolutely positioned, possibly nested, boxes.
// They are laid out in such a way that transformations do NOT affect layout, as in HTML/CSS world, but take affect during
// painting, hit-testing etc. This allows to minimize the amount of re-layouts when animating transformations in SVG
// (not using CSS Animations/Transitions / Web Animations, but e.g. SMIL <animateTransform>, JS, ...).
virtual FloatRect objectBoundingBoxWithoutTransformations() const { return objectBoundingBox(); }
// Returns the smallest rectangle enclosing all of the painted content
// respecting clipping, masking, filters, opacity, stroke-width and markers
// This returns approximate rectangle for SVG renderers when RepaintRectCalculation::Fast is specified.
virtual FloatRect repaintRectInLocalCoordinates(RepaintRectCalculation = RepaintRectCalculation::Fast) const;
// Returns the bounding box including fill, stroke, and markers.
// This is the geometric visual extent, used for masks/gradients/clippers.
// Unlike repaintRectInLocalCoordinates, this is always accurate.
virtual FloatRect decoratedBoundingBox() const;
// This only returns the transform="" value from the element
// most callsites want localToParentTransform() instead.
virtual AffineTransform localTransform() const;
// Returns the full transform mapping from local coordinates to local coords for the parent SVG renderer
// This includes any viewport transforms and x/y offsets as well as the transform="" value off the element.
virtual const AffineTransform& localToParentTransform() const;
// SVG uses FloatPoint precise hit testing, and passes the point in parent
// coordinates instead of in repaint container coordinates. Eventually the
// rest of the rendering tree will move to a similar model.
virtual bool nodeAtFloatPoint(const HitTestRequest&, HitTestResult&, const FloatPoint& pointInParent, HitTestAction);
bool isAnonymous() const { return m_typeFlags.contains(TypeFlag::IsAnonymous); }
bool isFloating() const { return m_stateBitfields.hasFlag(StateFlag::Floating); }
bool isPositioned() const { return m_stateBitfields.isPositioned(); }
bool isInFlowPositioned() const { return m_stateBitfields.isRelativelyPositioned() || m_stateBitfields.isStickilyPositioned(); }
bool isOutOfFlowPositioned() const { return m_stateBitfields.isOutOfFlowPositioned(); } // absolute or fixed positioning
bool isRelativelyPositioned() const { return m_stateBitfields.isRelativelyPositioned(); }
bool isStickilyPositioned() const { return m_stateBitfields.isStickilyPositioned(); }
bool isRenderText() const { return m_typeFlags.contains(TypeFlag::IsText); }
bool isRenderLineBreak() const { return type() == Type::LineBreak; }
bool isBR() const { return isRenderLineBreak() && !hasWBRLineBreakFlag(); }
bool isWBR() const { return isRenderLineBreak() && hasWBRLineBreakFlag(); }
bool isLineBreakOpportunity() const { return isRenderLineBreak() && isWBR(); }
bool isRenderTextOrLineBreak() const { return isRenderText() || isRenderLineBreak(); }
bool isRenderBox() const { return m_typeFlags.contains(TypeFlag::IsBox); }
bool isRenderTableRow() const { return type() == Type::TableRow; }
bool isRenderView() const { return type() == Type::View; }
bool isInline() const { return !m_stateBitfields.hasFlag(StateFlag::IsBlock); } // inline object
bool isBlockLevelReplacedOrAtomicInline() const { return m_stateBitfields.hasFlag(StateFlag::IsBlockLevelReplacedOrAtomicInline); }
bool isHorizontalWritingMode() const { return !m_stateBitfields.hasFlag(StateFlag::VerticalWritingMode); }
bool hasReflection() const { return hasRareData() && rareData().hasReflection; }
bool isRenderFragmentedFlow() const { return isRenderBlockFlow() && m_typeSpecificFlags.blockFlowFlags().contains(BlockFlowFlag::IsFragmentedFlow); }
bool hasOutlineAutoAncestor() const { return hasRareData() && rareData().hasOutlineAutoAncestor; }
bool paintContainmentApplies() const { return m_stateBitfields.hasFlag(StateFlag::PaintContainmentApplies); }
bool hasSVGTransform() const { return m_stateBitfields.hasFlag(StateFlag::HasSVGTransform); }
bool isExcludedFromNormalLayout() const { return m_stateBitfields.hasFlag(StateFlag::IsExcludedFromNormalLayout); }
void setIsExcludedFromNormalLayout(bool excluded) { m_stateBitfields.setFlag(StateFlag::IsExcludedFromNormalLayout, excluded); }
bool isExcludedAndPlacedInBorder() const { return isExcludedFromNormalLayout() && isLegend(); }
bool isYouTubeReplacement() const { return hasRareData() && rareData().isYouTubeReplacement; }
void markIsYouTubeReplacement();
bool hasLayer() const { return m_stateBitfields.hasFlag(StateFlag::HasLayer); }
enum class BoxDecorationState : uint8_t {
None,
InvalidObscurationStatus,
IsKnownToBeObscured,
MayBeVisible,
};
bool hasVisibleBoxDecorations() const { return boxDecorationState() != BoxDecorationState::None; }
bool needsLayout() const;
bool needsPreferredLogicalWidthsUpdate() const { return m_stateBitfields.hasFlag(StateFlag::PreferredLogicalWidthsNeedUpdate); }
bool selfNeedsLayout() const { return m_stateBitfields.hasFlag(StateFlag::NeedsLayout); }
bool needsOutOfFlowMovementLayout() const { return m_stateBitfields.hasFlag(StateFlag::NeedsOutOfFlowMovementLayout); }
bool needsOutOfFlowMovementLayoutOnly() const;
bool outOfFlowChildNeedsLayout() const { return m_stateBitfields.hasFlag(StateFlag::OutOfFlowChildNeedsLayout); }
bool needsSimplifiedNormalFlowLayout() const { return m_stateBitfields.hasFlag(StateFlag::NeedsSimplifiedNormalFlowLayout); }
bool needsSimplifiedNormalFlowLayoutOnly() const;
bool needsNormalChildOrSimplifiedLayoutOnly() const;
bool normalChildNeedsLayout() const { return m_stateBitfields.hasFlag(StateFlag::NormalChildNeedsLayout); }
bool outOfFlowChildNeedsStaticPositionLayout() const { return m_stateBitfields.hasFlag(StateFlag::OutOfFlowChildNeedsStaticPositionLayout); }
bool isSelectionBorder() const;
bool hasNonVisibleOverflow() const { return m_stateBitfields.hasFlag(StateFlag::HasNonVisibleOverflow); }
bool hasTransformRelatedProperty() const { return m_stateBitfields.hasFlag(StateFlag::HasTransformRelatedProperty); } // Transform, perspective or transform-style: preserve-3d.
inline bool isTransformed() const; // Defined in RenderObjectStyle.h
inline bool hasTransformOrPerspective() const; // Defined in RenderObjectStyle.h
bool capturedInViewTransition() const { return m_stateBitfields.hasFlag(StateFlag::CapturedInViewTransition); }
bool setCapturedInViewTransition(bool);
// When the document element is captured, the captured contents uses the RenderView
// instead. Returns the capture state with this adjustment applied.
bool effectiveCapturedInViewTransition() const;
inline RenderView& view() const; // Defined in RenderObjectDocument.h
CheckedRef<RenderView> checkedView() const;
inline LocalFrameViewLayoutContext& layoutContext() const;
HostWindow* hostWindow() const;
// Returns true if this renderer is rooted.
bool isRooted() const;
inline Node* node() const; // Defined in RenderObjectNode.h
inline RefPtr<Node> protectedNode() const; // Defined in RenderObjectNode.h
inline Node* nonPseudoNode() const; // Defined in RenderObjectNode.h
inline Document& document() const; // Defined in RenderObjectDocument.h
inline Ref<Document> protectedDocument() const; // Defined in RenderObjectDocument.h
inline TreeScope& treeScopeForSVGReferences() const; // Defined in RenderObjectInlines.h
inline Ref<TreeScope> protectedTreeScopeForSVGReferences() const; // Defined in RenderObjectInlines.h
inline LocalFrame& frame() const; // Defined in RenderObjectInlines.h
inline Ref<LocalFrame> protectedFrame() const; // Defined in RenderObjectInlines.h
inline Page& page() const; // Defined in RenderObjectInlines.h
inline Ref<Page> protectedPage() const; // Defined in RenderObjectInlines.h
inline Settings& settings() const; // Defined in RenderObjectInlines.h
// Returns the object containing this one. Can be different from parent for positioned elements.
// If repaintContainer and repaintContainerSkipped are not null, on return *repaintContainerSkipped
// is true if the renderer returned is an ancestor of repaintContainer.
RenderElement* container() const;
RenderElement* container(const RenderLayerModelObject* repaintContainer, bool& repaintContainerSkipped) const;
RenderElement* markContainingBlocksForLayout(RenderElement* layoutRoot = nullptr);
inline void setNeedsLayout(MarkingBehavior = MarkContainingBlockChain);
enum class HadSkippedLayout { No, Yes };
void clearNeedsLayout(HadSkippedLayout = HadSkippedLayout::No);
void setNeedsPreferredWidthsUpdate(MarkingBehavior = MarkContainingBlockChain);
void clearNeedsPreferredWidthsUpdate() { m_stateBitfields.setFlag(StateFlag::PreferredLogicalWidthsNeedUpdate, { }); }
inline void setNeedsLayoutAndPreferredWidthsUpdate();
void setPositionState(PositionType);
void clearPositionedState() { m_stateBitfields.clearPositionedState(); }
void setFloating(bool b = true) { m_stateBitfields.setFlag(StateFlag::Floating, b); }
void setInline(bool b) { m_stateBitfields.setFlag(StateFlag::IsBlock, !b); }
void setHasVisibleBoxDecorations(bool = true);
void invalidateBackgroundObscurationStatus();
void setBlockLevelReplacedOrAtomicInline(bool b = true) { m_stateBitfields.setFlag(StateFlag::IsBlockLevelReplacedOrAtomicInline, b); }
void setHorizontalWritingMode(bool b = true) { m_stateBitfields.setFlag(StateFlag::VerticalWritingMode, !b); }
void setHasNonVisibleOverflow(bool b = true) { m_stateBitfields.setFlag(StateFlag::HasNonVisibleOverflow, b); }
void setHasLayer(bool b = true) { m_stateBitfields.setFlag(StateFlag::HasLayer, b); }
void setHasTransformRelatedProperty(bool b = true) { m_stateBitfields.setFlag(StateFlag::HasTransformRelatedProperty, b); }
void setHasReflection(bool = true);
void setHasOutlineAutoAncestor(bool = true);
void setPaintContainmentApplies(bool value = true) { m_stateBitfields.setFlag(StateFlag::PaintContainmentApplies, value); }
void setHasSVGTransform(bool value = true) { m_stateBitfields.setFlag(StateFlag::HasSVGTransform, value); }
// used for element state updates that cannot be fixed with a
// repaint and do not need a relayout
virtual void updateFromElement() { }
bool isComposited() const;
bool hitTest(const HitTestRequest&, HitTestResult&, const HitTestLocation& locationInContainer, const LayoutPoint& accumulatedOffset, HitTestFilter = HitTestAll);
virtual Node* nodeForHitTest() const;
RefPtr<Node> protectedNodeForHitTest() const;
virtual void updateHitTestResult(HitTestResult&, const LayoutPoint&) const;
virtual bool nodeAtPoint(const HitTestRequest&, HitTestResult&, const HitTestLocation& locationInContainer, const LayoutPoint& accumulatedOffset, HitTestAction);
virtual Position positionForPoint(const LayoutPoint&, HitTestSource);
virtual PositionWithAffinity positionForPoint(const LayoutPoint&, HitTestSource, const RenderFragmentContainer*);
PositionWithAffinity createPositionWithAffinity(int offset, Affinity) const;
PositionWithAffinity createPositionWithAffinity(const Position&) const;
WEBCORE_EXPORT VisiblePosition visiblePositionForPoint(const LayoutPoint&, HitTestSource);
// Returns the containing block level element for this element.
WEBCORE_EXPORT RenderBlock* containingBlock() const;
CheckedPtr<RenderBlock> checkedContainingBlock() const;
static RenderBlock* containingBlockForPositionType(PositionType, const RenderObject&);
// Convert the given local point to absolute coordinates. If OptionSet<MapCoordinatesMode> includes UseTransforms, take transforms into account.
WEBCORE_EXPORT FloatPoint localToAbsolute(const FloatPoint& localPoint = FloatPoint(), OptionSet<MapCoordinatesMode> = { }, bool* wasFixed = nullptr) const;
TransformState viewTransitionTransform() const;
FloatPoint absoluteToLocal(const DoublePoint&, OptionSet<MapCoordinatesMode> = { }) const;
// Convert a local quad to absolute coordinates, taking transforms into account.
inline FloatQuad localToAbsoluteQuad(const FloatQuad&, OptionSet<MapCoordinatesMode> = UseTransforms, bool* wasFixed = nullptr) const; // Defined in RenderObjectInlines.h
// Convert an absolute quad to local coordinates.
FloatQuad absoluteToLocalQuad(const FloatQuad&, OptionSet<MapCoordinatesMode> = UseTransforms) const;
// Convert a local quad into the coordinate system of container, taking transforms into account.
WEBCORE_EXPORT FloatQuad localToContainerQuad(const FloatQuad&, const RenderLayerModelObject* container, OptionSet<MapCoordinatesMode> = UseTransforms, bool* wasFixed = nullptr) const;
WEBCORE_EXPORT FloatPoint localToContainerPoint(const FloatPoint&, const RenderLayerModelObject* container, OptionSet<MapCoordinatesMode> = UseTransforms, bool* wasFixed = nullptr) const;
// Return the offset from the container() renderer (excluding transforms). In multi-column layout,
// different offsets apply at different points, so return the offset that applies to the given point.
virtual LayoutSize offsetFromContainer(const RenderElement&, const LayoutPoint&, bool* offsetDependsOnPoint = nullptr) const;
// Return the offset from an object up the container() chain. Asserts that none of the intermediate objects have transforms.
LayoutSize offsetFromAncestorContainer(const RenderElement&) const;
virtual void collectSelectionGeometries(Vector<SelectionGeometry>&, unsigned startOffset = 0, unsigned endOffset = std::numeric_limits<unsigned>::max());
virtual void absoluteQuadsForSelection(Vector<FloatQuad>& quads) const { absoluteQuads(quads); }
struct SelectionGeometries {
Vector<SelectionGeometry> geometries;
Vector<PlatformLayerIdentifier> intersectingLayerIDs;
};
WEBCORE_EXPORT static SelectionGeometries collectSelectionGeometries(const SimpleRange&);
WEBCORE_EXPORT static Vector<SelectionGeometry> collectSelectionGeometriesWithoutUnionInteriorLines(const SimpleRange&);
virtual void boundingRects(Vector<LayoutRect>&, const LayoutPoint& /* offsetFromRoot */) const { }
WEBCORE_EXPORT IntRect absoluteBoundingBoxRect(bool useTransform = true, bool* wasFixed = nullptr) const;
IntRect absoluteBoundingBoxRectIgnoringTransforms() const { return absoluteBoundingBoxRect(false); }
// Build an array of quads in absolute coords for line boxes
virtual void absoluteQuads(Vector<FloatQuad>&, bool* /*wasFixed*/ = nullptr) const { }
virtual void absoluteFocusRingQuads(Vector<FloatQuad>&);
enum class BoundingRectBehavior : uint8_t {
RespectClipping = 1 << 0,
UseVisibleBounds = 1 << 1,
IgnoreTinyRects = 1 << 2,
IgnoreEmptyTextSelections = 1 << 3,
UseSelectionHeight = 1 << 4,
ComputeIndividualCharacterRects = 1 << 5,
};
WEBCORE_EXPORT static Vector<FloatQuad> absoluteTextQuads(const SimpleRange&, OptionSet<BoundingRectBehavior> = { });
WEBCORE_EXPORT static Vector<IntRect> absoluteTextRects(const SimpleRange&, OptionSet<BoundingRectBehavior> = { });
WEBCORE_EXPORT static Vector<FloatRect> absoluteBorderAndTextRects(const SimpleRange&, OptionSet<BoundingRectBehavior> = { });
static Vector<FloatRect> clientBorderAndTextRects(const SimpleRange&);
// the rect that will be painted if this object is passed as the paintingRoot
WEBCORE_EXPORT LayoutRect paintingRootRect(LayoutRect& topLevelRect);
inline const RenderStyle& style() const; // Defined in RenderObjectStyle.h.
inline CheckedRef<const RenderStyle> checkedStyle() const; // Defined in RenderObjectStyle.h.
inline const RenderStyle& firstLineStyle() const;
inline WritingMode writingMode() const; // Defined in RenderObjectStyle.h.
// writingMode().isHorizontal() is cached by isHorizontalWritingMode() above.
// Anonymous blocks that are part of of a continuation chain will return their inline continuation's outline style instead.
// This is typically only relevant when repainting.
virtual const RenderStyle& outlineStyleForRepaint() const;
virtual CursorDirective getCursor(const LayoutPoint&, Cursor&) const;
// Return the RenderLayerModelObject in the container chain which is responsible for painting this object, or nullptr
// if painting is root-relative. This is the container that should be passed to the 'forRepaint' functions.
struct RepaintContainerStatus {
bool fullRepaintIsScheduled { false }; // Either the repaint container or a layer in-between has already been scheduled for full repaint.
CheckedPtr<const RenderLayerModelObject> renderer { nullptr };
};
RepaintContainerStatus containerForRepaint() const;
// Actually do the repaint of rect r for this object which has been computed in the coordinate space
// of repaintContainer. If repaintContainer is nullptr, repaint via the view.
void repaintUsingContainer(SingleThreadWeakPtr<const RenderLayerModelObject>&& repaintContainer, const LayoutRect&, bool shouldClipToLayer = true) const;
// Repaint the entire object. Called when, e.g., the color of a border changes, or when a border
// style changes.
enum class ForceRepaint : bool { No, Yes };
void repaint(ForceRepaint = ForceRepaint::No) const;
// Repaint a specific subrectangle within a given object. The rect |r| is in the object's coordinate space.
WEBCORE_EXPORT void repaintRectangle(const LayoutRect&, bool shouldClipToLayer = true) const;
enum class ClipRepaintToLayer : bool { No, Yes };
void repaintRectangle(const LayoutRect&, ClipRepaintToLayer, ForceRepaint, std::optional<LayoutBoxExtent> additionalRepaintOutsets = std::nullopt) const;
// Repaint a slow repaint object, which, at this time, means we are repainting an object with background-attachment:fixed.
void repaintSlowRepaintObject() const;
struct RepaintRects {
LayoutRect clippedOverflowRect; // Some rect (normally the visual overflow rect) mapped up to the repaint container, respecting clipping.
std::optional<LayoutRect> outlineBoundsRect; // A rect representing the extent of outlines and shadows, mapped to the repaint container, but not clipped.
RepaintRects(LayoutRect rect = { }, const std::optional<LayoutRect>& outlineBounds = { })
: clippedOverflowRect(rect)
, outlineBoundsRect(outlineBounds)
{ }
bool operator==(const RepaintRects&) const = default;
void move(LayoutSize size)
{
clippedOverflowRect.move(size);
if (outlineBoundsRect)
outlineBoundsRect->move(size);
}
void moveBy(LayoutPoint size)
{
clippedOverflowRect.moveBy(size);
if (outlineBoundsRect)
outlineBoundsRect->moveBy(size);
}
void expand(LayoutSize size)
{
clippedOverflowRect.expand(size);
if (outlineBoundsRect)
outlineBoundsRect->expand(size);
}
void encloseToIntRects()
{
clippedOverflowRect = enclosingIntRect(clippedOverflowRect);
if (outlineBoundsRect)
*outlineBoundsRect = enclosingIntRect(*outlineBoundsRect);
}
void unite(const RepaintRects& other)
{
clippedOverflowRect.unite(other.clippedOverflowRect);
if (outlineBoundsRect && other.outlineBoundsRect)
outlineBoundsRect->unite(*other.outlineBoundsRect);
}
void flipForWritingMode(LayoutSize containerSize, bool isHorizontalWritingMode)
{
if (isHorizontalWritingMode) {
clippedOverflowRect.setY(containerSize.height() - clippedOverflowRect.maxY());
if (outlineBoundsRect)
outlineBoundsRect->setY(containerSize.height() - outlineBoundsRect->maxY());
} else {
clippedOverflowRect.setX(containerSize.width() - clippedOverflowRect.maxX());
if (outlineBoundsRect)
outlineBoundsRect->setX(containerSize.width() - outlineBoundsRect->maxX());
}
}
// Returns true if intersecting (clippedOverflowRect remains non-empty).
bool intersect(LayoutRect clipRect)
{
// Note the we only intersect clippedOverflowRect.
clippedOverflowRect.intersect(clipRect);
return !clippedOverflowRect.isEmpty();
}
// Returns true if intersecting (clippedOverflowRect remains non-empty).
bool edgeInclusiveIntersect(LayoutRect clipRect)
{
// Note the we only intersect clippedOverflowRect.
return clippedOverflowRect.edgeInclusiveIntersect(clipRect);
}
void transform(const TransformationMatrix&);
void transform(const TransformationMatrix&, float deviceScaleFactor);
};
// Returns the rect that should be repainted whenever this object changes. The rect is in the view's
// coordinate space. This method deals with outlines and overflow.
inline LayoutRect absoluteClippedOverflowRectForRepaint() const;
inline LayoutRect absoluteClippedOverflowRectForSpatialNavigation() const;
inline LayoutRect absoluteClippedOverflowRectForRenderTreeAsText() const;
WEBCORE_EXPORT IntRect pixelSnappedAbsoluteClippedOverflowRect() const;
virtual LayoutRect clippedOverflowRect(const RenderLayerModelObject* repaintContainer, VisibleRectContext) const;
inline LayoutRect clippedOverflowRectForRepaint(const RenderLayerModelObject* repaintContainer) const;
virtual LayoutRect rectWithOutlineForRepaint(const RenderLayerModelObject* repaintContainer, LayoutUnit outlineWidth) const;
virtual LayoutRect outlineBoundsForRepaint(const RenderLayerModelObject* /*repaintContainer*/, const RenderGeometryMap* = nullptr) const { return { }; }
// Given a rect in the object's coordinate space, compute a rect in the coordinate space
// of repaintContainer suitable for the given VisibleRectContext.
RepaintRects computeRects(const RepaintRects&, const RenderLayerModelObject* repaintContainer, VisibleRectContext) const;
inline LayoutRect computeRectForRepaint(const LayoutRect& rect, const RenderLayerModelObject* repaintContainer) const;
FloatRect computeFloatRectForRepaint(const FloatRect&, const RenderLayerModelObject* repaintContainer) const;
virtual RepaintRects rectsForRepaintingAfterLayout(const RenderLayerModelObject* repaintContainer, RepaintOutlineBounds) const;
// Given a rect in the object's coordinate space, compute the location in container space where this rect is visible,
// when clipping and scrolling as specified by the context. When using edge-inclusive intersection, return std::nullopt
// rather than an empty rect if the rect is completely clipped out in container space.
virtual std::optional<RepaintRects> computeVisibleRectsInContainer(const RepaintRects&, const RenderLayerModelObject* repaintContainer, VisibleRectContext) const;
virtual std::optional<FloatRect> computeFloatVisibleRectInContainer(const FloatRect&, const RenderLayerModelObject* repaintContainer, VisibleRectContext) const;
WEBCORE_EXPORT bool hasEmptyVisibleRectRespectingParentFrames() const;
bool isFloatingOrOutOfFlowPositioned() const { return (isFloating() || isOutOfFlowPositioned()); }
bool isInFlow() const { return !isFloatingOrOutOfFlowPositioned(); }
enum class HighlightState : uint8_t {
None, // The object is not selected.
Start, // The object either contains the start of a selection run or is the start of a run
Inside, // The object is fully encompassed by a selection run
End, // The object either contains the end of a selection run or is the end of a run
Both // The object contains an entire run or is the sole selected object in that run
};
// The current selection state for an object. For blocks, the state refers to the state of the leaf
// descendants (as described above in the HighlightState enum declaration).
HighlightState selectionState() const { return m_stateBitfields.selectionState(); }
virtual void setSelectionState(HighlightState);
inline void setSelectionStateIfNeeded(HighlightState);
bool canUpdateSelectionOnRootLineBoxes();
// A single rectangle that encompasses all of the selected objects within this object. Used to determine the tightest
// possible bounding box for the selection. The rect returned is in the coordinate space of the paint invalidation container's backing.
virtual LayoutRect selectionRectForRepaint(const RenderLayerModelObject* /*repaintContainer*/, bool /*clipToVisibleContent*/ = true) { return LayoutRect(); }
virtual bool canBeSelectionLeaf() const { return false; }
// Whether or not a given block needs to paint selection gaps.
virtual bool shouldPaintSelectionGaps() const { return false; }
// When performing a global document tear-down, or when going into the back/forward cache, the renderer of the document is cleared.
bool renderTreeBeingDestroyed() const; // Defined in RenderObjectInlines.h
void destroy();
// Virtual function helpers for the deprecated Flexible Box Layout (display: -webkit-box).
bool isRenderDeprecatedFlexibleBox() const { return m_type == RenderObject::Type::DeprecatedFlexibleBox; }
// Virtual function helper for the new FlexibleBox Layout (display: -webkit-flex).
inline bool isRenderFlexibleBox() const { return m_typeFlags.contains(TypeFlag::IsFlexibleBox); }
inline bool isFlexibleBoxIncludingDeprecated() const { return isRenderFlexibleBox() || isRenderDeprecatedFlexibleBox(); }
bool isRenderCombineText() const { return type() == Type::CombineText; }
virtual int caretMinOffset() const;
virtual int caretMaxOffset() const;
virtual int previousOffset(int current) const;
virtual int previousOffsetForBackwardDeletion(int current) const;
virtual int nextOffset(int current) const;
// CachedImageClient emulation.
virtual void notifyFinished(CachedResource&, const NetworkLoadMetrics&, LoadWillContinueInAnotherProcess) { }
virtual void imageChanged(WrappedImagePtr, const IntRect* = nullptr) { }
virtual bool allowsAnimation() const { return true; }
virtual bool canDestroyDecodedData() const { return true; }
virtual bool useSystemDarkAppearance() const { return false; }
virtual VisibleInViewportState imageFrameAvailable(CachedImage&, ImageAnimatingState, const IntRect*);
virtual VisibleInViewportState imageVisibleInViewport(const Document&) const { return VisibleInViewportState::No; }
virtual void didRemoveCachedImageClient(CachedImage&) { }
virtual void imageContentChanged(CachedImage&) { }
virtual void scheduleRenderingUpdateForImage(CachedImage&) { }
CachedImageClient& cachedImageClient() const;
Ref<CachedImageClient> protectedCachedImageClient() const { return cachedImageClient(); }
// Map points and quads through elements, potentially via 3d transforms. You should never need to call these directly; use
// localToAbsolute/absoluteToLocal methods instead.
virtual void mapLocalToContainer(const RenderLayerModelObject* repaintContainer, TransformState&, OptionSet<MapCoordinatesMode>, bool* wasFixed = nullptr) const;
virtual void mapAbsoluteToLocalPoint(OptionSet<MapCoordinatesMode>, TransformState&) const;
bool shouldUseTransformFromContainer(const RenderElement* container) const;
void getTransformFromContainer(const LayoutSize& offsetInContainer, TransformationMatrix&) const;
void pushOntoTransformState(TransformState&, OptionSet<MapCoordinatesMode>, const RenderLayerModelObject* repaintContainer, const RenderElement* container, const LayoutSize& offsetInContainer, bool containerSkipped) const;
bool participatesInPreserve3D() const;
LayoutRect absoluteOutlineBounds() const { return outlineBoundsForRepaint(nullptr); }
// FIXME: Renderers should not need to be notified about internal reparenting (webkit.org/b/224143).
virtual void insertedIntoTree();
virtual void willBeRemovedFromTree();
void resetFragmentedFlowStateOnRemoval();
void initializeFragmentedFlowStateOnInsertion();
virtual String description() const;
virtual String debugDescription() const;
bool isSkippedContent() const;
PointerEvents usedPointerEvents() const;
protected:
//////////////////////////////////////////
// Helper functions. Dangerous to use!
void setPreviousSibling(RenderObject* previous) { m_previous = previous; }
void setNextSibling(RenderObject* next) { m_next = next; }
void setParent(RenderElement*);
//////////////////////////////////////////
inline Node& nodeForNonAnonymous() const; // Defined in RenderObjectInlines.h
virtual void willBeDestroyed();
void setIsBeingDestroyed() { m_stateBitfields.setFlag(StateFlag::BeingDestroyed); }
void scheduleLayout(RenderElement* layoutRoot);
void setNeedsOutOfFlowMovementLayoutBit(bool b) { m_stateBitfields.setFlag(StateFlag::NeedsOutOfFlowMovementLayout, b); }
void setNormalChildNeedsLayoutBit(bool b) { m_stateBitfields.setFlag(StateFlag::NormalChildNeedsLayout, b); }
void setOutOfFlowChildNeedsLayoutBit(bool b) { m_stateBitfields.setFlag(StateFlag::OutOfFlowChildNeedsLayout, b); }
void setNeedsSimplifiedNormalFlowLayoutBit(bool b) { m_stateBitfields.setFlag(StateFlag::NeedsSimplifiedNormalFlowLayout, b); }
void setOutOfFlowChildNeedsStaticPositionLayoutBit(bool b) { m_stateBitfields.setFlag(StateFlag::OutOfFlowChildNeedsStaticPositionLayout, b); }
virtual RenderFragmentedFlow* locateEnclosingFragmentedFlow() const;
static FragmentedFlowState computedFragmentedFlowState(const RenderObject&);
static VisibleRectContext visibleRectContextForRepaint();
static VisibleRectContext visibleRectContextForSpatialNavigation();
static VisibleRectContext visibleRectContextForRenderTreeAsText();
bool isSetNeedsLayoutForbidden() const;
void issueRepaint(std::optional<LayoutRect> partialRepaintRect = std::nullopt, ClipRepaintToLayer = ClipRepaintToLayer::No, ForceRepaint = ForceRepaint::No, std::optional<LayoutBoxExtent> additionalRepaintOutsets = std::nullopt) const;
bool hasWBRLineBreakFlag() const { return m_typeSpecificFlags.lineBreakFlags().contains(LineBreakFlag::IsWBR); }
void setBoxDecorationState(BoxDecorationState boxDecorationState) { m_stateBitfields.setBoxDecorationState(boxDecorationState); }
BoxDecorationState boxDecorationState() const { return m_stateBitfields.boxDecorationState(); }
private:
// This class is to avoid making RenderObject refcounted.
class CachedImageListener final : public CachedImageClient, public RefCounted<CachedImageListener> {
WTF_MAKE_TZONE_ALLOCATED(CachedImageListener);
public:
static Ref<CachedImageListener> create(RenderObject&);
// CachedImageClient.
void ref() const final { RefCounted::ref(); }
void deref() const final { RefCounted::deref(); }
private:
// CachedResourceClient.
void notifyFinished(CachedResource&, const NetworkLoadMetrics&, LoadWillContinueInAnotherProcess) final;
// CachedImageClient.
void imageChanged(CachedImage*, const IntRect* = nullptr) final;
bool allowsAnimation() const final;
bool canDestroyDecodedData() const final;
bool useSystemDarkAppearance() const final;
VisibleInViewportState imageFrameAvailable(CachedImage&, ImageAnimatingState, const IntRect*) final;
VisibleInViewportState imageVisibleInViewport(const Document&) const final;
void didRemoveCachedImageClient(CachedImage&) final;
void imageContentChanged(CachedImage&) final;
void scheduleRenderingUpdateForImage(CachedImage&) final;
explicit CachedImageListener(RenderObject&);
SingleThreadWeakPtr<RenderObject> m_renderer;
};
virtual RepaintRects localRectsForRepaint(RepaintOutlineBounds) const;
void addAbsoluteRectForLayer(LayoutRect& result);
void setLayerNeedsFullRepaint();
void setLayerNeedsFullRepaintForOutOfFlowMovementLayout();
void invalidateContainerPreferredLogicalWidths();
struct SelectionGeometriesInternal {
Vector<SelectionGeometry> geometries;
int maxLineNumber { 0 };
bool hasBidirectionalText { false };
Vector<PlatformLayerIdentifier> intersectingLayerIDs;
};
static SelectionGeometriesInternal collectSelectionGeometriesInternal(const SimpleRange&);
void propagateRepaintToParentWithOutlineAutoIfNeeded(const RenderLayerModelObject& repaintContainer, const LayoutRect& repaintRect) const;
void setEverHadLayout() { m_stateBitfields.setFlag(StateFlag::EverHadLayout); }
void setHadSkippedLayout(bool b) { m_stateBitfields.setFlag(StateFlag::WasSkippedDuringLastLayoutDueToContentVisibility, b); }
bool hasRareData() const { return m_stateBitfields.hasFlag(StateFlag::HasRareData); }
#if ASSERT_ENABLED
void setNeedsLayoutIsForbidden(bool flag) const { m_setNeedsLayoutForbidden = flag; }
mutable bool m_setNeedsLayoutForbidden : 1;
#endif
enum class StateFlag : uint32_t {
IsBlock = 1 << 0,
IsBlockLevelReplacedOrAtomicInline = 1 << 1,
BeingDestroyed = 1 << 2,
NeedsLayout = 1 << 3,
NeedsOutOfFlowMovementLayout = 1 << 4,
NormalChildNeedsLayout = 1 << 5,
OutOfFlowChildNeedsLayout = 1 << 6,
NeedsSimplifiedNormalFlowLayout = 1 << 7,
OutOfFlowChildNeedsStaticPositionLayout = 1 << 8,
EverHadLayout = 1 << 9,
IsExcludedFromNormalLayout = 1 << 10,
Floating = 1 << 11,
VerticalWritingMode = 1 << 12,
PreferredLogicalWidthsNeedUpdate = 1 << 13,
HasRareData = 1 << 14,
HasLayer = 1 << 15,
HasNonVisibleOverflow = 1 << 16,
HasTransformRelatedProperty = 1 << 17,
ChildrenInline = 1 << 18,
PaintContainmentApplies = 1 << 19,
HasSVGTransform = 1 << 20,
WasSkippedDuringLastLayoutDueToContentVisibility = 1 << 21,
CapturedInViewTransition = 1 << 22
};
class StateBitfields {
enum PositionedState {
IsStaticallyPositioned = 0,
IsRelativelyPositioned = 1,
IsOutOfFlowPositioned = 2,
IsStickilyPositioned = 3
};
private:
uint32_t m_flags : 23 { 0 };
uint32_t m_positionedState : 2 { IsStaticallyPositioned }; // PositionedState
uint32_t m_selectionState : 3 { enumToUnderlyingType(HighlightState::None) }; // HighlightState
uint32_t m_fragmentedFlowState : 1 { enumToUnderlyingType(FragmentedFlowState::NotInsideFlow) }; // FragmentedFlowState
uint32_t m_boxDecorationState : 2 { enumToUnderlyingType(BoxDecorationState::None) }; // BoxDecorationState
// 1 bit free
public:
OptionSet<StateFlag> flags() const { return OptionSet<StateFlag>::fromRaw(m_flags); }
bool hasFlag(StateFlag flag) const { return flags().contains(flag); }
void setFlag(StateFlag flag, bool value = true)
{
auto newFlags = flags();
newFlags.set(flag, value);
m_flags = newFlags.toRaw();
ASSERT(flags() == newFlags);
}
void clearFlag(StateFlag flag) { setFlag(flag, false); }
bool isOutOfFlowPositioned() const { return m_positionedState == IsOutOfFlowPositioned; }
bool isRelativelyPositioned() const { return m_positionedState == IsRelativelyPositioned; }
bool isStickilyPositioned() const { return m_positionedState == IsStickilyPositioned; }
bool isPositioned() const { return m_positionedState != IsStaticallyPositioned; }
void setPositionedState(PositionType positionState)
{
// This mask maps PositionType::Fixed and PositionType::Absolute to IsOutOfFlowPositioned, saving one bit.
m_positionedState = static_cast<PositionedState>(static_cast<uint32_t>(positionState) & 0x3);
}
void clearPositionedState() { m_positionedState = static_cast<unsigned>(PositionType::Static); }
ALWAYS_INLINE HighlightState selectionState() const { return static_cast<HighlightState>(m_selectionState); }
ALWAYS_INLINE void setSelectionState(HighlightState selectionState) { m_selectionState = static_cast<uint32_t>(selectionState); }
ALWAYS_INLINE FragmentedFlowState fragmentedFlowState() const { return static_cast<FragmentedFlowState>(m_fragmentedFlowState); }
ALWAYS_INLINE void setFragmentedFlowState(FragmentedFlowState fragmentedFlowState) { m_fragmentedFlowState = static_cast<uint32_t>(fragmentedFlowState); }
ALWAYS_INLINE BoxDecorationState boxDecorationState() const { return static_cast<BoxDecorationState>(m_boxDecorationState); }
ALWAYS_INLINE void setBoxDecorationState(BoxDecorationState boxDecorationState) { m_boxDecorationState = static_cast<uint32_t>(boxDecorationState); }
};
StateBitfields m_stateBitfields;
WeakRef<Node, WeakPtrImplWithEventTargetData> m_node;
SingleThreadWeakPtr<RenderElement> m_parent;
SingleThreadPackedWeakPtr<RenderObject> m_previous;
const OptionSet<TypeFlag> m_typeFlags;
const Type m_type;
SingleThreadPackedWeakPtr<RenderObject> m_next;
const TypeSpecificFlags m_typeSpecificFlags;
CheckedPtr<Layout::Box> m_layoutBox;
const RefPtr<CachedImageListener> m_cachedImageClient;
// FIXME: This should be RenderElementRareData.
class RenderObjectRareData {
WTF_MAKE_PREFERABLY_COMPACT_TZONE_ALLOCATED(RenderObjectRareData);
public:
RenderObjectRareData();
~RenderObjectRareData();
bool hasReflection { false };
bool hasOutlineAutoAncestor { false };
// Dirty bit was set with MarkingBehavior::MarkOnlyThis
bool preferredLogicalWidthsNeedUpdateIsMarkOnlyThis { false };
bool isYouTubeReplacement { false };
EnumSet<Style::MarginTrimSide> trimmedMargins;
// From RenderElement
std::unique_ptr<ReferencedSVGResources> referencedSVGResources;
SingleThreadWeakPtr<RenderBlockFlow> backdropRenderer;
// From RenderBox
RefPtr<ControlPart> controlPart;
};
WEBCORE_EXPORT const RenderObject::RenderObjectRareData& rareData() const;
RenderObjectRareData& ensureRareData();
void removeRareData();
using RareDataMap = SingleThreadWeakHashMap<const RenderObject, std::unique_ptr<RenderObjectRareData>>;
static RareDataMap& rareDataMap();
};
class RenderObject::SetLayoutNeededForbiddenScope {
public:
explicit SetLayoutNeededForbiddenScope(const RenderObject&, bool isForbidden = true);
#if ASSERT_ENABLED
~SetLayoutNeededForbiddenScope();
private:
CheckedRef<const RenderObject> m_renderObject;
bool m_preexistingForbidden;
#endif
};
inline void RenderObject::setSelectionStateIfNeeded(HighlightState state)
{
if (selectionState() == state)
return;
setSelectionState(state);
}
inline void RenderObject::setHasVisibleBoxDecorations(bool b)
{
if (!b) {
m_stateBitfields.setBoxDecorationState(BoxDecorationState::None);
return;
}
if (hasVisibleBoxDecorations())
return;
m_stateBitfields.setBoxDecorationState(BoxDecorationState::InvalidObscurationStatus);
}
inline void RenderObject::invalidateBackgroundObscurationStatus()
{
if (!hasVisibleBoxDecorations())
return;
m_stateBitfields.setBoxDecorationState(BoxDecorationState::InvalidObscurationStatus);
}
inline bool RenderObject::needsLayout() const
{
return selfNeedsLayout()
|| normalChildNeedsLayout()
|| outOfFlowChildNeedsLayout()
|| needsSimplifiedNormalFlowLayout()
|| needsOutOfFlowMovementLayout();
}
inline bool RenderObject::needsOutOfFlowMovementLayoutOnly() const
{
return needsOutOfFlowMovementLayout()
&& !selfNeedsLayout()
&& !normalChildNeedsLayout()
&& !outOfFlowChildNeedsLayout()
&& !needsSimplifiedNormalFlowLayout();
}
inline bool RenderObject::needsSimplifiedNormalFlowLayoutOnly() const
{
return needsSimplifiedNormalFlowLayout()
&& !selfNeedsLayout()
&& !normalChildNeedsLayout()
&& !outOfFlowChildNeedsLayout()
&& !needsOutOfFlowMovementLayout();
}
inline bool RenderObject::needsNormalChildOrSimplifiedLayoutOnly() const
{
return (normalChildNeedsLayout() || needsSimplifiedNormalFlowLayout())
&& !selfNeedsLayout()
&& !outOfFlowChildNeedsLayout()
&& !needsOutOfFlowMovementLayout();
}
inline RenderFragmentedFlow* RenderObject::enclosingFragmentedFlow() const
{
if (fragmentedFlowState() == FragmentedFlowState::NotInsideFlow)
return nullptr;
return locateEnclosingFragmentedFlow();
}
inline void RenderObject::setPositionState(PositionType position)
{
ASSERT((position != PositionType::Absolute && position != PositionType::Fixed) || isRenderBox());
m_stateBitfields.setPositionedState(position);
}
inline bool RenderObject::isSetNeedsLayoutForbidden() const
{
#if ASSERT_ENABLED
return m_setNeedsLayoutForbidden;
#else
return false;
#endif
}
#if !ASSERT_ENABLED
inline RenderObject::SetLayoutNeededForbiddenScope::SetLayoutNeededForbiddenScope(const RenderObject&, bool)
{
}
#endif
inline RenderObject* RenderObject::previousInFlowSibling() const
{
auto* previousSibling = this->previousSibling();
while (previousSibling && !previousSibling->isInFlow())
previousSibling = previousSibling->previousSibling();
return previousSibling;
}
inline RenderObject* RenderObject::nextInFlowSibling() const
{
auto* nextSibling = this->nextSibling();
while (nextSibling && !nextSibling->isInFlow())
nextSibling = nextSibling->nextSibling();
return nextSibling;
}
#if ENABLE(MATHML)
inline bool RenderObject::isRenderMathMLRow() const
{
switch (type()) {
case Type::MathMLFenced:
case Type::MathMLMath:
case Type::MathMLMenclose:
case Type::MathMLPadded:
case Type::MathMLRoot:
case Type::MathMLRow:
case Type::MathMLFraction:
case Type::MathMLScripts:
case Type::MathMLUnderOver:
return true;
default:
break;
}
return false;
}
#endif
inline bool RenderObject::isRenderTable() const
{
switch (type()) {
case Type::Table:
#if ENABLE(MATHML)
case Type::MathMLTable:
#endif
return true;
default:
break;
}
return false;
}
inline bool RenderObject::usesBoundaryCaching() const
{
// Use the same bit for UsesBoundaryCaching so that clang collapse two comparisons into one.
ASSERT(enumToUnderlyingType(ReplacedFlag::UsesBoundaryCaching) == enumToUnderlyingType(SVGModelObjectFlag::UsesBoundaryCaching));
return (m_typeSpecificFlags.kind() == TypeSpecificFlags::Kind::Replaced && m_typeSpecificFlags.replacedFlags().contains(ReplacedFlag::UsesBoundaryCaching))
|| (m_typeSpecificFlags.kind() == TypeSpecificFlags::Kind::SVGModelObject && m_typeSpecificFlags.svgFlags().contains(SVGModelObjectFlag::UsesBoundaryCaching));
}
inline CachedImageClient& RenderObject::cachedImageClient() const
{
if (!m_cachedImageClient)
lazyInitialize(m_cachedImageClient, CachedImageListener::create(*const_cast<RenderObject*>(this)));
return *m_cachedImageClient.get();
}
WTF::TextStream& operator<<(WTF::TextStream&, const RenderObject&);
WTF::TextStream& operator<<(WTF::TextStream&, const RenderObject::RepaintRects&);
#if ENABLE(TREE_DEBUGGING)
void printAccessibilityTreeForLiveDocuments();
void printPaintOrderTreeForLiveDocuments();
void printRenderTreeForLiveDocuments();
void printLayerTreeForLiveDocuments();
void printGraphicsLayerTreeForLiveDocuments();
#endif
} // namespace WebCore
#define SPECIALIZE_TYPE_TRAITS_RENDER_OBJECT(ToValueTypeName, predicate) \
SPECIALIZE_TYPE_TRAITS_BEGIN(WebCore::ToValueTypeName) \
static bool isType(const WebCore::RenderObject& renderer) { return renderer.predicate; } \
SPECIALIZE_TYPE_TRAITS_END()
#if ENABLE(TREE_DEBUGGING)
// Outside the WebCore namespace for ease of invocation from the debugger.
void showNodeTree(const WebCore::RenderObject*);
void showLineTree(const WebCore::RenderObject*);
void showRenderTree(const WebCore::RenderObject*);
#endif