blob: 8c41bbade0f1abf4edd291f83f31a76ea7a56eb5 [file] [log] [blame]
/**
* Copyright (C) 2003-2025 Apple 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/DocumentView.h>
#include <WebCore/RenderBlock.h>
#include <WebCore/RenderBox.h>
#include <WebCore/RenderBoxModelObjectInlines.h>
#include <WebCore/RenderElementInlines.h>
namespace WebCore {
inline LayoutUnit RenderBox::marginBoxLogicalHeight(WritingMode writingMode) const { return writingMode.isHorizontal() ? m_marginBox.top() + height() + m_marginBox.bottom() : m_marginBox.right() + width() + m_marginBox.left(); }
inline LayoutSize RenderBox::borderBoxLogicalSize() const { return logicalSize(); }
inline LayoutRect RenderBox::clientBoxRect() const { return LayoutRect(clientLeft(), clientTop(), clientWidth(), clientHeight()); }
inline LayoutUnit RenderBox::clientLeft() const { return borderLeft(); }
inline LayoutUnit RenderBox::clientLogicalBottom() const { return borderBefore() + clientLogicalHeight(); }
inline LayoutUnit RenderBox::clientLogicalHeight() const { return writingMode().isHorizontal() ? clientHeight() : clientWidth(); }
inline LayoutUnit RenderBox::clientLogicalWidth() const { return writingMode().isHorizontal() ? clientWidth() : clientHeight(); }
inline LayoutUnit RenderBox::clientTop() const { return borderTop(); }
inline LayoutRect RenderBox::computedCSSContentBoxRect() const { return LayoutRect(borderLeft() + computedCSSPaddingLeft(), borderTop() + computedCSSPaddingTop(), paddingBoxWidth() - computedCSSPaddingLeft() - computedCSSPaddingRight() - (style().scrollbarGutter().isStableBothEdges() ? verticalScrollbarWidth() : 0), paddingBoxHeight() - computedCSSPaddingTop() - computedCSSPaddingBottom() - (style().scrollbarGutter().isStableBothEdges() ? horizontalScrollbarHeight() : 0)); }
inline LayoutUnit RenderBox::contentBoxHeight() const { return std::max(0_lu, paddingBoxHeight() - paddingTop() - paddingBottom() - (style().scrollbarGutter().isStableBothEdges() ? horizontalScrollbarHeight() : 0)); }
inline LayoutUnit RenderBox::contentBoxLogicalHeight() const { return writingMode().isHorizontal() ? contentBoxHeight() : contentBoxWidth(); }
inline LayoutUnit RenderBox::contentBoxLogicalHeight(LayoutUnit overridingBorderBoxHeight) const { return std::max(0_lu, overridingBorderBoxHeight - borderAndPaddingLogicalHeight() - scrollbarLogicalHeight() - (style().scrollbarGutter().isStableBothEdges() ? scrollbarLogicalHeight() : 0)); }
inline LayoutSize RenderBox::contentBoxLogicalSize() const { return writingMode().isHorizontal() ? contentBoxSize() : contentBoxSize().transposedSize(); }
inline LayoutUnit RenderBox::contentBoxLogicalWidth() const { return writingMode().isHorizontal() ? contentBoxWidth() : contentBoxHeight(); }
inline LayoutUnit RenderBox::contentBoxLogicalWidth(LayoutUnit overridingBorderBoxWidth) const { return std::max(LayoutUnit(), overridingBorderBoxWidth - borderAndPaddingLogicalWidth() - scrollbarLogicalWidth() - (style().scrollbarGutter().isStableBothEdges() ? scrollbarLogicalWidth() : 0)); }
inline LayoutSize RenderBox::contentBoxSize() const { return { contentBoxWidth(), contentBoxHeight() }; }
inline LayoutUnit RenderBox::contentBoxWidth() const { return std::max(0_lu, paddingBoxWidth() - paddingLeft() - paddingRight() - (style().scrollbarGutter().isStableBothEdges() ? verticalScrollbarWidth() : 0)); }
inline std::optional<LayoutUnit> RenderBox::explicitIntrinsicInnerLogicalHeight() const { return writingMode().isHorizontal() ? explicitIntrinsicInnerHeight() : explicitIntrinsicInnerWidth(); }
inline std::optional<LayoutUnit> RenderBox::explicitIntrinsicInnerLogicalWidth() const { return writingMode().isHorizontal() ? explicitIntrinsicInnerWidth() : explicitIntrinsicInnerHeight(); }
inline bool RenderBox::hasHorizontalOverflow() const { return scrollWidth() != roundToInt(paddingBoxWidth()); }
inline bool RenderBox::hasScrollableOverflow() const { return hasScrollableOverflowX() || hasScrollableOverflowY(); }
inline bool RenderBox::hasScrollableOverflowX() const { return scrollsOverflowX() && hasHorizontalOverflow(); }
inline bool RenderBox::hasScrollableOverflowY() const { return scrollsOverflowY() && hasVerticalOverflow(); }
inline bool RenderBox::hasVerticalOverflow() const { return scrollHeight() != roundToInt(paddingBoxHeight()); }
inline LayoutUnit RenderBox::intrinsicLogicalHeight() const { return writingMode().isHorizontal() ? intrinsicSize().height() : intrinsicSize().width(); }
inline LayoutUnit RenderBox::logicalBottom() const { return logicalTop() + logicalHeight(); }
inline LayoutUnit RenderBox::logicalHeight() const { return writingMode().isHorizontal() ? height() : width(); }
inline LayoutUnit RenderBox::logicalLeft() const { return writingMode().isHorizontal() ? x() : y(); }
inline LayoutUnit RenderBox::logicalLeftLayoutOverflow() const { return writingMode().isHorizontal() ? layoutOverflowRect().x() : layoutOverflowRect().y(); }
inline LayoutUnit RenderBox::logicalLeftVisualOverflow() const { return writingMode().isHorizontal() ? visualOverflowRect().x() : visualOverflowRect().y(); }
inline LayoutUnit RenderBox::logicalRight() const { return logicalLeft() + logicalWidth(); }
inline LayoutUnit RenderBox::logicalRightLayoutOverflow() const { return writingMode().isHorizontal() ? layoutOverflowRect().maxX() : layoutOverflowRect().maxY(); }
inline LayoutUnit RenderBox::logicalRightVisualOverflow() const { return writingMode().isHorizontal() ? visualOverflowRect().maxX() : visualOverflowRect().maxY(); }
inline LayoutSize RenderBox::logicalSize() const { return writingMode().isHorizontal() ? m_frameRect.size() : m_frameRect.size().transposedSize(); }
inline LayoutUnit RenderBox::logicalTop() const { return writingMode().isHorizontal() ? y() : x(); }
inline LayoutUnit RenderBox::logicalWidth() const { return writingMode().isHorizontal() ? width() : height(); }
inline LayoutUnit RenderBox::paddingBoxHeight() const { return std::max(0_lu, height() - borderTop() - borderBottom() - horizontalScrollbarHeight()); }
inline LayoutUnit RenderBox::paddingBoxWidth() const { return std::max(0_lu, width() - borderLeft() - borderRight() - verticalScrollbarWidth()); }
inline int RenderBox::scrollbarLogicalHeight() const { return writingMode().isHorizontal() ? horizontalScrollbarHeight() : verticalScrollbarWidth(); }
inline int RenderBox::scrollbarLogicalWidth() const { return writingMode().isHorizontal() ? verticalScrollbarWidth() : horizontalScrollbarHeight(); }
inline void RenderBox::setLogicalLocation(LayoutPoint location) { setLocation(writingMode().isHorizontal() ? location : location.transposedPoint()); }
inline void RenderBox::setLogicalSize(LayoutSize size) { setSize(writingMode().isHorizontal() ? size : size.transposedSize()); }
inline bool RenderBox::shouldTrimChildMargin(Style::MarginTrimSide type, const RenderBox& child) const { return style().marginTrim().contains(type) && isChildEligibleForMarginTrim(type, child); }
inline bool RenderBox::stretchesToViewport() const { return document().inQuirksMode() && style().logicalHeight().isAuto() && !isFloatingOrOutOfFlowPositioned() && (isDocumentElementRenderer() || isBody()) && !shouldComputeLogicalHeightFromAspectRatio() && !isInline(); }
inline bool RenderBox::isColumnSpanner() const { return style().columnSpan() == ColumnSpan::All; }
inline bool RenderBox::scrollsOverflow() const
{
return scrollsOverflowX() || scrollsOverflowY();
}
inline bool RenderBox::scrollsOverflowX() const
{
return hasNonVisibleOverflow() && (style().overflowX() == Overflow::Scroll || style().overflowX() == Overflow::Auto);
}
inline bool RenderBox::scrollsOverflowY() const
{
return hasNonVisibleOverflow() && (style().overflowY() == Overflow::Scroll || style().overflowY() == Overflow::Auto);
}
inline bool RenderBox::isScrollContainerX() const
{
return style().overflowX() == Overflow::Scroll || style().overflowX() == Overflow::Hidden || style().overflowX() == Overflow::Auto;
}
inline bool RenderBox::isScrollContainerY() const
{
return style().overflowY() == Overflow::Scroll || style().overflowY() == Overflow::Hidden || style().overflowY() == Overflow::Auto;
}
inline LayoutPoint RenderBox::topLeftLocation() const
{
// This is inlined for speed, since it is used by updateLayerPosition() during scrolling.
if (!document().view() || !document().view()->hasFlippedBlockRenderers())
return location();
return topLeftLocationWithFlipping();
}
inline LayoutSize RenderBox::topLeftLocationOffset() const
{
// This is inlined for speed, since it is used by updateLayerPosition() during scrolling.
if (!document().view() || !document().view()->hasFlippedBlockRenderers())
return locationOffset();
return toLayoutSize(topLeftLocationWithFlipping());
}
inline LayoutRect RenderBox::paddingBoxRectIncludingScrollbar() const
{
auto borderWidths = this->borderWidths();
return LayoutRect(borderWidths.left(), borderWidths.top(), width() - borderWidths.left() - borderWidths.right(), height() - borderWidths.top() - borderWidths.bottom());
}
inline LayoutRect RenderBox::contentBoxRect() const
{
auto verticalScrollbarWidth = 0_lu;
auto horizontalScrollbarHeight = 0_lu;
auto leftScrollbarSpace = 0_lu;
auto topScrollbarSpace = 0_lu;
if (hasNonVisibleOverflow()) {
verticalScrollbarWidth = this->verticalScrollbarWidth();
horizontalScrollbarHeight = this->horizontalScrollbarHeight();
bool bothEdgeScrollbarGutters = style().scrollbarGutter().isStableBothEdges();
if ((shouldPlaceVerticalScrollbarOnLeft() || bothEdgeScrollbarGutters))
leftScrollbarSpace = verticalScrollbarWidth;
// FIXME: It's wrong that scrollbar-gutter: both-edges affects height: webkit.org/b/266938
if (bothEdgeScrollbarGutters)
topScrollbarSpace = horizontalScrollbarHeight;
}
auto padding = this->padding();
auto borderWidths = this->borderWidths();
auto location = LayoutPoint { borderWidths.left() + padding.left() + leftScrollbarSpace, borderWidths.top() + padding.top() + topScrollbarSpace };
auto paddingBoxWidth = std::max(0_lu, width() - borderWidths.left() - borderWidths.right() - verticalScrollbarWidth);
auto paddingBoxHeight = std::max(0_lu, height() - borderWidths.top() - borderWidths.bottom() - horizontalScrollbarHeight);
auto width = std::max(0_lu, paddingBoxWidth - padding.left() - padding.right() - leftScrollbarSpace);
auto height = std::max(0_lu, paddingBoxHeight - padding.top() - padding.bottom() - topScrollbarSpace);
auto size = LayoutSize { width, height };
return { location, size };
}
inline LayoutRect RenderBox::flippedContentBoxRect() const
{
auto rect = flippedClientBoxRect();
auto padding = this->padding();
if (!padding.isZero()) {
if (writingMode().isBlockFlipped())
padding = padding.blockFlippedCopy(writingMode());
rect.contract(padding);
rect.floorSize();
}
return rect;
}
inline LayoutRect RenderBox::marginBoxRect() const
{
auto zoomFactor = style().usedZoomForLength();
auto left = resolveLengthPercentageUsingContainerLogicalWidth(style().marginLeft(), zoomFactor);
auto right = resolveLengthPercentageUsingContainerLogicalWidth(style().marginRight(), zoomFactor);
auto top = resolveLengthPercentageUsingContainerLogicalWidth(style().marginTop(), zoomFactor);
auto bottom = resolveLengthPercentageUsingContainerLogicalWidth(style().marginBottom(), zoomFactor);
return { -left, -top, size().width() + left + right, size().height() + top + bottom };
}
inline void RenderBox::setLogicalHeight(LayoutUnit size)
{
if (writingMode().isHorizontal())
setHeight(size);
else
setWidth(size);
}
inline void RenderBox::setLogicalLeft(LayoutUnit left)
{
if (writingMode().isHorizontal())
setX(left);
else
setY(left);
}
inline void RenderBox::setLogicalTop(LayoutUnit top)
{
if (writingMode().isHorizontal())
setY(top);
else
setX(top);
}
inline void RenderBox::setLogicalWidth(LayoutUnit size)
{
if (writingMode().isHorizontal())
setWidth(size);
else
setHeight(size);
}
inline bool RenderBox::hasStretchedLogicalHeight(StretchingMode mode) const
{
CheckedPtr containingBlock = this->containingBlock();
if (!containingBlock)
return false;
auto containingAxis = writingMode().isOrthogonal(containingBlock->writingMode())
? LogicalBoxAxis::Inline : LogicalBoxAxis::Block;
return containingBlock->willStretchItem(*this, containingAxis, mode);
}
inline bool RenderBox::hasStretchedLogicalWidth(StretchingMode mode) const
{
CheckedPtr containingBlock = this->containingBlock();
if (!containingBlock)
return false;
auto containingAxis = writingMode().isOrthogonal(containingBlock->writingMode())
? LogicalBoxAxis::Block : LogicalBoxAxis::Inline;
return containingBlock->willStretchItem(*this, containingAxis, mode);
}
inline LayoutUnit resolveHeightForRatio(LayoutUnit borderAndPaddingLogicalWidth, LayoutUnit borderAndPaddingLogicalHeight, LayoutUnit logicalWidth, double aspectRatio, BoxSizing boxSizing)
{
if (boxSizing == BoxSizing::BorderBox)
return LayoutUnit((logicalWidth + borderAndPaddingLogicalWidth) * aspectRatio) - borderAndPaddingLogicalHeight;
return LayoutUnit(logicalWidth * aspectRatio);
}
inline bool isSkippedContentRoot(const RenderBox& renderBox)
{
return renderBox.element() && WebCore::isSkippedContentRoot(renderBox.style(), *renderBox.protectedElement());
}
inline bool RenderBox::backgroundIsKnownToBeObscured(const LayoutPoint& paintOffset)
{
if (boxDecorationState() == BoxDecorationState::InvalidObscurationStatus) {
auto computedBoxDecorationState = [&] {
if (isSkippedContentRoot(*this))
return BoxDecorationState::MayBeVisible;
return computeBackgroundIsKnownToBeObscured(paintOffset) ? BoxDecorationState::IsKnownToBeObscured : BoxDecorationState::MayBeVisible;
};
setBoxDecorationState(computedBoxDecorationState());
}
return boxDecorationState() == BoxDecorationState::IsKnownToBeObscured;
}
inline LayoutUnit RenderBox::blockSizeFromAspectRatio(LayoutUnit borderPaddingInlineSum, LayoutUnit borderPaddingBlockSum, double aspectRatioValue, BoxSizing boxSizing, LayoutUnit inlineSize, const Style::AspectRatio& aspectRatio, bool isRenderReplaced)
{
if (boxSizing == BoxSizing::BorderBox && aspectRatio.isRatio() && !isRenderReplaced)
return std::max(borderPaddingBlockSum, LayoutUnit(inlineSize / aspectRatioValue));
return LayoutUnit((inlineSize - borderPaddingInlineSum) / aspectRatioValue) + borderPaddingBlockSum;
}
} // namespace WebCore