FormattingContext.h 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. /*
  2. * Copyright (c) 2020-2022, Andreas Kling <kling@serenityos.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. class FormattingContext {
  13. public:
  14. virtual ~FormattingContext();
  15. enum class Type {
  16. Block,
  17. Inline,
  18. Flex,
  19. Table,
  20. SVG,
  21. };
  22. virtual void run(Box const&, LayoutMode, AvailableSpace const&) = 0;
  23. // This function returns the automatic content height of the context's root box.
  24. virtual float automatic_content_width() const { return 0; }
  25. // This function returns the automatic content height of the context's root box.
  26. virtual float automatic_content_height() const = 0;
  27. Box const& context_box() const { return m_context_box; }
  28. FormattingContext* parent() { return m_parent; }
  29. FormattingContext const* parent() const { return m_parent; }
  30. Type type() const { return m_type; }
  31. bool is_block_formatting_context() const { return type() == Type::Block; }
  32. virtual bool inhibits_floating() const { return false; }
  33. static bool creates_block_formatting_context(Box const&);
  34. static float compute_width_for_replaced_element(LayoutState const&, ReplacedBox const&, AvailableSpace const&);
  35. static float compute_height_for_replaced_element(LayoutState const&, ReplacedBox const&, AvailableSpace const&);
  36. OwnPtr<FormattingContext> create_independent_formatting_context_if_needed(LayoutState&, Box const& child_box);
  37. virtual void parent_context_did_dimension_child_root_box() { }
  38. float calculate_min_content_width(Layout::Box const&) const;
  39. float calculate_max_content_width(Layout::Box const&) const;
  40. float calculate_min_content_height(Layout::Box const&, AvailableSize const& available_width) const;
  41. float calculate_max_content_height(Layout::Box const&, AvailableSize const& available_width) const;
  42. float calculate_fit_content_height(Layout::Box const&, AvailableSpace const&) const;
  43. float calculate_fit_content_width(Layout::Box const&, AvailableSpace const&) const;
  44. virtual float greatest_child_width(Box const&);
  45. float containing_block_width_for(Box const& box) const { return containing_block_width_for(box, m_state); }
  46. float containing_block_height_for(Box const& box) const { return containing_block_height_for(box, m_state); }
  47. static float containing_block_width_for(Box const&, LayoutState const&);
  48. static float containing_block_height_for(Box const&, LayoutState const&);
  49. float compute_box_y_position_with_respect_to_siblings(Box const&) const;
  50. float calculate_stretch_fit_width(Box const&, AvailableSize const&) const;
  51. virtual bool can_determine_size_of_child() const { return false; }
  52. virtual void determine_width_of_child(Box const&, AvailableSpace const&) { }
  53. virtual void determine_height_of_child(Box const&, AvailableSpace const&) { }
  54. virtual Gfx::FloatPoint calculate_static_position(Box const&) const;
  55. protected:
  56. FormattingContext(Type, LayoutState&, Box const&, FormattingContext* parent = nullptr);
  57. float calculate_fit_content_size(float min_content_size, float max_content_size, AvailableSize const&) const;
  58. OwnPtr<FormattingContext> layout_inside(Box const&, LayoutMode, AvailableSpace const&);
  59. void compute_inset(Box const& box);
  60. struct SpaceUsedByFloats {
  61. float left { 0 };
  62. float right { 0 };
  63. };
  64. struct ShrinkToFitResult {
  65. float preferred_width { 0 };
  66. float preferred_minimum_width { 0 };
  67. };
  68. static float tentative_width_for_replaced_element(LayoutState const&, ReplacedBox const&, CSS::Size const& computed_width, AvailableSpace const&);
  69. static float tentative_height_for_replaced_element(LayoutState const&, ReplacedBox const&, CSS::Size const& computed_height, AvailableSpace const&);
  70. float compute_auto_height_for_block_formatting_context_root(BlockContainer const&) const;
  71. float compute_auto_height_for_block_level_element(Box const&, AvailableSpace const&) const;
  72. ShrinkToFitResult calculate_shrink_to_fit_widths(Box const&);
  73. void layout_absolutely_positioned_element(Box const&, AvailableSpace const&);
  74. void compute_width_for_absolutely_positioned_element(Box const&, AvailableSpace const&);
  75. void compute_width_for_absolutely_positioned_non_replaced_element(Box const&, AvailableSpace const&);
  76. void compute_width_for_absolutely_positioned_replaced_element(ReplacedBox const&, AvailableSpace const&);
  77. void compute_height_for_absolutely_positioned_element(Box const&, AvailableSpace const&);
  78. void compute_height_for_absolutely_positioned_non_replaced_element(Box const&, AvailableSpace const&);
  79. void compute_height_for_absolutely_positioned_replaced_element(ReplacedBox const&, AvailableSpace const&);
  80. Type m_type {};
  81. FormattingContext* m_parent { nullptr };
  82. Box const& m_context_box;
  83. LayoutState& m_state;
  84. };
  85. }