FormattingContext.h 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. /*
  2. * Copyright (c) 2020-2022, Andreas Kling <andreas@ladybird.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #pragma once
  7. #include <AK/OwnPtr.h>
  8. #include <LibWeb/Forward.h>
  9. #include <LibWeb/Layout/AvailableSpace.h>
  10. #include <LibWeb/Layout/LayoutState.h>
  11. namespace Web::Layout {
  12. // NOTE: We use a custom clamping function here instead of AK::clamp(), since the AK version
  13. // will VERIFY(max >= min) and CSS explicitly allows that (see css-values-4.)
  14. template<typename T>
  15. [[nodiscard]] constexpr T css_clamp(T const& value, T const& min, T const& max)
  16. {
  17. return ::max(min, ::min(value, max));
  18. }
  19. class FormattingContext {
  20. public:
  21. virtual ~FormattingContext();
  22. enum class Type {
  23. Block,
  24. Inline,
  25. Flex,
  26. Grid,
  27. Table,
  28. SVG,
  29. InternalReplaced, // Internal hack formatting context for replaced elements. FIXME: Get rid of this.
  30. InternalDummy, // Internal hack formatting context for unimplemented things. FIXME: Get rid of this.
  31. };
  32. virtual void run(AvailableSpace const&) = 0;
  33. // This function returns the automatic content height of the context's root box.
  34. virtual CSSPixels automatic_content_width() const = 0;
  35. // This function returns the automatic content height of the context's root box.
  36. virtual CSSPixels automatic_content_height() const = 0;
  37. Box const& context_box() const { return m_context_box; }
  38. FormattingContext* parent() { return m_parent; }
  39. FormattingContext const* parent() const { return m_parent; }
  40. Type type() const { return m_type; }
  41. bool is_block_formatting_context() const { return type() == Type::Block; }
  42. virtual bool inhibits_floating() const { return false; }
  43. [[nodiscard]] static Optional<Type> formatting_context_type_created_by_box(Box const&);
  44. static bool creates_block_formatting_context(Box const&);
  45. CSSPixels compute_table_box_width_inside_table_wrapper(Box const&, AvailableSpace const&);
  46. CSSPixels compute_table_box_height_inside_table_wrapper(Box const&, AvailableSpace const&);
  47. CSSPixels compute_width_for_replaced_element(Box const&, AvailableSpace const&) const;
  48. CSSPixels compute_height_for_replaced_element(Box const&, AvailableSpace const&) const;
  49. OwnPtr<FormattingContext> create_independent_formatting_context_if_needed(LayoutState&, LayoutMode, Box const& child_box);
  50. virtual void parent_context_did_dimension_child_root_box() { }
  51. CSSPixels calculate_min_content_width(Layout::Box const&) const;
  52. CSSPixels calculate_max_content_width(Layout::Box const&) const;
  53. CSSPixels calculate_min_content_height(Layout::Box const&, CSSPixels width) const;
  54. CSSPixels calculate_max_content_height(Layout::Box const&, CSSPixels width) const;
  55. CSSPixels calculate_fit_content_height(Layout::Box const&, AvailableSpace const&) const;
  56. CSSPixels calculate_fit_content_width(Layout::Box const&, AvailableSpace const&) const;
  57. CSSPixels calculate_inner_width(Layout::Box const&, AvailableSize const&, CSS::Size const& width) const;
  58. CSSPixels calculate_inner_height(Layout::Box const&, AvailableSize const&, CSS::Size const& height) const;
  59. virtual CSSPixels greatest_child_width(Box const&) const;
  60. [[nodiscard]] CSSPixelRect absolute_content_rect(Box const&) const;
  61. [[nodiscard]] CSSPixelRect absolute_content_rect(LayoutState::UsedValues const&) const;
  62. [[nodiscard]] CSSPixelRect margin_box_rect(Box const&) const;
  63. [[nodiscard]] CSSPixelRect margin_box_rect_in_ancestor_coordinate_space(Box const&, Box const& ancestor_box) const;
  64. [[nodiscard]] CSSPixelRect margin_box_rect(LayoutState::UsedValues const&) const;
  65. [[nodiscard]] CSSPixelRect margin_box_rect_in_ancestor_coordinate_space(LayoutState::UsedValues const&, Box const& ancestor_box) const;
  66. [[nodiscard]] CSSPixelRect border_box_rect(Box const&) const;
  67. [[nodiscard]] CSSPixelRect border_box_rect(LayoutState::UsedValues const&) const;
  68. [[nodiscard]] CSSPixelRect border_box_rect_in_ancestor_coordinate_space(Box const&, Box const& ancestor_box) const;
  69. [[nodiscard]] CSSPixelRect border_box_rect_in_ancestor_coordinate_space(LayoutState::UsedValues const&, Box const& ancestor_box) const;
  70. [[nodiscard]] CSSPixelRect content_box_rect(Box const&) const;
  71. [[nodiscard]] CSSPixelRect content_box_rect(LayoutState::UsedValues const&) const;
  72. [[nodiscard]] CSSPixelRect content_box_rect_in_ancestor_coordinate_space(Box const&, Box const& ancestor_box) const;
  73. [[nodiscard]] CSSPixelRect content_box_rect_in_ancestor_coordinate_space(LayoutState::UsedValues const&, Box const& ancestor_box) const;
  74. [[nodiscard]] CSSPixels box_baseline(Box const&) const;
  75. [[nodiscard]] CSSPixelRect content_box_rect_in_static_position_ancestor_coordinate_space(Box const&, Box const& ancestor_box) const;
  76. [[nodiscard]] CSSPixels containing_block_width_for(NodeWithStyleAndBoxModelMetrics const&) const;
  77. [[nodiscard]] CSSPixels containing_block_height_for(NodeWithStyleAndBoxModelMetrics const&) const;
  78. [[nodiscard]] CSSPixels calculate_stretch_fit_width(Box const&, AvailableSize const&) const;
  79. [[nodiscard]] CSSPixels calculate_stretch_fit_height(Box const&, AvailableSize const&) const;
  80. bool can_skip_is_anonymous_text_run(Box&);
  81. void compute_inset(NodeWithStyleAndBoxModelMetrics const&, CSSPixelSize containing_block_size);
  82. protected:
  83. FormattingContext(Type, LayoutMode, LayoutState&, Box const&, FormattingContext* parent = nullptr);
  84. static bool should_treat_width_as_auto(Box const&, AvailableSpace const&);
  85. static bool should_treat_height_as_auto(Box const&, AvailableSpace const&);
  86. [[nodiscard]] bool should_treat_max_width_as_none(Box const&, AvailableSize const&) const;
  87. [[nodiscard]] bool should_treat_max_height_as_none(Box const&, AvailableSize const&) const;
  88. OwnPtr<FormattingContext> layout_inside(Box const&, LayoutMode, AvailableSpace const&);
  89. struct SpaceUsedByFloats {
  90. CSSPixels left { 0 };
  91. CSSPixels right { 0 };
  92. };
  93. struct SpaceUsedAndContainingMarginForFloats {
  94. // Width for left / right floats, including their own margins.
  95. CSSPixels left_used_space;
  96. CSSPixels right_used_space;
  97. // Left / right total margins from the outermost containing block to the floating element.
  98. // Each block in the containing chain adds its own margin and we store the total here.
  99. CSSPixels left_total_containing_margin;
  100. CSSPixels right_total_containing_margin;
  101. JS::GCPtr<Box const> matching_left_float_box;
  102. };
  103. struct ShrinkToFitResult {
  104. CSSPixels preferred_width { 0 };
  105. CSSPixels preferred_minimum_width { 0 };
  106. };
  107. CSSPixels tentative_width_for_replaced_element(Box const&, CSS::Size const& computed_width, AvailableSpace const&) const;
  108. CSSPixels tentative_height_for_replaced_element(Box const&, CSS::Size const& computed_height, AvailableSpace const&) const;
  109. CSSPixels compute_auto_height_for_block_formatting_context_root(Box const&) const;
  110. [[nodiscard]] CSSPixelSize solve_replaced_size_constraint(CSSPixels input_width, CSSPixels input_height, Box const&, AvailableSpace const&) const;
  111. ShrinkToFitResult calculate_shrink_to_fit_widths(Box const&);
  112. void layout_absolutely_positioned_element(Box const&, AvailableSpace const&);
  113. void compute_width_for_absolutely_positioned_element(Box const&, AvailableSpace const&);
  114. void compute_width_for_absolutely_positioned_non_replaced_element(Box const&, AvailableSpace const&);
  115. void compute_width_for_absolutely_positioned_replaced_element(Box const&, AvailableSpace const&);
  116. enum class BeforeOrAfterInsideLayout {
  117. Before,
  118. After,
  119. };
  120. void compute_height_for_absolutely_positioned_element(Box const&, AvailableSpace const&, BeforeOrAfterInsideLayout);
  121. void compute_height_for_absolutely_positioned_non_replaced_element(Box const&, AvailableSpace const&, BeforeOrAfterInsideLayout);
  122. void compute_height_for_absolutely_positioned_replaced_element(Box const&, AvailableSpace const&, BeforeOrAfterInsideLayout);
  123. [[nodiscard]] Optional<CSSPixels> compute_auto_height_for_absolutely_positioned_element(Box const&, AvailableSpace const&, BeforeOrAfterInsideLayout) const;
  124. [[nodiscard]] Box const* box_child_to_derive_baseline_from(Box const&) const;
  125. Type m_type {};
  126. LayoutMode m_layout_mode;
  127. FormattingContext* m_parent { nullptr };
  128. JS::NonnullGCPtr<Box const> m_context_box;
  129. LayoutState& m_state;
  130. };
  131. bool box_is_sized_as_replaced_element(Box const&);
  132. }