FlexFormattingContext.h 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. /*
  2. * Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #pragma once
  7. #include <LibWeb/Layout/Box.h>
  8. #include <LibWeb/Layout/FormattingContext.h>
  9. namespace Web::Layout {
  10. class FlexFormattingContext final : public FormattingContext {
  11. public:
  12. FlexFormattingContext(FormattingState&, Box const& flex_container, FormattingContext* parent);
  13. ~FlexFormattingContext();
  14. virtual bool inhibits_floating() const override { return true; }
  15. virtual void run(Box const&, LayoutMode) override;
  16. Box const& flex_container() const { return context_box(); }
  17. private:
  18. void dump_items() const;
  19. struct DirectionAgnosticMargins {
  20. float main_before { 0 };
  21. float main_after { 0 };
  22. float cross_before { 0 };
  23. float cross_after { 0 };
  24. };
  25. struct FlexItem {
  26. Box& box;
  27. float flex_base_size { 0 };
  28. float hypothetical_main_size { 0 };
  29. float hypothetical_cross_size { 0 };
  30. float hypothetical_cross_size_with_margins() { return hypothetical_cross_size + margins.cross_before + margins.cross_after; }
  31. float target_main_size { 0 };
  32. bool frozen { false };
  33. Optional<float> flex_factor {};
  34. float scaled_flex_shrink_factor { 0 };
  35. float max_content_flex_fraction { 0 };
  36. float main_size { 0 };
  37. float cross_size { 0 };
  38. float main_offset { 0 };
  39. float cross_offset { 0 };
  40. DirectionAgnosticMargins margins {};
  41. bool is_min_violation { false };
  42. bool is_max_violation { false };
  43. };
  44. struct FlexLine {
  45. Vector<FlexItem*> items;
  46. float cross_size { 0 };
  47. };
  48. bool has_definite_main_size(Box const&) const;
  49. bool has_definite_cross_size(Box const&) const;
  50. float specified_main_size(Box const&) const;
  51. float specified_cross_size(Box const&) const;
  52. float resolved_definite_main_size(Box const&) const;
  53. float resolved_definite_cross_size(Box const&) const;
  54. bool has_main_min_size(Box const&) const;
  55. bool has_cross_min_size(Box const&) const;
  56. float specified_main_max_size(Box const&) const;
  57. float specified_cross_max_size(Box const&) const;
  58. float calculated_main_size(Box const&) const;
  59. bool is_cross_auto(Box const&) const;
  60. bool is_main_axis_margin_first_auto(Box const&) const;
  61. bool is_main_axis_margin_second_auto(Box const&) const;
  62. float specified_main_size_of_child_box(Box const& child_box) const;
  63. float specified_main_min_size(Box const&) const;
  64. float specified_cross_min_size(Box const&) const;
  65. bool has_main_max_size(Box const&) const;
  66. bool has_cross_max_size(Box const&) const;
  67. float sum_of_margin_padding_border_in_main_axis(Box const&) const;
  68. void set_main_size(Box const&, float size);
  69. void set_cross_size(Box const&, float size);
  70. void set_offset(Box const&, float main_offset, float cross_offset);
  71. void set_main_axis_first_margin(Box const&, float margin);
  72. void set_main_axis_second_margin(Box const&, float margin);
  73. void copy_dimensions_from_flex_items_to_boxes();
  74. void generate_anonymous_flex_items();
  75. void determine_available_main_and_cross_space(bool& main_is_constrained, bool& cross_is_constrained, float& main_min_size, float& main_max_size, float& cross_min_size, float& cross_max_size);
  76. float calculate_indefinite_main_size(FlexItem const&);
  77. void determine_flex_base_size_and_hypothetical_main_size(FlexItem&);
  78. void determine_main_size_of_flex_container(bool main_is_constrained, float main_min_size, float main_max_size);
  79. void collect_flex_items_into_flex_lines();
  80. void resolve_flexible_lengths();
  81. void determine_hypothetical_cross_size_of_item(FlexItem&);
  82. void calculate_cross_size_of_each_flex_line(float cross_min_size, float cross_max_size);
  83. void determine_used_cross_size_of_each_flex_item();
  84. void distribute_any_remaining_free_space();
  85. void align_all_flex_items_along_the_cross_axis();
  86. void determine_flex_container_used_cross_size(float cross_min_size, float cross_max_size);
  87. void align_all_flex_lines();
  88. bool is_row_layout() const { return m_flex_direction == CSS::FlexDirection::Row || m_flex_direction == CSS::FlexDirection::RowReverse; }
  89. bool is_single_line() const { return flex_container().computed_values().flex_wrap() == CSS::FlexWrap::Nowrap; }
  90. void populate_specified_margins(FlexItem&, CSS::FlexDirection) const;
  91. FormattingState::NodeState& m_flex_container_state;
  92. Vector<FlexLine> m_flex_lines;
  93. Vector<FlexItem> m_flex_items;
  94. CSS::FlexDirection m_flex_direction {};
  95. struct AvailableSpace {
  96. Optional<float> main;
  97. Optional<float> cross;
  98. };
  99. Optional<AvailableSpace> m_available_space;
  100. };
  101. }