| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  |  * Copyright (c) 2020-2022, Andreas Kling <kling@serenityos.org> | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-04-22 01:24:48 -07:00
										 |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <LibWeb/CSS/Length.h>
 | 
					
						
							|  |  |  | #include <LibWeb/DOM/Node.h>
 | 
					
						
							|  |  |  | #include <LibWeb/Dump.h>
 | 
					
						
							| 
									
										
										
										
											2021-10-06 20:02:41 +02:00
										 |  |  | #include <LibWeb/Layout/BlockContainer.h>
 | 
					
						
							| 
									
										
										
										
											2020-11-25 20:08:52 +01:00
										 |  |  | #include <LibWeb/Layout/BlockFormattingContext.h>
 | 
					
						
							| 
									
										
										
										
											2020-11-22 15:53:01 +01:00
										 |  |  | #include <LibWeb/Layout/Box.h>
 | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  | #include <LibWeb/Layout/InlineFormattingContext.h>
 | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  | #include <LibWeb/Layout/InlineLevelIterator.h>
 | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  | #include <LibWeb/Layout/LineBuilder.h>
 | 
					
						
							| 
									
										
										
										
											2020-11-22 15:53:01 +01:00
										 |  |  | #include <LibWeb/Layout/ReplacedBox.h>
 | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Web::Layout { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-23 15:22:18 +01:00
										 |  |  | InlineFormattingContext::InlineFormattingContext(BlockContainer& containing_block, BlockFormattingContext& parent) | 
					
						
							|  |  |  |     : FormattingContext(Type::Inline, containing_block, &parent) | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | InlineFormattingContext::~InlineFormattingContext() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-23 15:22:18 +01:00
										 |  |  | BlockFormattingContext& InlineFormattingContext::parent() | 
					
						
							| 
									
										
										
										
											2020-12-05 20:10:39 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-01-23 15:22:18 +01:00
										 |  |  |     return static_cast<BlockFormattingContext&>(*FormattingContext::parent()); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-09-23 16:58:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-23 15:22:18 +01:00
										 |  |  | BlockFormattingContext const& InlineFormattingContext::parent() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return static_cast<BlockFormattingContext const&>(*FormattingContext::parent()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | InlineFormattingContext::AvailableSpaceForLineInfo InlineFormattingContext::available_space_for_line(float y) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-01-24 02:03:29 +01:00
										 |  |  |     // NOTE: Floats are relative to the BFC root box, not necessarily the containing block of this IFC.
 | 
					
						
							|  |  |  |     auto box_in_root_rect = containing_block().margin_box_rect_in_ancestor_coordinate_space(parent().root()); | 
					
						
							|  |  |  |     float y_in_root = box_in_root_rect.y() + y; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-05 20:10:39 +01:00
										 |  |  |     AvailableSpaceForLineInfo info; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-23 15:22:18 +01:00
										 |  |  |     auto const& bfc = parent(); | 
					
						
							| 
									
										
										
										
											2020-12-05 20:10:39 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-22 15:19:18 +01:00
										 |  |  |     for (ssize_t i = bfc.left_side_floats().boxes.size() - 1; i >= 0; --i) { | 
					
						
							|  |  |  |         auto const& floating_box = bfc.left_side_floats().boxes.at(i); | 
					
						
							| 
									
										
										
										
											2020-12-12 19:58:23 +01:00
										 |  |  |         auto rect = floating_box.margin_box_as_relative_rect(); | 
					
						
							| 
									
										
										
										
											2022-01-24 02:03:29 +01:00
										 |  |  |         if (rect.contains_vertically(y_in_root)) { | 
					
						
							| 
									
										
										
										
											2020-12-05 20:10:39 +01:00
										 |  |  |             info.left = rect.right() + 1; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-06 00:49:09 +01:00
										 |  |  |     info.right = containing_block().content_width(); | 
					
						
							| 
									
										
										
										
											2020-12-05 20:10:39 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-22 15:19:18 +01:00
										 |  |  |     for (ssize_t i = bfc.right_side_floats().boxes.size() - 1; i >= 0; --i) { | 
					
						
							|  |  |  |         auto const& floating_box = bfc.right_side_floats().boxes.at(i); | 
					
						
							| 
									
										
										
										
											2020-12-12 19:58:23 +01:00
										 |  |  |         auto rect = floating_box.margin_box_as_relative_rect(); | 
					
						
							| 
									
										
										
										
											2022-01-24 02:03:29 +01:00
										 |  |  |         if (rect.contains_vertically(y_in_root)) { | 
					
						
							| 
									
										
										
										
											2020-12-05 20:10:39 +01:00
										 |  |  |             info.right = rect.left() - 1; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return info; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-06 19:48:02 +01:00
										 |  |  | void InlineFormattingContext::run(Box&, LayoutMode layout_mode) | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-02-23 20:42:32 +01:00
										 |  |  |     VERIFY(containing_block().children_are_inline()); | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     generate_line_boxes(layout_mode); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-06 19:48:02 +01:00
										 |  |  |     containing_block().for_each_child([&](auto& child) { | 
					
						
							| 
									
										
										
										
											2021-02-23 20:42:32 +01:00
										 |  |  |         VERIFY(child.is_inline()); | 
					
						
							| 
									
										
										
										
											2021-01-06 18:18:46 +01:00
										 |  |  |         if (is<Box>(child) && child.is_absolutely_positioned()) { | 
					
						
							| 
									
										
										
										
											2021-06-24 19:53:42 +02:00
										 |  |  |             layout_absolutely_positioned_element(verify_cast<Box>(child)); | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2021-01-06 18:18:46 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-06 11:31:19 +01:00
										 |  |  |     float min_line_height = containing_block().line_height(); | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  |     float max_line_width = 0; | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |     float content_height = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  |     for (auto& line_box : containing_block().line_boxes()) { | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |         float max_height = min_line_height; | 
					
						
							|  |  |  |         for (auto& fragment : line_box.fragments()) { | 
					
						
							|  |  |  |             max_height = max(max_height, fragment.height()); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  |         max_line_width = max(max_line_width, line_box.width()); | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |         content_height += max_height; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (layout_mode != LayoutMode::Default) { | 
					
						
							| 
									
										
										
										
											2022-02-06 00:49:09 +01:00
										 |  |  |         containing_block().set_content_width(max_line_width); | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-06 00:49:09 +01:00
										 |  |  |     containing_block().set_content_height(content_height); | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-22 15:53:01 +01:00
										 |  |  | void InlineFormattingContext::dimension_box_on_line(Box& box, LayoutMode layout_mode) | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-01-01 18:55:47 +01:00
										 |  |  |     if (is<ReplacedBox>(box)) { | 
					
						
							| 
									
										
										
										
											2021-06-24 19:53:42 +02:00
										 |  |  |         auto& replaced = verify_cast<ReplacedBox>(box); | 
					
						
							| 
									
										
										
										
											2022-02-06 00:49:09 +01:00
										 |  |  |         replaced.set_content_width(compute_width_for_replaced_element(replaced)); | 
					
						
							|  |  |  |         replaced.set_content_height(compute_height_for_replaced_element(replaced)); | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (box.is_inline_block()) { | 
					
						
							| 
									
										
										
										
											2021-10-06 20:02:41 +02:00
										 |  |  |         auto& inline_block = const_cast<BlockContainer&>(verify_cast<BlockContainer>(box)); | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-19 16:19:43 +00:00
										 |  |  |         if (inline_block.computed_values().width().is_length() && inline_block.computed_values().width().length().is_undefined_or_auto()) { | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |             auto result = calculate_shrink_to_fit_widths(inline_block); | 
					
						
							| 
									
										
										
										
											2022-02-06 00:49:09 +01:00
										 |  |  |             auto width_of_containing_block = CSS::Length::make_px(containing_block().content_width()); | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-27 14:38:59 +00:00
										 |  |  |             auto margin_left = inline_block.computed_values().margin().left.resolved(box, width_of_containing_block).resolved_or_zero(inline_block).to_px(inline_block); | 
					
						
							| 
									
										
										
										
											2021-01-06 10:34:31 +01:00
										 |  |  |             auto border_left_width = inline_block.computed_values().border_left().width; | 
					
						
							| 
									
										
										
										
											2022-01-27 14:38:59 +00:00
										 |  |  |             auto padding_left = inline_block.computed_values().padding().left.resolved(box, width_of_containing_block).resolved_or_zero(inline_block).to_px(inline_block); | 
					
						
							| 
									
										
										
										
											2020-11-29 22:00:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-27 14:38:59 +00:00
										 |  |  |             auto margin_right = inline_block.computed_values().margin().right.resolved(box, width_of_containing_block).resolved_or_zero(inline_block).to_px(inline_block); | 
					
						
							| 
									
										
										
										
											2021-01-06 10:34:31 +01:00
										 |  |  |             auto border_right_width = inline_block.computed_values().border_right().width; | 
					
						
							| 
									
										
										
										
											2022-01-27 14:38:59 +00:00
										 |  |  |             auto padding_right = inline_block.computed_values().padding().right.resolved(box, width_of_containing_block).resolved_or_zero(inline_block).to_px(inline_block); | 
					
						
							| 
									
										
										
										
											2020-11-29 22:00:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-06 00:49:09 +01:00
										 |  |  |             auto available_width = containing_block().content_width() | 
					
						
							| 
									
										
										
										
											2020-11-29 22:00:44 +01:00
										 |  |  |                 - margin_left | 
					
						
							|  |  |  |                 - border_left_width | 
					
						
							|  |  |  |                 - padding_left | 
					
						
							|  |  |  |                 - padding_right | 
					
						
							|  |  |  |                 - border_right_width | 
					
						
							|  |  |  |                 - margin_right; | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             auto width = min(max(result.preferred_minimum_width, available_width), result.preferred_width); | 
					
						
							| 
									
										
										
										
											2022-02-06 00:49:09 +01:00
										 |  |  |             inline_block.set_content_width(width); | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2022-02-06 00:49:09 +01:00
										 |  |  |             auto container_width = CSS::Length::make_px(containing_block().content_width()); | 
					
						
							|  |  |  |             inline_block.set_content_width(inline_block.computed_values().width().resolved(box, container_width).resolved_or_zero(inline_block).to_px(inline_block)); | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-12-02 12:54:34 +00:00
										 |  |  |         (void)layout_inside(inline_block, layout_mode); | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-19 16:19:43 +00:00
										 |  |  |         if (inline_block.computed_values().height().is_length() && inline_block.computed_values().height().length().is_undefined_or_auto()) { | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |             // FIXME: (10.6.6) If 'height' is 'auto', the height depends on the element's descendants per 10.6.7.
 | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2022-02-06 00:49:09 +01:00
										 |  |  |             auto container_height = CSS::Length::make_px(containing_block().content_height()); | 
					
						
							|  |  |  |             inline_block.set_content_height(inline_block.computed_values().height().resolved(box, container_height).resolved_or_zero(inline_block).to_px(inline_block)); | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Non-replaced, non-inline-block, box on a line!?
 | 
					
						
							|  |  |  |     // I don't think we should be here. Dump the box tree so we can take a look at it.
 | 
					
						
							|  |  |  |     dbgln("FIXME: I've been asked to dimension a non-replaced, non-inline-block box on a line:"); | 
					
						
							|  |  |  |     dump_tree(box); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  | void InlineFormattingContext::generate_line_boxes(LayoutMode layout_mode) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  |     containing_block().line_boxes().clear(); | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     InlineLevelIterator iterator(containing_block(), layout_mode); | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  |     LineBuilder line_builder(*this); | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (;;) { | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  |         auto item_opt = iterator.next(line_builder.available_width_for_current_line()); | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  |         if (!item_opt.has_value()) | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         auto& item = item_opt.value(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-21 14:15:10 +01:00
										 |  |  |         // Ignore collapsible whitespace chunks at the start of line, and if the last fragment already ends in whitespace.
 | 
					
						
							|  |  |  |         if (item.is_collapsible_whitespace && containing_block().line_boxes().last().is_empty_or_ends_in_whitespace()) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  |         switch (item.type) { | 
					
						
							|  |  |  |         case InlineLevelIterator::Item::Type::ForcedBreak: | 
					
						
							|  |  |  |             line_builder.break_line(); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case InlineLevelIterator::Item::Type::Element: { | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  |             auto& box = verify_cast<Layout::Box>(*item.node); | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  |             dimension_box_on_line(box, layout_mode); | 
					
						
							| 
									
										
										
										
											2022-02-06 00:49:09 +01:00
										 |  |  |             line_builder.break_if_needed(layout_mode, box.content_width(), item.should_force_break); | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  |             line_builder.append_box(box); | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  |         case InlineLevelIterator::Item::Type::Text: { | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  |             auto& text_node = verify_cast<Layout::TextNode>(*item.node); | 
					
						
							| 
									
										
										
										
											2022-01-20 13:46:04 +01:00
										 |  |  |             line_builder.break_if_needed(layout_mode, item.width, item.should_force_break); | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  |             line_builder.append_text_chunk( | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  |                 text_node, | 
					
						
							|  |  |  |                 item.offset_in_node, | 
					
						
							|  |  |  |                 item.length_in_node, | 
					
						
							|  |  |  |                 item.width, | 
					
						
							|  |  |  |                 text_node.font().glyph_height()); | 
					
						
							| 
									
										
										
										
											2022-01-19 11:59:44 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto& line_box : containing_block().line_boxes()) { | 
					
						
							|  |  |  |         line_box.trim_trailing_whitespace(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-22 10:32:49 +01:00
										 |  |  |     line_builder.remove_last_line_if_empty(); | 
					
						
							| 
									
										
										
										
											2022-01-17 15:07:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-22 13:38:18 +01:00
										 |  |  | } |