
This patch adds a map of Layout::Node to FormattingState::NodeState. Instead of updating layout nodes incrementally as layout progresses through the formatting contexts, all updates are now written to the corresponding NodeState instead. At the end of layout, FormattingState::commit() is called, which transfers all the values from the NodeState objects to the Node. This will soon allow us to perform completely non-destructive layouts which don't affect the tree. Note that there are many imperfections here, and still many places where we assign to the NodeState, but later read directly from the Node instead. I'm just committing at this stage to make subsequent diffs easier to understand.
81 lines
2.6 KiB
C++
81 lines
2.6 KiB
C++
/*
|
|
* Copyright (c) 2018-2021, Andreas Kling <kling@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <LibWeb/Layout/Box.h>
|
|
#include <LibWeb/Layout/LineBox.h>
|
|
|
|
namespace Web::Layout {
|
|
|
|
// https://www.w3.org/TR/css-display/#block-container
|
|
class BlockContainer : public Box {
|
|
public:
|
|
BlockContainer(DOM::Document&, DOM::Node*, NonnullRefPtr<CSS::StyleProperties>);
|
|
BlockContainer(DOM::Document&, DOM::Node*, CSS::ComputedValues);
|
|
virtual ~BlockContainer() override;
|
|
|
|
virtual void paint(PaintContext&, PaintPhase) override;
|
|
|
|
virtual HitTestResult hit_test(const Gfx::IntPoint&, HitTestType) const override;
|
|
|
|
BlockContainer* previous_sibling() { return verify_cast<BlockContainer>(Node::previous_sibling()); }
|
|
const BlockContainer* previous_sibling() const { return verify_cast<BlockContainer>(Node::previous_sibling()); }
|
|
BlockContainer* next_sibling() { return verify_cast<BlockContainer>(Node::next_sibling()); }
|
|
const BlockContainer* next_sibling() const { return verify_cast<BlockContainer>(Node::next_sibling()); }
|
|
|
|
template<typename Callback>
|
|
void for_each_fragment(Callback);
|
|
template<typename Callback>
|
|
void for_each_fragment(Callback) const;
|
|
|
|
bool is_scrollable() const;
|
|
const Gfx::FloatPoint& scroll_offset() const { return m_scroll_offset; }
|
|
void set_scroll_offset(const Gfx::FloatPoint&);
|
|
|
|
const Vector<LineBox>& line_boxes() const { return m_line_boxes; }
|
|
|
|
void set_line_boxes(Vector<LineBox>&& line_boxes) { m_line_boxes = move(line_boxes); }
|
|
|
|
protected:
|
|
Vector<LineBox> m_line_boxes;
|
|
|
|
private:
|
|
virtual bool is_block_container() const final { return true; }
|
|
virtual bool wants_mouse_events() const override { return false; }
|
|
virtual bool handle_mousewheel(Badge<EventHandler>, const Gfx::IntPoint&, unsigned buttons, unsigned modifiers, int wheel_delta_x, int wheel_delta_y) override;
|
|
|
|
bool should_clip_overflow() const;
|
|
|
|
Gfx::FloatPoint m_scroll_offset;
|
|
};
|
|
|
|
template<>
|
|
inline bool Node::fast_is<BlockContainer>() const { return is_block_container(); }
|
|
|
|
template<typename Callback>
|
|
void BlockContainer::for_each_fragment(Callback callback)
|
|
{
|
|
for (auto& line_box : line_boxes()) {
|
|
for (auto& fragment : line_box.fragments()) {
|
|
if (callback(fragment) == IterationDecision::Break)
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
template<typename Callback>
|
|
void BlockContainer::for_each_fragment(Callback callback) const
|
|
{
|
|
for (auto& line_box : line_boxes()) {
|
|
for (auto& fragment : line_box.fragments()) {
|
|
if (callback(fragment) == IterationDecision::Break)
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|