blob: 0d53c1c34dc9036685e1af95a6ad2a8e568e42d7 [file] [log] [blame]
// Copyright 2017 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "core/layout/ng/ng_layout_input_node.h"
#include "core/layout/LayoutReplaced.h"
#include "core/layout/MinMaxSize.h"
#include "core/layout/ng/geometry/ng_logical_size.h"
#include "core/layout/ng/inline/ng_inline_node.h"
#include "core/layout/ng/ng_block_node.h"
#include "core/layout/ng/ng_layout_result.h"
#include "platform/wtf/text/StringBuilder.h"
namespace blink {
namespace {
#ifndef NDEBUG
void AppendNodeToString(NGLayoutInputNode node,
StringBuilder* string_builder,
unsigned indent = 2) {
if (!node)
return;
DCHECK(string_builder);
string_builder->Append(node.ToString());
string_builder->Append("\n");
StringBuilder indent_builder;
for (unsigned i = 0; i < indent; i++)
indent_builder.Append(" ");
if (node.IsBlock()) {
NGLayoutInputNode first_child = ToNGBlockNode(node).FirstChild();
for (NGLayoutInputNode node_runner = first_child; node_runner;
node_runner = node_runner.NextSibling()) {
string_builder->Append(indent_builder.ToString());
AppendNodeToString(node_runner, string_builder, indent + 2);
}
}
if (node.IsInline()) {
for (const NGInlineItem& inline_item : ToNGInlineNode(node).Items()) {
string_builder->Append(indent_builder.ToString());
string_builder->Append(inline_item.ToString());
string_builder->Append("\n");
}
NGLayoutInputNode next_sibling = ToNGInlineNode(node).NextSibling();
for (NGLayoutInputNode node_runner = next_sibling; node_runner;
node_runner = node_runner.NextSibling()) {
string_builder->Append(indent_builder.ToString());
AppendNodeToString(node_runner, string_builder, indent + 2);
}
}
}
#endif
} // namespace
bool NGLayoutInputNode::IsInline() const {
return type_ == kInline;
}
bool NGLayoutInputNode::IsBlock() const {
return type_ == kBlock;
}
bool NGLayoutInputNode::IsFloating() const {
return IsBlock() && Style().IsFloating();
}
bool NGLayoutInputNode::IsOutOfFlowPositioned() const {
return IsBlock() && Style().HasOutOfFlowPosition();
}
bool NGLayoutInputNode::IsReplaced() const {
return box_->IsLayoutReplaced();
}
bool NGLayoutInputNode::IsQuirkyContainer() const {
return box_->GetDocument().InQuirksMode() &&
(box_->IsBody() || box_->IsTableCell());
}
bool NGLayoutInputNode::IsAbsoluteContainer() const {
return box_->CanContainAbsolutePositionObjects();
}
bool NGLayoutInputNode::IsFixedContainer() const {
return box_->CanContainFixedPositionObjects();
}
bool NGLayoutInputNode::CreatesNewFormattingContext() const {
return IsBlock() && box_->AvoidsFloats();
}
RefPtr<NGLayoutResult> NGLayoutInputNode::Layout(const NGConstraintSpace& space,
NGBreakToken* break_token) {
return IsInline() ? ToNGInlineNode(*this).Layout(space, break_token)
: ToNGBlockNode(*this).Layout(space, break_token);
}
MinMaxSize NGLayoutInputNode::ComputeMinMaxSize() {
return IsInline() ? ToNGInlineNode(*this).ComputeMinMaxSize()
: ToNGBlockNode(*this).ComputeMinMaxSize();
}
void NGLayoutInputNode::IntrinsicSize(
NGLogicalSize* default_intrinsic_size,
Optional<LayoutUnit>* computed_inline_size,
Optional<LayoutUnit>* computed_block_size,
NGLogicalSize* aspect_ratio) const {
DCHECK(IsReplaced());
LayoutSize box_intrinsic_size = box_->IntrinsicSize();
// Transform to logical coordinates if needed.
if (!Style().IsHorizontalWritingMode())
box_intrinsic_size = box_intrinsic_size.TransposedSize();
*default_intrinsic_size =
NGLogicalSize(box_intrinsic_size.Width(), box_intrinsic_size.Height());
LayoutReplaced::IntrinsicSizingInfo legacy_sizing_info;
ToLayoutReplaced(box_)->ComputeIntrinsicSizingInfo(legacy_sizing_info);
if (legacy_sizing_info.has_width)
*computed_inline_size = LayoutUnit(legacy_sizing_info.size.Width());
if (legacy_sizing_info.has_height)
*computed_block_size = LayoutUnit(legacy_sizing_info.size.Height());
*aspect_ratio =
NGLogicalSize(LayoutUnit(legacy_sizing_info.aspect_ratio.Width()),
LayoutUnit(legacy_sizing_info.aspect_ratio.Height()));
}
NGLayoutInputNode NGLayoutInputNode::NextSibling() {
return IsInline() ? ToNGInlineNode(*this).NextSibling()
: ToNGBlockNode(*this).NextSibling();
}
LayoutObject* NGLayoutInputNode::GetLayoutObject() const {
return box_;
}
const ComputedStyle& NGLayoutInputNode::Style() const {
return box_->StyleRef();
}
String NGLayoutInputNode::ToString() const {
return IsInline() ? ToNGInlineNode(*this).ToString()
: ToNGBlockNode(*this).ToString();
}
#ifndef NDEBUG
void NGLayoutInputNode::ShowNodeTree() const {
StringBuilder string_builder;
string_builder.Append("\n.:: LayoutNG Node Tree ::.\n\n");
AppendNodeToString(*this, &string_builder);
fprintf(stderr, "%s\n", string_builder.ToString().Utf8().data());
}
#endif
} // namespace blink