blob: e612916583a2ac5bac2bdb6e3e60ba36b6ec441e [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/inline/ng_line_box_fragment_builder.h"
#include "core/layout/ng/exclusions/ng_exclusion_space.h"
#include "core/layout/ng/geometry/ng_logical_size.h"
#include "core/layout/ng/inline/ng_inline_break_token.h"
#include "core/layout/ng/inline/ng_inline_node.h"
#include "core/layout/ng/inline/ng_physical_line_box_fragment.h"
#include "core/layout/ng/ng_fragment.h"
#include "core/layout/ng/ng_layout_result.h"
#include "core/layout/ng/ng_positioned_float.h"
namespace blink {
NGLineBoxFragmentBuilder::NGLineBoxFragmentBuilder(
NGInlineNode node,
scoped_refptr<const ComputedStyle> style,
WritingMode writing_mode,
TextDirection)
: NGContainerFragmentBuilder(style, writing_mode, TextDirection::kLtr),
node_(node) {}
NGLineBoxFragmentBuilder::~NGLineBoxFragmentBuilder() {}
void NGLineBoxFragmentBuilder::Reset() {
children_.clear();
offsets_.clear();
metrics_ = NGLineHeightMetrics();
inline_size_ = LayoutUnit();
}
NGLogicalSize NGLineBoxFragmentBuilder::Size() const {
return {inline_size_, metrics_.LineHeight().ClampNegativeToZero()};
}
LayoutUnit NGLineBoxFragmentBuilder::ComputeBlockSize() const {
LayoutUnit block_size;
WritingMode writing_mode(node_.Style().GetWritingMode());
for (size_t i = 0; i < children_.size(); ++i) {
block_size = std::max(
block_size, offsets_[i].block_offset +
NGFragment(writing_mode, *children_[i]).BlockSize());
}
return block_size;
}
const NGPhysicalFragment* NGLineBoxFragmentBuilder::Child::PhysicalFragment()
const {
return layout_result ? layout_result->PhysicalFragment().get()
: fragment.get();
}
void NGLineBoxFragmentBuilder::ChildList::InsertChild(
unsigned index,
scoped_refptr<NGLayoutResult> layout_result,
const NGLogicalOffset& offset,
LayoutUnit inline_size,
UBiDiLevel bidi_level) {
children_.insert(
index, Child{std::move(layout_result), offset, inline_size, bidi_level});
}
void NGLineBoxFragmentBuilder::ChildList::MoveInInlineDirection(
LayoutUnit delta,
unsigned start,
unsigned end) {
for (unsigned index = start; index < end; index++)
children_[index].offset.inline_offset += delta;
}
void NGLineBoxFragmentBuilder::ChildList::MoveInBlockDirection(
LayoutUnit delta) {
for (auto& child : children_)
child.offset.block_offset += delta;
}
void NGLineBoxFragmentBuilder::ChildList::MoveInBlockDirection(LayoutUnit delta,
unsigned start,
unsigned end) {
for (unsigned index = start; index < end; index++)
children_[index].offset.block_offset += delta;
}
void NGLineBoxFragmentBuilder::SetMetrics(const NGLineHeightMetrics& metrics) {
metrics_ = metrics;
}
void NGLineBoxFragmentBuilder::SwapPositionedFloats(
Vector<NGPositionedFloat>* positioned_floats) {
positioned_floats_.swap(*positioned_floats);
}
void NGLineBoxFragmentBuilder::SetBreakToken(
scoped_refptr<NGInlineBreakToken> break_token) {
break_token_ = std::move(break_token);
}
void NGLineBoxFragmentBuilder::AddChildren(ChildList& children) {
offsets_.ReserveCapacity(children.size());
children_.ReserveCapacity(children.size());
for (auto& child : children) {
if (child.layout_result) {
DCHECK(!child.fragment);
AddChild(std::move(child.layout_result), child.offset);
DCHECK(!child.layout_result);
} else if (child.fragment) {
AddChild(std::move(child.fragment), child.offset);
DCHECK(!child.fragment);
}
}
}
scoped_refptr<NGLayoutResult> NGLineBoxFragmentBuilder::ToLineBoxFragment() {
DCHECK_EQ(offsets_.size(), children_.size());
WritingMode writing_mode(node_.Style().GetWritingMode());
NGPhysicalSize physical_size =
NGLogicalSize(inline_size_, block_size_).ConvertToPhysical(writing_mode);
NGPhysicalOffsetRect contents_visual_rect({}, physical_size);
for (size_t i = 0; i < children_.size(); ++i) {
NGPhysicalFragment* child = children_[i].get();
child->SetOffset(offsets_[i].ConvertToPhysical(
writing_mode, Direction(), physical_size, child->Size()));
child->PropagateContentsVisualRect(&contents_visual_rect);
}
scoped_refptr<NGPhysicalLineBoxFragment> fragment =
base::AdoptRef(new NGPhysicalLineBoxFragment(
Style(), physical_size, children_, contents_visual_rect, metrics_,
break_token_ ? std::move(break_token_)
: NGInlineBreakToken::Create(node_)));
return base::AdoptRef(new NGLayoutResult(
std::move(fragment), oof_positioned_descendants_, positioned_floats_,
unpositioned_floats_, std::move(exclusion_space_), bfc_offset_,
end_margin_strut_,
/* intrinsic_block_size */ LayoutUnit(),
/* minimal_space_shortage */ LayoutUnit::Max(), EBreakBetween::kAuto,
EBreakBetween::kAuto,
/* has_forced_break */ false, NGLayoutResult::kSuccess));
}
} // namespace blink