FlexFormattingContext.cpp 105 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187
  1. /*
  2. * Copyright (c) 2021-2024, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2021, Tobias Christiansen <tobyase@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include "InlineFormattingContext.h"
  8. #include <AK/Function.h>
  9. #include <AK/QuickSort.h>
  10. #include <AK/StdLibExtras.h>
  11. #include <LibWeb/Layout/BlockContainer.h>
  12. #include <LibWeb/Layout/BlockFormattingContext.h>
  13. #include <LibWeb/Layout/Box.h>
  14. #include <LibWeb/Layout/FlexFormattingContext.h>
  15. #include <LibWeb/Layout/ReplacedBox.h>
  16. #include <LibWeb/Layout/TextNode.h>
  17. #include <LibWeb/Layout/Viewport.h>
  18. namespace Web::Layout {
  19. // NOTE: We use a custom clamping function here instead of AK::clamp(), since the AK version
  20. // will VERIFY(max >= min) and CSS explicitly allows that (see css-values-4.)
  21. template<typename T>
  22. [[nodiscard]] constexpr T css_clamp(T const& value, T const& min, T const& max)
  23. {
  24. return ::max(min, ::min(value, max));
  25. }
  26. CSSPixels FlexFormattingContext::get_pixel_width(Box const& box, CSS::Size const& size) const
  27. {
  28. return calculate_inner_width(box, containing_block_width_as_available_size(box), size);
  29. }
  30. CSSPixels FlexFormattingContext::get_pixel_height(Box const& box, CSS::Size const& size) const
  31. {
  32. return calculate_inner_height(box, containing_block_height_as_available_size(box), size);
  33. }
  34. FlexFormattingContext::FlexFormattingContext(LayoutState& state, Box const& flex_container, FormattingContext* parent)
  35. : FormattingContext(Type::Flex, state, flex_container, parent)
  36. , m_flex_container_state(m_state.get_mutable(flex_container))
  37. , m_flex_direction(flex_container.computed_values().flex_direction())
  38. {
  39. }
  40. FlexFormattingContext::~FlexFormattingContext() = default;
  41. CSSPixels FlexFormattingContext::automatic_content_width() const
  42. {
  43. return m_flex_container_state.content_width();
  44. }
  45. CSSPixels FlexFormattingContext::automatic_content_height() const
  46. {
  47. return m_flex_container_state.content_height();
  48. }
  49. void FlexFormattingContext::run(Box const& run_box, LayoutMode, AvailableSpace const& available_space)
  50. {
  51. VERIFY(&run_box == &flex_container());
  52. // This implements https://www.w3.org/TR/css-flexbox-1/#layout-algorithm
  53. // 1. Generate anonymous flex items
  54. generate_anonymous_flex_items();
  55. // 2. Determine the available main and cross space for the flex items
  56. determine_available_space_for_items(available_space);
  57. {
  58. // https://drafts.csswg.org/css-flexbox-1/#definite-sizes
  59. // 3. If a single-line flex container has a definite cross size,
  60. // the automatic preferred outer cross size of any stretched flex items is the flex container’s inner cross size
  61. // (clamped to the flex item’s min and max cross size) and is considered definite.
  62. if (is_single_line() && has_definite_cross_size(m_flex_container_state)) {
  63. auto flex_container_inner_cross_size = inner_cross_size(m_flex_container_state);
  64. for (auto& item : m_flex_items) {
  65. if (!flex_item_is_stretched(item))
  66. continue;
  67. auto item_min_cross_size = has_cross_min_size(item.box) ? specified_cross_min_size(item.box) : 0;
  68. auto item_max_cross_size = has_cross_max_size(item.box) ? specified_cross_max_size(item.box) : CSSPixels::max();
  69. auto item_preferred_outer_cross_size = css_clamp(flex_container_inner_cross_size, item_min_cross_size, item_max_cross_size);
  70. auto item_inner_cross_size = item_preferred_outer_cross_size - item.margins.cross_before - item.margins.cross_after - item.padding.cross_before - item.padding.cross_after - item.borders.cross_before - item.borders.cross_after;
  71. set_cross_size(item.box, item_inner_cross_size);
  72. }
  73. }
  74. }
  75. // 3. Determine the flex base size and hypothetical main size of each item
  76. for (auto& item : m_flex_items) {
  77. if (item.box->is_replaced_box()) {
  78. // FIXME: Get rid of prepare_for_replaced_layout() and make replaced elements figure out their intrinsic size lazily.
  79. static_cast<ReplacedBox&>(*item.box).prepare_for_replaced_layout();
  80. }
  81. determine_flex_base_size_and_hypothetical_main_size(item);
  82. }
  83. if (available_space.width.is_intrinsic_sizing_constraint() || available_space.height.is_intrinsic_sizing_constraint()) {
  84. // We're computing intrinsic size for the flex container. This happens at the end of run().
  85. } else {
  86. // 4. Determine the main size of the flex container
  87. // Determine the main size of the flex container using the rules of the formatting context in which it participates.
  88. // NOTE: The automatic block size of a block-level flex container is its max-content size.
  89. // NOTE: We've already handled this in the parent formatting context.
  90. // Specifically, all formatting contexts will have assigned width & height to the flex container
  91. // before this formatting context runs.
  92. }
  93. // 5. Collect flex items into flex lines:
  94. // After this step no additional items are to be added to flex_lines or any of its items!
  95. collect_flex_items_into_flex_lines();
  96. // 6. Resolve the flexible lengths
  97. resolve_flexible_lengths();
  98. // Cross Size Determination
  99. // 7. Determine the hypothetical cross size of each item
  100. for (auto& item : m_flex_items) {
  101. determine_hypothetical_cross_size_of_item(item, false);
  102. }
  103. // 8. Calculate the cross size of each flex line.
  104. calculate_cross_size_of_each_flex_line();
  105. // 9. Handle 'align-content: stretch'.
  106. handle_align_content_stretch();
  107. // 10. Collapse visibility:collapse items.
  108. // FIXME: This
  109. // 11. Determine the used cross size of each flex item.
  110. determine_used_cross_size_of_each_flex_item();
  111. // 12. Distribute any remaining free space.
  112. distribute_any_remaining_free_space();
  113. // 13. Resolve cross-axis auto margins.
  114. resolve_cross_axis_auto_margins();
  115. // 14. Align all flex items along the cross-axis
  116. align_all_flex_items_along_the_cross_axis();
  117. // 15. Determine the flex container’s used cross size:
  118. determine_flex_container_used_cross_size();
  119. {
  120. // https://drafts.csswg.org/css-flexbox-1/#definite-sizes
  121. // 4. Once the cross size of a flex line has been determined,
  122. // the cross sizes of items in auto-sized flex containers are also considered definite for the purpose of layout.
  123. auto const& flex_container_computed_cross_size = is_row_layout() ? flex_container().computed_values().height() : flex_container().computed_values().width();
  124. if (flex_container_computed_cross_size.is_auto()) {
  125. for (auto& item : m_flex_items) {
  126. set_cross_size(item.box, item.cross_size.value());
  127. }
  128. }
  129. }
  130. {
  131. // NOTE: We re-resolve cross sizes here, now that we can resolve percentages.
  132. // 7. Determine the hypothetical cross size of each item
  133. for (auto& item : m_flex_items) {
  134. determine_hypothetical_cross_size_of_item(item, true);
  135. }
  136. // 11. Determine the used cross size of each flex item.
  137. determine_used_cross_size_of_each_flex_item();
  138. }
  139. // 16. Align all flex lines (per align-content)
  140. align_all_flex_lines();
  141. if (available_space.width.is_intrinsic_sizing_constraint() || available_space.height.is_intrinsic_sizing_constraint()) {
  142. // We're computing intrinsic size for the flex container.
  143. determine_intrinsic_size_of_flex_container();
  144. } else {
  145. // This is a normal layout (not intrinsic sizing).
  146. // AD-HOC: Finally, layout the inside of all flex items.
  147. copy_dimensions_from_flex_items_to_boxes();
  148. for (auto& item : m_flex_items) {
  149. if (auto independent_formatting_context = layout_inside(item.box, LayoutMode::Normal, item.used_values.available_inner_space_or_constraints_from(m_available_space_for_items->space)))
  150. independent_formatting_context->parent_context_did_dimension_child_root_box();
  151. compute_inset(item.box);
  152. }
  153. }
  154. }
  155. void FlexFormattingContext::parent_context_did_dimension_child_root_box()
  156. {
  157. flex_container().for_each_child_of_type<Box>([&](Layout::Box& box) {
  158. if (box.is_absolutely_positioned()) {
  159. auto& cb_state = m_state.get(*box.containing_block());
  160. auto available_width = AvailableSize::make_definite(cb_state.content_width() + cb_state.padding_left + cb_state.padding_right);
  161. auto available_height = AvailableSize::make_definite(cb_state.content_height() + cb_state.padding_top + cb_state.padding_bottom);
  162. layout_absolutely_positioned_element(box, AvailableSpace(available_width, available_height));
  163. }
  164. });
  165. }
  166. void FlexFormattingContext::populate_specified_margins(FlexItem& item, CSS::FlexDirection flex_direction) const
  167. {
  168. auto width_of_containing_block = m_flex_container_state.content_width();
  169. // FIXME: This should also take reverse-ness into account
  170. if (flex_direction == CSS::FlexDirection::Row || flex_direction == CSS::FlexDirection::RowReverse) {
  171. item.borders.main_before = item.box->computed_values().border_left().width;
  172. item.borders.main_after = item.box->computed_values().border_right().width;
  173. item.borders.cross_before = item.box->computed_values().border_top().width;
  174. item.borders.cross_after = item.box->computed_values().border_bottom().width;
  175. item.padding.main_before = item.box->computed_values().padding().left().to_px(item.box, width_of_containing_block);
  176. item.padding.main_after = item.box->computed_values().padding().right().to_px(item.box, width_of_containing_block);
  177. item.padding.cross_before = item.box->computed_values().padding().top().to_px(item.box, width_of_containing_block);
  178. item.padding.cross_after = item.box->computed_values().padding().bottom().to_px(item.box, width_of_containing_block);
  179. item.margins.main_before = item.box->computed_values().margin().left().to_px(item.box, width_of_containing_block);
  180. item.margins.main_after = item.box->computed_values().margin().right().to_px(item.box, width_of_containing_block);
  181. item.margins.cross_before = item.box->computed_values().margin().top().to_px(item.box, width_of_containing_block);
  182. item.margins.cross_after = item.box->computed_values().margin().bottom().to_px(item.box, width_of_containing_block);
  183. item.margins.main_before_is_auto = item.box->computed_values().margin().left().is_auto();
  184. item.margins.main_after_is_auto = item.box->computed_values().margin().right().is_auto();
  185. item.margins.cross_before_is_auto = item.box->computed_values().margin().top().is_auto();
  186. item.margins.cross_after_is_auto = item.box->computed_values().margin().bottom().is_auto();
  187. } else {
  188. item.borders.main_before = item.box->computed_values().border_top().width;
  189. item.borders.main_after = item.box->computed_values().border_bottom().width;
  190. item.borders.cross_before = item.box->computed_values().border_left().width;
  191. item.borders.cross_after = item.box->computed_values().border_right().width;
  192. item.padding.main_before = item.box->computed_values().padding().top().to_px(item.box, width_of_containing_block);
  193. item.padding.main_after = item.box->computed_values().padding().bottom().to_px(item.box, width_of_containing_block);
  194. item.padding.cross_before = item.box->computed_values().padding().left().to_px(item.box, width_of_containing_block);
  195. item.padding.cross_after = item.box->computed_values().padding().right().to_px(item.box, width_of_containing_block);
  196. item.margins.main_before = item.box->computed_values().margin().top().to_px(item.box, width_of_containing_block);
  197. item.margins.main_after = item.box->computed_values().margin().bottom().to_px(item.box, width_of_containing_block);
  198. item.margins.cross_before = item.box->computed_values().margin().left().to_px(item.box, width_of_containing_block);
  199. item.margins.cross_after = item.box->computed_values().margin().right().to_px(item.box, width_of_containing_block);
  200. item.margins.main_before_is_auto = item.box->computed_values().margin().top().is_auto();
  201. item.margins.main_after_is_auto = item.box->computed_values().margin().bottom().is_auto();
  202. item.margins.cross_before_is_auto = item.box->computed_values().margin().left().is_auto();
  203. item.margins.cross_after_is_auto = item.box->computed_values().margin().right().is_auto();
  204. }
  205. }
  206. // https://www.w3.org/TR/css-flexbox-1/#flex-items
  207. void FlexFormattingContext::generate_anonymous_flex_items()
  208. {
  209. // More like, sift through the already generated items.
  210. // After this step no items are to be added or removed from flex_items!
  211. // It holds every item we need to consider and there should be nothing in the following
  212. // calculations that could change that.
  213. // This is particularly important since we take references to the items stored in flex_items
  214. // later, whose addresses won't be stable if we added or removed any items.
  215. HashMap<int, Vector<FlexItem>> order_item_bucket;
  216. flex_container().for_each_child_of_type<Box>([&](Box& child_box) {
  217. if (can_skip_is_anonymous_text_run(child_box))
  218. return IterationDecision::Continue;
  219. // Skip any "out-of-flow" children
  220. if (child_box.is_out_of_flow(*this))
  221. return IterationDecision::Continue;
  222. child_box.set_flex_item(true);
  223. FlexItem item = { child_box, m_state.get_mutable(child_box) };
  224. populate_specified_margins(item, m_flex_direction);
  225. auto& order_bucket = order_item_bucket.ensure(child_box.computed_values().order());
  226. order_bucket.append(move(item));
  227. return IterationDecision::Continue;
  228. });
  229. auto keys = order_item_bucket.keys();
  230. if (is_direction_reverse()) {
  231. quick_sort(keys, [](auto& a, auto& b) { return a > b; });
  232. } else {
  233. quick_sort(keys, [](auto& a, auto& b) { return a < b; });
  234. }
  235. for (auto key : keys) {
  236. auto order_bucket = order_item_bucket.get(key);
  237. if (order_bucket.has_value()) {
  238. auto& items = order_bucket.value();
  239. if (is_direction_reverse()) {
  240. for (auto item : items.in_reverse()) {
  241. m_flex_items.append(move(item));
  242. }
  243. } else {
  244. for (auto item : items) {
  245. m_flex_items.append(move(item));
  246. }
  247. }
  248. }
  249. }
  250. }
  251. bool FlexFormattingContext::has_definite_main_size(LayoutState::UsedValues const& used_values) const
  252. {
  253. return is_row_layout() ? used_values.has_definite_width() : used_values.has_definite_height();
  254. }
  255. CSSPixels FlexFormattingContext::inner_main_size(LayoutState::UsedValues const& used_values) const
  256. {
  257. return is_row_layout() ? used_values.content_width() : used_values.content_height();
  258. }
  259. CSSPixels FlexFormattingContext::inner_cross_size(LayoutState::UsedValues const& used_values) const
  260. {
  261. return is_row_layout() ? used_values.content_height() : used_values.content_width();
  262. }
  263. bool FlexFormattingContext::has_main_min_size(Box const& box) const
  264. {
  265. auto const& value = is_row_layout() ? box.computed_values().min_width() : box.computed_values().min_height();
  266. return !value.is_auto();
  267. }
  268. bool FlexFormattingContext::has_cross_min_size(Box const& box) const
  269. {
  270. auto const& value = is_row_layout() ? box.computed_values().min_height() : box.computed_values().min_width();
  271. return !value.is_auto();
  272. }
  273. bool FlexFormattingContext::has_definite_cross_size(LayoutState::UsedValues const& used_values) const
  274. {
  275. return is_row_layout() ? used_values.has_definite_height() : used_values.has_definite_width();
  276. }
  277. CSSPixels FlexFormattingContext::specified_main_min_size(Box const& box) const
  278. {
  279. return is_row_layout()
  280. ? get_pixel_width(box, box.computed_values().min_width())
  281. : get_pixel_height(box, box.computed_values().min_height());
  282. }
  283. CSSPixels FlexFormattingContext::specified_cross_min_size(Box const& box) const
  284. {
  285. return is_row_layout()
  286. ? get_pixel_height(box, box.computed_values().min_height())
  287. : get_pixel_width(box, box.computed_values().min_width());
  288. }
  289. bool FlexFormattingContext::has_main_max_size(Box const& box) const
  290. {
  291. return !should_treat_main_max_size_as_none(box);
  292. }
  293. bool FlexFormattingContext::has_cross_max_size(Box const& box) const
  294. {
  295. return !should_treat_cross_max_size_as_none(box);
  296. }
  297. CSSPixels FlexFormattingContext::specified_main_max_size(Box const& box) const
  298. {
  299. return is_row_layout()
  300. ? get_pixel_width(box, box.computed_values().max_width())
  301. : get_pixel_height(box, box.computed_values().max_height());
  302. }
  303. CSSPixels FlexFormattingContext::specified_cross_max_size(Box const& box) const
  304. {
  305. return is_row_layout()
  306. ? get_pixel_height(box, box.computed_values().max_height())
  307. : get_pixel_width(box, box.computed_values().max_width());
  308. }
  309. bool FlexFormattingContext::is_cross_auto(Box const& box) const
  310. {
  311. auto& cross_length = is_row_layout() ? box.computed_values().height() : box.computed_values().width();
  312. return cross_length.is_auto();
  313. }
  314. void FlexFormattingContext::set_main_size(Box const& box, CSSPixels size)
  315. {
  316. if (is_row_layout())
  317. m_state.get_mutable(box).set_content_width(size);
  318. else
  319. m_state.get_mutable(box).set_content_height(size);
  320. }
  321. void FlexFormattingContext::set_cross_size(Box const& box, CSSPixels size)
  322. {
  323. if (is_row_layout())
  324. m_state.get_mutable(box).set_content_height(size);
  325. else
  326. m_state.get_mutable(box).set_content_width(size);
  327. }
  328. void FlexFormattingContext::set_offset(Box const& box, CSSPixels main_offset, CSSPixels cross_offset)
  329. {
  330. if (is_row_layout())
  331. m_state.get_mutable(box).offset = CSSPixelPoint { main_offset, cross_offset };
  332. else
  333. m_state.get_mutable(box).offset = CSSPixelPoint { cross_offset, main_offset };
  334. }
  335. void FlexFormattingContext::set_main_axis_first_margin(FlexItem& item, CSSPixels margin)
  336. {
  337. item.margins.main_before = margin;
  338. if (is_row_layout())
  339. m_state.get_mutable(item.box).margin_left = margin;
  340. else
  341. m_state.get_mutable(item.box).margin_top = margin;
  342. }
  343. void FlexFormattingContext::set_main_axis_second_margin(FlexItem& item, CSSPixels margin)
  344. {
  345. item.margins.main_after = margin;
  346. if (is_row_layout())
  347. m_state.get_mutable(item.box).margin_right = margin;
  348. else
  349. m_state.get_mutable(item.box).margin_bottom = margin;
  350. }
  351. // https://drafts.csswg.org/css-flexbox-1/#algo-available
  352. void FlexFormattingContext::determine_available_space_for_items(AvailableSpace const& available_space)
  353. {
  354. if (is_row_layout()) {
  355. m_available_space_for_items = AxisAgnosticAvailableSpace {
  356. .main = available_space.width,
  357. .cross = available_space.height,
  358. .space = { available_space.width, available_space.height },
  359. };
  360. } else {
  361. m_available_space_for_items = AxisAgnosticAvailableSpace {
  362. .main = available_space.height,
  363. .cross = available_space.width,
  364. .space = { available_space.width, available_space.height },
  365. };
  366. }
  367. }
  368. // https://drafts.csswg.org/css-flexbox-1/#propdef-flex-basis
  369. CSS::FlexBasis FlexFormattingContext::used_flex_basis_for_item(FlexItem const& item) const
  370. {
  371. auto flex_basis = item.box->computed_values().flex_basis();
  372. if (flex_basis.has<CSS::Size>() && flex_basis.get<CSS::Size>().is_auto()) {
  373. // https://drafts.csswg.org/css-flexbox-1/#valdef-flex-basis-auto
  374. // When specified on a flex item, the auto keyword retrieves the value of the main size property as the used flex-basis.
  375. // If that value is itself auto, then the used value is content.
  376. auto const& main_size = is_row_layout() ? item.box->computed_values().width() : item.box->computed_values().height();
  377. if (main_size.is_auto()) {
  378. flex_basis = CSS::FlexBasisContent {};
  379. } else {
  380. flex_basis = main_size;
  381. }
  382. }
  383. // For example, percentage values of flex-basis are resolved against the flex item’s containing block
  384. // (i.e. its flex container); and if that containing block’s size is indefinite,
  385. // the used value for flex-basis is content.
  386. if (flex_basis.has<CSS::Size>()
  387. && flex_basis.get<CSS::Size>().is_percentage()
  388. && !has_definite_main_size(m_flex_container_state)) {
  389. flex_basis = CSS::FlexBasisContent {};
  390. }
  391. return flex_basis;
  392. }
  393. CSSPixels FlexFormattingContext::calculate_main_size_from_cross_size_and_aspect_ratio(CSSPixels cross_size, CSSPixelFraction aspect_ratio) const
  394. {
  395. if (is_row_layout())
  396. return cross_size * aspect_ratio;
  397. return cross_size / aspect_ratio;
  398. }
  399. CSSPixels FlexFormattingContext::calculate_cross_size_from_main_size_and_aspect_ratio(CSSPixels main_size, CSSPixelFraction aspect_ratio) const
  400. {
  401. if (is_row_layout())
  402. return main_size / aspect_ratio;
  403. return main_size * aspect_ratio;
  404. }
  405. // This function takes a size in the main axis and adjusts it according to the aspect ratio of the box
  406. // if the min/max constraints in the cross axis forces us to come up with a new main axis size.
  407. CSSPixels FlexFormattingContext::adjust_main_size_through_aspect_ratio_for_cross_size_min_max_constraints(Box const& box, CSSPixels main_size, CSS::Size const& min_cross_size, CSS::Size const& max_cross_size) const
  408. {
  409. if (!should_treat_cross_max_size_as_none(box)) {
  410. auto max_cross_size_px = max_cross_size.to_px(box, !is_row_layout() ? m_flex_container_state.content_width() : m_flex_container_state.content_height());
  411. main_size = min(main_size, calculate_main_size_from_cross_size_and_aspect_ratio(max_cross_size_px, box.preferred_aspect_ratio().value()));
  412. }
  413. if (!min_cross_size.is_auto()) {
  414. auto min_cross_size_px = min_cross_size.to_px(box, !is_row_layout() ? m_flex_container_state.content_width() : m_flex_container_state.content_height());
  415. main_size = max(main_size, calculate_main_size_from_cross_size_and_aspect_ratio(min_cross_size_px, box.preferred_aspect_ratio().value()));
  416. }
  417. return main_size;
  418. }
  419. // https://www.w3.org/TR/css-flexbox-1/#algo-main-item
  420. void FlexFormattingContext::determine_flex_base_size_and_hypothetical_main_size(FlexItem& item)
  421. {
  422. auto& child_box = item.box;
  423. item.flex_base_size = [&] {
  424. item.used_flex_basis = used_flex_basis_for_item(item);
  425. item.used_flex_basis_is_definite = [&](CSS::FlexBasis const& flex_basis) -> bool {
  426. if (!flex_basis.has<CSS::Size>())
  427. return false;
  428. auto const& size = flex_basis.get<CSS::Size>();
  429. if (size.is_auto() || size.is_min_content() || size.is_max_content() || size.is_fit_content())
  430. return false;
  431. if (size.is_length())
  432. return true;
  433. bool can_resolve_percentages = is_row_layout()
  434. ? m_flex_container_state.has_definite_width()
  435. : m_flex_container_state.has_definite_height();
  436. if (size.is_calculated()) {
  437. auto const& calc_value = size.calculated();
  438. if (calc_value.resolves_to_percentage())
  439. return can_resolve_percentages;
  440. if (calc_value.resolves_to_length()) {
  441. if (calc_value.contains_percentage())
  442. return can_resolve_percentages;
  443. return true;
  444. }
  445. return false;
  446. }
  447. VERIFY(size.is_percentage());
  448. return can_resolve_percentages;
  449. }(*item.used_flex_basis);
  450. // A. If the item has a definite used flex basis, that’s the flex base size.
  451. if (item.used_flex_basis_is_definite) {
  452. auto const& size = item.used_flex_basis->get<CSS::Size>();
  453. if (is_row_layout())
  454. return get_pixel_width(child_box, size);
  455. return get_pixel_height(child_box, size);
  456. }
  457. // B. If the flex item has ...
  458. // - an intrinsic aspect ratio,
  459. // - a used flex basis of content, and
  460. // - a definite cross size,
  461. if (item.box->has_preferred_aspect_ratio()
  462. && item.used_flex_basis->has<CSS::FlexBasisContent>()
  463. && has_definite_cross_size(item)) {
  464. // flex_base_size is calculated from definite cross size and intrinsic aspect ratio
  465. return adjust_main_size_through_aspect_ratio_for_cross_size_min_max_constraints(
  466. item.box,
  467. calculate_main_size_from_cross_size_and_aspect_ratio(inner_cross_size(item), item.box->preferred_aspect_ratio().value()),
  468. computed_cross_min_size(item.box),
  469. computed_cross_max_size(item.box));
  470. }
  471. // C. If the used flex basis is content or depends on its available space,
  472. // and the flex container is being sized under a min-content or max-content constraint
  473. // (e.g. when performing automatic table layout [CSS21]), size the item under that constraint.
  474. // The flex base size is the item’s resulting main size.
  475. if (item.used_flex_basis->has<CSS::FlexBasisContent>() && m_available_space_for_items->main.is_intrinsic_sizing_constraint()) {
  476. if (m_available_space_for_items->main.is_min_content())
  477. return calculate_min_content_main_size(item);
  478. return calculate_max_content_main_size(item);
  479. }
  480. // D. Otherwise, if the used flex basis is content or depends on its available space,
  481. // the available main size is infinite, and the flex item’s inline axis is parallel to the main axis,
  482. // lay the item out using the rules for a box in an orthogonal flow [CSS3-WRITING-MODES].
  483. // The flex base size is the item’s max-content main size.
  484. if (item.used_flex_basis->has<CSS::FlexBasisContent>()
  485. // FIXME: && main_size is infinite && inline axis is parallel to the main axis
  486. && false && false) {
  487. TODO();
  488. // Use rules for a flex_container in orthogonal flow
  489. }
  490. // E. Otherwise, size the item into the available space using its used flex basis in place of its main size,
  491. // treating a value of content as max-content. If a cross size is needed to determine the main size
  492. // (e.g. when the flex item’s main size is in its block axis) and the flex item’s cross size is auto and not definite,
  493. // in this calculation use fit-content as the flex item’s cross size.
  494. // The flex base size is the item’s resulting main size.
  495. // NOTE: If the flex item has a definite main size, just use that as the flex base size.
  496. if (has_definite_main_size(item))
  497. return inner_main_size(item);
  498. // NOTE: There's a fundamental problem with many CSS specifications in that they neglect to mention
  499. // which width to provide when calculating the intrinsic height of a box in various situations.
  500. // Spec bug: https://github.com/w3c/csswg-drafts/issues/2890
  501. // NOTE: This is one of many situations where that causes trouble: if this is a flex column layout,
  502. // we may need to calculate the intrinsic height of a flex item. This requires a width, but a
  503. // width won't be determined until later on in the flex layout algorithm.
  504. // In the specific case above (E), the spec mentions using `fit-content` in place of `auto`
  505. // if "a cross size is needed to determine the main size", so that's exactly what we do.
  506. // NOTE: Finding a suitable width for intrinsic height determination actually happens elsewhere,
  507. // in the various helpers that calculate the intrinsic sizes of a flex item,
  508. // e.g. calculate_min_content_main_size().
  509. if (item.used_flex_basis->has<CSS::FlexBasisContent>()) {
  510. return calculate_max_content_main_size(item);
  511. }
  512. return calculate_fit_content_main_size(item);
  513. }();
  514. // AD-HOC: This is not mentioned in the spec, but if the item has an aspect ratio,
  515. // we may need to adjust the main size in response to cross size min/max constraints.
  516. if (item.box->has_preferred_aspect_ratio()) {
  517. item.flex_base_size = adjust_main_size_through_aspect_ratio_for_cross_size_min_max_constraints(child_box, item.flex_base_size, computed_cross_min_size(child_box), computed_cross_max_size(child_box));
  518. }
  519. // The hypothetical main size is the item’s flex base size clamped according to its used min and max main sizes (and flooring the content box size at zero).
  520. auto clamp_min = has_main_min_size(child_box) ? specified_main_min_size(child_box) : automatic_minimum_size(item);
  521. auto clamp_max = has_main_max_size(child_box) ? specified_main_max_size(child_box) : CSSPixels::max();
  522. item.hypothetical_main_size = max(CSSPixels(0), css_clamp(item.flex_base_size, clamp_min, clamp_max));
  523. // NOTE: At this point, we set the hypothetical main size as the flex item's *temporary* main size.
  524. // The size may change again when we resolve flexible lengths, but this is necessary in order for
  525. // descendants of this flex item to resolve percentage sizes against something.
  526. //
  527. // The spec just barely hand-waves about this, but it seems to *roughly* match what other engines do.
  528. // See "Note" section here: https://drafts.csswg.org/css-flexbox-1/#definite-sizes
  529. if (is_row_layout())
  530. item.used_values.set_temporary_content_width(item.hypothetical_main_size);
  531. else
  532. item.used_values.set_temporary_content_height(item.hypothetical_main_size);
  533. }
  534. // https://drafts.csswg.org/css-flexbox-1/#min-size-auto
  535. CSSPixels FlexFormattingContext::automatic_minimum_size(FlexItem const& item) const
  536. {
  537. // To provide a more reasonable default minimum size for flex items,
  538. // the used value of a main axis automatic minimum size on a flex item that is not a scroll container is its content-based minimum size;
  539. // for scroll containers the automatic minimum size is zero, as usual.
  540. if (!item.box->is_scroll_container())
  541. return content_based_minimum_size(item);
  542. return 0;
  543. }
  544. // https://drafts.csswg.org/css-flexbox-1/#specified-size-suggestion
  545. Optional<CSSPixels> FlexFormattingContext::specified_size_suggestion(FlexItem const& item) const
  546. {
  547. // If the item’s preferred main size is definite and not automatic,
  548. // then the specified size suggestion is that size. It is otherwise undefined.
  549. if (has_definite_main_size(item) && !should_treat_main_size_as_auto(item.box)) {
  550. // NOTE: We use get_pixel_{width,height} to ensure that CSS box-sizing is respected.
  551. return is_row_layout() ? get_pixel_width(item.box, computed_main_size(item.box)) : get_pixel_height(item.box, computed_main_size(item.box));
  552. }
  553. return {};
  554. }
  555. // https://drafts.csswg.org/css-flexbox-1/#content-size-suggestion
  556. CSSPixels FlexFormattingContext::content_size_suggestion(FlexItem const& item) const
  557. {
  558. auto suggestion = calculate_min_content_main_size(item);
  559. if (item.box->has_preferred_aspect_ratio()) {
  560. suggestion = adjust_main_size_through_aspect_ratio_for_cross_size_min_max_constraints(item.box, suggestion, computed_cross_min_size(item.box), computed_cross_max_size(item.box));
  561. }
  562. return suggestion;
  563. }
  564. // https://drafts.csswg.org/css-flexbox-1/#transferred-size-suggestion
  565. Optional<CSSPixels> FlexFormattingContext::transferred_size_suggestion(FlexItem const& item) const
  566. {
  567. // If the item has a preferred aspect ratio and its preferred cross size is definite,
  568. // then the transferred size suggestion is that size
  569. // (clamped by its minimum and maximum cross sizes if they are definite), converted through the aspect ratio.
  570. if (item.box->has_preferred_aspect_ratio() && has_definite_cross_size(item)) {
  571. auto aspect_ratio = item.box->preferred_aspect_ratio().value();
  572. return adjust_main_size_through_aspect_ratio_for_cross_size_min_max_constraints(
  573. item.box,
  574. calculate_main_size_from_cross_size_and_aspect_ratio(inner_cross_size(item), aspect_ratio),
  575. computed_cross_min_size(item.box),
  576. computed_cross_max_size(item.box));
  577. }
  578. // It is otherwise undefined.
  579. return {};
  580. }
  581. // https://drafts.csswg.org/css-flexbox-1/#content-based-minimum-size
  582. CSSPixels FlexFormattingContext::content_based_minimum_size(FlexItem const& item) const
  583. {
  584. auto unclamped_size = [&] {
  585. // The content-based minimum size of a flex item is the smaller of its specified size suggestion
  586. // and its content size suggestion if its specified size suggestion exists;
  587. if (auto specified_size_suggestion = this->specified_size_suggestion(item); specified_size_suggestion.has_value()) {
  588. return min(specified_size_suggestion.value(), content_size_suggestion(item));
  589. }
  590. // otherwise, the smaller of its transferred size suggestion and its content size suggestion
  591. // if the element is replaced and its transferred size suggestion exists;
  592. if (item.box->is_replaced_box()) {
  593. if (auto transferred_size_suggestion = this->transferred_size_suggestion(item); transferred_size_suggestion.has_value()) {
  594. return min(transferred_size_suggestion.value(), content_size_suggestion(item));
  595. }
  596. }
  597. // otherwise its content size suggestion.
  598. return content_size_suggestion(item);
  599. }();
  600. // In all cases, the size is clamped by the maximum main size if it’s definite.
  601. if (has_main_max_size(item.box)) {
  602. return min(unclamped_size, specified_main_max_size(item.box));
  603. }
  604. return unclamped_size;
  605. }
  606. // https://www.w3.org/TR/css-flexbox-1/#algo-line-break
  607. void FlexFormattingContext::collect_flex_items_into_flex_lines()
  608. {
  609. // FIXME: Also support wrap-reverse
  610. // If the flex container is single-line, collect all the flex items into a single flex line.
  611. if (is_single_line()) {
  612. FlexLine line;
  613. for (auto& item : m_flex_items) {
  614. line.items.append(item);
  615. }
  616. m_flex_lines.append(move(line));
  617. return;
  618. }
  619. // Otherwise, starting from the first uncollected item, collect consecutive items one by one
  620. // until the first time that the next collected item would not fit into the flex container’s inner main size
  621. // (or until a forced break is encountered, see §10 Fragmenting Flex Layout).
  622. // If the very first uncollected item wouldn't fit, collect just it into the line.
  623. // For this step, the size of a flex item is its outer hypothetical main size. (Note: This can be negative.)
  624. // Repeat until all flex items have been collected into flex lines.
  625. FlexLine line;
  626. CSSPixels line_main_size = 0;
  627. for (auto& item : m_flex_items) {
  628. auto const outer_hypothetical_main_size = item.outer_hypothetical_main_size();
  629. if (!line.items.is_empty() && (line_main_size + outer_hypothetical_main_size) > m_available_space_for_items->main) {
  630. m_flex_lines.append(move(line));
  631. line = {};
  632. line_main_size = 0;
  633. }
  634. line.items.append(item);
  635. line_main_size += outer_hypothetical_main_size;
  636. // CSS-FLEXBOX-2: Account for gap between flex items.
  637. line_main_size += main_gap();
  638. }
  639. m_flex_lines.append(move(line));
  640. }
  641. // https://drafts.csswg.org/css-flexbox-1/#resolve-flexible-lengths
  642. void FlexFormattingContext::resolve_flexible_lengths_for_line(FlexLine& line)
  643. {
  644. // AD-HOC: The spec tells us to use the "flex container’s inner main size" in this algorithm,
  645. // but that doesn't work when we're sizing under a max-content constraint.
  646. // In that case, there is effectively infinite size available in the main axis,
  647. // but the inner main size has not been assigned yet.
  648. // We solve this by calculating our own "available main size" here, which is essentially
  649. // infinity under max-content, 0 under min-content, and the inner main size otherwise.
  650. AvailableSize available_main_size { AvailableSize::make_indefinite() };
  651. if (m_available_space_for_items->main.is_intrinsic_sizing_constraint())
  652. available_main_size = m_available_space_for_items->main;
  653. else
  654. available_main_size = AvailableSize::make_definite(inner_main_size(m_flex_container_state));
  655. // 1. Determine the used flex factor.
  656. // Sum the outer hypothetical main sizes of all items on the line.
  657. // If the sum is less than the flex container’s inner main size,
  658. // use the flex grow factor for the rest of this algorithm; otherwise, use the flex shrink factor
  659. enum FlexFactor {
  660. FlexGrowFactor,
  661. FlexShrinkFactor
  662. };
  663. auto used_flex_factor = [&]() -> FlexFactor {
  664. CSSPixels sum = 0;
  665. for (auto const& item : line.items) {
  666. sum += item.outer_hypothetical_main_size();
  667. }
  668. // CSS-FLEXBOX-2: Account for gap between flex items.
  669. sum += main_gap() * (line.items.size() - 1);
  670. // AD-HOC: Note that we're using our own "available main size" explained above
  671. // instead of the flex container’s inner main size.
  672. if (sum < available_main_size)
  673. return FlexFactor::FlexGrowFactor;
  674. return FlexFactor::FlexShrinkFactor;
  675. }();
  676. // 2. Each item in the flex line has a target main size, initially set to its flex base size.
  677. // Each item is initially unfrozen and may become frozen.
  678. for (auto& item : line.items) {
  679. item.target_main_size = item.flex_base_size;
  680. item.frozen = false;
  681. }
  682. // 3. Size inflexible items.
  683. for (FlexItem& item : line.items) {
  684. if (used_flex_factor == FlexFactor::FlexGrowFactor) {
  685. item.flex_factor = item.box->computed_values().flex_grow();
  686. } else if (used_flex_factor == FlexFactor::FlexShrinkFactor) {
  687. item.flex_factor = item.box->computed_values().flex_shrink();
  688. }
  689. // Freeze, setting its target main size to its hypothetical main size…
  690. // - any item that has a flex factor of zero
  691. // - if using the flex grow factor: any item that has a flex base size greater than its hypothetical main size
  692. // - if using the flex shrink factor: any item that has a flex base size smaller than its hypothetical main size
  693. if (item.flex_factor.value() == 0
  694. || (used_flex_factor == FlexFactor::FlexGrowFactor && item.flex_base_size > item.hypothetical_main_size)
  695. || (used_flex_factor == FlexFactor::FlexShrinkFactor && item.flex_base_size < item.hypothetical_main_size)) {
  696. item.frozen = true;
  697. item.target_main_size = item.hypothetical_main_size;
  698. }
  699. }
  700. // 4. Calculate initial free space
  701. // Sum the outer sizes of all items on the line, and subtract this from the flex container’s inner main size.
  702. // For frozen items, use their outer target main size; for other items, use their outer flex base size.
  703. auto calculate_remaining_free_space = [&]() -> Optional<CSSPixels> {
  704. // AD-HOC: If the container is sized under max-content constraints, then remaining_free_space won't have
  705. // a value to avoid leaking an infinite value into layout calculations.
  706. if (available_main_size.is_intrinsic_sizing_constraint())
  707. return {};
  708. CSSPixels sum = 0;
  709. for (auto const& item : line.items) {
  710. if (item.frozen)
  711. sum += item.outer_target_main_size();
  712. else
  713. sum += item.outer_flex_base_size();
  714. }
  715. // CSS-FLEXBOX-2: Account for gap between flex items.
  716. sum += main_gap() * (line.items.size() - 1);
  717. // AD-HOC: Note that we're using our own "available main size" explained above
  718. // instead of the flex container’s inner main size.
  719. return available_main_size.to_px_or_zero() - sum;
  720. };
  721. auto const initial_free_space = calculate_remaining_free_space();
  722. // 5. Loop
  723. while (true) {
  724. // a. Check for flexible items.
  725. // If all the flex items on the line are frozen, free space has been distributed; exit this loop.
  726. if (all_of(line.items, [](auto const& item) { return item.frozen; })) {
  727. break;
  728. }
  729. // b. Calculate the remaining free space as for initial free space, above.
  730. line.remaining_free_space = calculate_remaining_free_space();
  731. // If the sum of the unfrozen flex items’ flex factors is less than one, multiply the initial free space by this sum.
  732. if (auto sum_of_flex_factor_of_unfrozen_items = line.sum_of_flex_factor_of_unfrozen_items(); sum_of_flex_factor_of_unfrozen_items < 1 && initial_free_space.has_value()) {
  733. auto value = CSSPixels::nearest_value_for(initial_free_space.value() * sum_of_flex_factor_of_unfrozen_items);
  734. // If the magnitude of this value is less than the magnitude of the remaining free space, use this as the remaining free space.
  735. if (abs(value) < abs(line.remaining_free_space.value()))
  736. line.remaining_free_space = value;
  737. }
  738. // AD-HOC: We allow the remaining free space to be infinite, but we can't let infinity
  739. // leak into the layout geometry, so we treat infinity as zero when used in arithmetic.
  740. auto remaining_free_space_or_zero_if_infinite = line.remaining_free_space.has_value() ? line.remaining_free_space.value() : 0;
  741. // c. If the remaining free space is non-zero, distribute it proportional to the flex factors:
  742. if (line.remaining_free_space != 0) {
  743. // If using the flex grow factor
  744. if (used_flex_factor == FlexFactor::FlexGrowFactor) {
  745. // For every unfrozen item on the line,
  746. // find the ratio of the item’s flex grow factor to the sum of the flex grow factors of all unfrozen items on the line.
  747. auto sum_of_flex_factor_of_unfrozen_items = line.sum_of_flex_factor_of_unfrozen_items();
  748. for (auto& item : line.items) {
  749. if (item.frozen)
  750. continue;
  751. double ratio = item.flex_factor.value() / sum_of_flex_factor_of_unfrozen_items;
  752. // Set the item’s target main size to its flex base size plus a fraction of the remaining free space proportional to the ratio.
  753. item.target_main_size = item.flex_base_size + remaining_free_space_or_zero_if_infinite.scaled(ratio);
  754. }
  755. }
  756. // If using the flex shrink factor
  757. else if (used_flex_factor == FlexFactor::FlexShrinkFactor) {
  758. // For every unfrozen item on the line, multiply its flex shrink factor by its inner flex base size, and note this as its scaled flex shrink factor.
  759. for (auto& item : line.items) {
  760. if (item.frozen)
  761. continue;
  762. item.scaled_flex_shrink_factor = item.flex_factor.value() * item.flex_base_size.to_double();
  763. }
  764. auto sum_of_scaled_flex_shrink_factors_of_all_unfrozen_items_on_line = line.sum_of_scaled_flex_shrink_factor_of_unfrozen_items();
  765. for (auto& item : line.items) {
  766. if (item.frozen)
  767. continue;
  768. // Find the ratio of the item’s scaled flex shrink factor to the sum of the scaled flex shrink factors of all unfrozen items on the line.
  769. double ratio = 1.0;
  770. if (sum_of_scaled_flex_shrink_factors_of_all_unfrozen_items_on_line != 0)
  771. ratio = item.scaled_flex_shrink_factor / sum_of_scaled_flex_shrink_factors_of_all_unfrozen_items_on_line;
  772. // Set the item’s target main size to its flex base size minus a fraction of the absolute value of the remaining free space proportional to the ratio.
  773. // (Note this may result in a negative inner main size; it will be corrected in the next step.)
  774. item.target_main_size = item.flex_base_size - abs(remaining_free_space_or_zero_if_infinite).scaled(ratio);
  775. }
  776. }
  777. }
  778. // d. Fix min/max violations.
  779. CSSPixels total_violation = 0;
  780. // Clamp each non-frozen item’s target main size by its used min and max main sizes and floor its content-box size at zero.
  781. for (auto& item : line.items) {
  782. if (item.frozen)
  783. continue;
  784. auto used_min_main_size = has_main_min_size(item.box)
  785. ? specified_main_min_size(item.box)
  786. : automatic_minimum_size(item);
  787. auto used_max_main_size = has_main_max_size(item.box)
  788. ? specified_main_max_size(item.box)
  789. : CSSPixels::max();
  790. auto original_target_main_size = item.target_main_size;
  791. item.target_main_size = css_clamp(item.target_main_size, used_min_main_size, used_max_main_size);
  792. item.target_main_size = max(item.target_main_size, CSSPixels(0));
  793. // If the item’s target main size was made smaller by this, it’s a max violation.
  794. if (item.target_main_size < original_target_main_size)
  795. item.is_max_violation = true;
  796. // If the item’s target main size was made larger by this, it’s a min violation.
  797. if (item.target_main_size > original_target_main_size)
  798. item.is_min_violation = true;
  799. total_violation += item.target_main_size - original_target_main_size;
  800. }
  801. // e. Freeze over-flexed items.
  802. // The total violation is the sum of the adjustments from the previous step ∑(clamped size - unclamped size).
  803. // If the total violation is:
  804. // Zero
  805. // Freeze all items.
  806. if (total_violation == 0) {
  807. for (auto& item : line.items) {
  808. if (!item.frozen)
  809. item.frozen = true;
  810. }
  811. }
  812. // Positive
  813. // Freeze all the items with min violations.
  814. else if (total_violation > 0) {
  815. for (auto& item : line.items) {
  816. if (!item.frozen && item.is_min_violation)
  817. item.frozen = true;
  818. }
  819. }
  820. // Negative
  821. // Freeze all the items with max violations.
  822. else {
  823. for (auto& item : line.items) {
  824. if (!item.frozen && item.is_max_violation)
  825. item.frozen = true;
  826. }
  827. }
  828. // NOTE: This freezes at least one item, ensuring that the loop makes progress and eventually terminates.
  829. // f. Return to the start of this loop.
  830. }
  831. // NOTE: Calculate the remaining free space once again here, since it's needed later when aligning items.
  832. line.remaining_free_space = calculate_remaining_free_space();
  833. // 6. Set each item’s used main size to its target main size.
  834. for (auto& item : line.items) {
  835. item.main_size = item.target_main_size;
  836. set_main_size(item.box, item.target_main_size);
  837. }
  838. }
  839. // https://drafts.csswg.org/css-flexbox-1/#resolve-flexible-lengths
  840. void FlexFormattingContext::resolve_flexible_lengths()
  841. {
  842. for (auto& line : m_flex_lines) {
  843. resolve_flexible_lengths_for_line(line);
  844. }
  845. }
  846. // https://drafts.csswg.org/css-flexbox-1/#algo-cross-item
  847. void FlexFormattingContext::determine_hypothetical_cross_size_of_item(FlexItem& item, bool resolve_percentage_min_max_sizes)
  848. {
  849. // Determine the hypothetical cross size of each item by performing layout
  850. // as if it were an in-flow block-level box with the used main size
  851. // and the given available space, treating auto as fit-content.
  852. auto const& computed_min_size = this->computed_cross_min_size(item.box);
  853. auto const& computed_max_size = this->computed_cross_max_size(item.box);
  854. auto clamp_min = (!computed_min_size.is_auto() && (resolve_percentage_min_max_sizes || !computed_min_size.contains_percentage())) ? specified_cross_min_size(item.box) : 0;
  855. auto clamp_max = (!should_treat_cross_max_size_as_none(item.box) && (resolve_percentage_min_max_sizes || !computed_max_size.contains_percentage())) ? specified_cross_max_size(item.box) : CSSPixels::max();
  856. // If we have a definite cross size, this is easy! No need to perform layout, we can just use it as-is.
  857. if (has_definite_cross_size(item)) {
  858. // To avoid subtracting padding and border twice for `box-sizing: border-box` only min and max clamp should happen on a second pass
  859. if (resolve_percentage_min_max_sizes) {
  860. item.hypothetical_cross_size = css_clamp(item.hypothetical_cross_size, clamp_min, clamp_max);
  861. return;
  862. }
  863. item.hypothetical_cross_size = css_clamp(inner_cross_size(item), clamp_min, clamp_max);
  864. return;
  865. }
  866. if (item.box->has_preferred_aspect_ratio() && item.main_size.has_value()) {
  867. item.hypothetical_cross_size = calculate_cross_size_from_main_size_and_aspect_ratio(item.main_size.value(), item.box->preferred_aspect_ratio().value());
  868. return;
  869. }
  870. auto computed_cross_size = [&]() -> CSS::Size {
  871. // "... treating auto as fit-content"
  872. if (should_treat_cross_size_as_auto(item.box))
  873. return CSS::Size::make_fit_content();
  874. return this->computed_cross_size(item.box);
  875. }();
  876. if (computed_cross_size.is_min_content()) {
  877. item.hypothetical_cross_size = css_clamp(calculate_min_content_cross_size(item), clamp_min, clamp_max);
  878. return;
  879. }
  880. if (computed_cross_size.is_max_content()) {
  881. item.hypothetical_cross_size = css_clamp(calculate_max_content_cross_size(item), clamp_min, clamp_max);
  882. return;
  883. }
  884. if (computed_cross_size.is_fit_content()) {
  885. CSSPixels fit_content_cross_size = 0;
  886. if (is_row_layout()) {
  887. auto available_width = item.main_size.has_value() ? AvailableSize::make_definite(item.main_size.value()) : AvailableSize::make_indefinite();
  888. auto available_height = AvailableSize::make_indefinite();
  889. fit_content_cross_size = calculate_fit_content_height(item.box, AvailableSpace(available_width, available_height));
  890. } else {
  891. fit_content_cross_size = calculate_fit_content_width(item.box, m_available_space_for_items->space);
  892. }
  893. item.hypothetical_cross_size = css_clamp(fit_content_cross_size, clamp_min, clamp_max);
  894. return;
  895. }
  896. // For indefinite cross sizes, we perform a throwaway layout and then measure it.
  897. LayoutState throwaway_state(&m_state);
  898. auto& box_state = throwaway_state.get_mutable(item.box);
  899. if (is_row_layout()) {
  900. box_state.set_content_width(item.main_size.value());
  901. } else {
  902. box_state.set_content_height(item.main_size.value());
  903. }
  904. // Item has definite main size, layout with that as the used main size.
  905. auto independent_formatting_context = create_independent_formatting_context_if_needed(throwaway_state, item.box);
  906. // NOTE: Flex items should always create an independent formatting context!
  907. VERIFY(independent_formatting_context);
  908. auto available_width = is_row_layout() ? AvailableSize::make_definite(item.main_size.value()) : AvailableSize::make_indefinite();
  909. auto available_height = is_row_layout() ? AvailableSize::make_indefinite() : AvailableSize::make_definite(item.main_size.value());
  910. independent_formatting_context->run(item.box, LayoutMode::Normal, AvailableSpace(available_width, available_height));
  911. auto automatic_cross_size = is_row_layout() ? independent_formatting_context->automatic_content_height()
  912. : independent_formatting_context->automatic_content_width();
  913. item.hypothetical_cross_size = css_clamp(automatic_cross_size, clamp_min, clamp_max);
  914. }
  915. // https://www.w3.org/TR/css-flexbox-1/#algo-cross-line
  916. void FlexFormattingContext::calculate_cross_size_of_each_flex_line()
  917. {
  918. // If the flex container is single-line and has a definite cross size, the cross size of the flex line is the flex container’s inner cross size.
  919. if (is_single_line() && has_definite_cross_size(m_flex_container_state)) {
  920. m_flex_lines[0].cross_size = inner_cross_size(m_flex_container_state);
  921. return;
  922. }
  923. // Otherwise, for each flex line:
  924. for (auto& flex_line : m_flex_lines) {
  925. // FIXME: 1. Collect all the flex items whose inline-axis is parallel to the main-axis, whose align-self is baseline,
  926. // and whose cross-axis margins are both non-auto. Find the largest of the distances between each item’s baseline
  927. // and its hypothetical outer cross-start edge, and the largest of the distances between each item’s baseline
  928. // and its hypothetical outer cross-end edge, and sum these two values.
  929. // 2. Among all the items not collected by the previous step, find the largest outer hypothetical cross size.
  930. CSSPixels largest_hypothetical_cross_size = 0;
  931. for (auto& item : flex_line.items) {
  932. if (largest_hypothetical_cross_size < item.hypothetical_cross_size_with_margins())
  933. largest_hypothetical_cross_size = item.hypothetical_cross_size_with_margins();
  934. }
  935. // 3. The used cross-size of the flex line is the largest of the numbers found in the previous two steps and zero.
  936. flex_line.cross_size = max(CSSPixels(0), largest_hypothetical_cross_size);
  937. }
  938. // If the flex container is single-line, then clamp the line’s cross-size to be within the container’s computed min and max cross sizes.
  939. // Note that if CSS 2.1’s definition of min/max-width/height applied more generally, this behavior would fall out automatically.
  940. // AD-HOC: We don't do this when the flex container is being sized under a min-content or max-content constraint.
  941. if (is_single_line() && !m_available_space_for_items->cross.is_intrinsic_sizing_constraint()) {
  942. auto const& computed_min_size = this->computed_cross_min_size(flex_container());
  943. auto const& computed_max_size = this->computed_cross_max_size(flex_container());
  944. auto cross_min_size = (!computed_min_size.is_auto() && !computed_min_size.contains_percentage()) ? specified_cross_min_size(flex_container()) : 0;
  945. auto cross_max_size = (!computed_max_size.is_none() && !computed_max_size.contains_percentage()) ? specified_cross_max_size(flex_container()) : CSSPixels::max();
  946. m_flex_lines[0].cross_size = css_clamp(m_flex_lines[0].cross_size, cross_min_size, cross_max_size);
  947. }
  948. }
  949. // https://www.w3.org/TR/css-flexbox-1/#algo-stretch
  950. void FlexFormattingContext::determine_used_cross_size_of_each_flex_item()
  951. {
  952. for (auto& flex_line : m_flex_lines) {
  953. for (auto& item : flex_line.items) {
  954. // If a flex item has align-self: stretch, its computed cross size property is auto,
  955. // and neither of its cross-axis margins are auto, the used outer cross size is the used cross size of its flex line,
  956. // clamped according to the item’s used min and max cross sizes.
  957. auto flex_item_alignment = alignment_for_item(item.box);
  958. if ((flex_item_alignment == CSS::AlignItems::Stretch || flex_item_alignment == CSS::AlignItems::Normal)
  959. && is_cross_auto(item.box)
  960. && !item.margins.cross_before_is_auto
  961. && !item.margins.cross_after_is_auto) {
  962. auto unclamped_cross_size = flex_line.cross_size
  963. - item.margins.cross_before - item.margins.cross_after
  964. - item.padding.cross_before - item.padding.cross_after
  965. - item.borders.cross_before - item.borders.cross_after;
  966. auto const& computed_min_size = computed_cross_min_size(item.box);
  967. auto const& computed_max_size = computed_cross_max_size(item.box);
  968. auto cross_min_size = (!computed_min_size.is_auto() && !computed_min_size.contains_percentage()) ? specified_cross_min_size(item.box) : 0;
  969. auto cross_max_size = (!should_treat_cross_max_size_as_none(item.box) && !computed_max_size.contains_percentage()) ? specified_cross_max_size(item.box) : CSSPixels::max();
  970. item.cross_size = css_clamp(unclamped_cross_size, cross_min_size, cross_max_size);
  971. } else {
  972. // Otherwise, the used cross size is the item’s hypothetical cross size.
  973. item.cross_size = item.hypothetical_cross_size;
  974. }
  975. }
  976. }
  977. }
  978. // https://www.w3.org/TR/css-flexbox-1/#algo-main-align
  979. void FlexFormattingContext::distribute_any_remaining_free_space()
  980. {
  981. for (auto& flex_line : m_flex_lines) {
  982. // 12.1.
  983. CSSPixels used_main_space = 0;
  984. size_t auto_margins = 0;
  985. for (auto& item : flex_line.items) {
  986. used_main_space += item.main_size.value();
  987. if (item.margins.main_before_is_auto)
  988. ++auto_margins;
  989. if (item.margins.main_after_is_auto)
  990. ++auto_margins;
  991. used_main_space += item.margins.main_before + item.margins.main_after
  992. + item.borders.main_before + item.borders.main_after
  993. + item.padding.main_before + item.padding.main_after;
  994. }
  995. // CSS-FLEXBOX-2: Account for gap between flex items.
  996. used_main_space += main_gap() * (flex_line.items.size() - 1);
  997. if (flex_line.remaining_free_space.has_value() && flex_line.remaining_free_space.value() > 0 && auto_margins > 0) {
  998. CSSPixels size_per_auto_margin = flex_line.remaining_free_space.value() / auto_margins;
  999. for (auto& item : flex_line.items) {
  1000. if (item.margins.main_before_is_auto)
  1001. set_main_axis_first_margin(item, size_per_auto_margin);
  1002. if (item.margins.main_after_is_auto)
  1003. set_main_axis_second_margin(item, size_per_auto_margin);
  1004. }
  1005. } else {
  1006. for (auto& item : flex_line.items) {
  1007. if (item.margins.main_before_is_auto)
  1008. set_main_axis_first_margin(item, 0);
  1009. if (item.margins.main_after_is_auto)
  1010. set_main_axis_second_margin(item, 0);
  1011. }
  1012. }
  1013. // 12.2.
  1014. // CSS-FLEXBOX-2: Account for gap between items.
  1015. CSSPixels space_between_items = main_gap();
  1016. CSSPixels initial_offset = 0;
  1017. auto number_of_items = flex_line.items.size();
  1018. if (auto_margins == 0 && number_of_items > 0) {
  1019. switch (flex_container().computed_values().justify_content()) {
  1020. case CSS::JustifyContent::Start:
  1021. initial_offset = 0;
  1022. break;
  1023. case CSS::JustifyContent::Stretch:
  1024. case CSS::JustifyContent::Normal:
  1025. case CSS::JustifyContent::FlexStart:
  1026. if (is_direction_reverse()) {
  1027. initial_offset = inner_main_size(m_flex_container_state);
  1028. } else {
  1029. initial_offset = 0;
  1030. }
  1031. break;
  1032. case CSS::JustifyContent::End:
  1033. initial_offset = inner_main_size(m_flex_container_state);
  1034. break;
  1035. case CSS::JustifyContent::FlexEnd:
  1036. if (is_direction_reverse()) {
  1037. initial_offset = 0;
  1038. } else {
  1039. initial_offset = inner_main_size(m_flex_container_state);
  1040. }
  1041. break;
  1042. case CSS::JustifyContent::Center:
  1043. initial_offset = (inner_main_size(m_flex_container_state) - used_main_space) / 2;
  1044. if (is_direction_reverse()) {
  1045. initial_offset = inner_main_size(m_flex_container_state) - initial_offset;
  1046. }
  1047. break;
  1048. case CSS::JustifyContent::SpaceBetween:
  1049. if (is_direction_reverse()) {
  1050. initial_offset = inner_main_size(m_flex_container_state);
  1051. } else {
  1052. initial_offset = 0;
  1053. }
  1054. if (flex_line.remaining_free_space.has_value() && number_of_items > 1)
  1055. space_between_items = max(CSSPixels(0), flex_line.remaining_free_space.value() / (number_of_items - 1));
  1056. break;
  1057. case CSS::JustifyContent::SpaceAround:
  1058. if (flex_line.remaining_free_space.has_value())
  1059. space_between_items = max(CSSPixels(0), flex_line.remaining_free_space.value() / number_of_items);
  1060. if (is_direction_reverse()) {
  1061. initial_offset = inner_main_size(m_flex_container_state) - space_between_items / 2;
  1062. } else {
  1063. initial_offset = space_between_items / 2;
  1064. }
  1065. break;
  1066. case CSS::JustifyContent::SpaceEvenly:
  1067. if (flex_line.remaining_free_space.has_value())
  1068. space_between_items = max(CSSPixels(0), flex_line.remaining_free_space.value() / (number_of_items + 1));
  1069. if (is_direction_reverse()) {
  1070. initial_offset = inner_main_size(m_flex_container_state) - space_between_items;
  1071. } else {
  1072. initial_offset = space_between_items;
  1073. }
  1074. break;
  1075. }
  1076. }
  1077. // For reverse, we use FlexRegionRenderCursor::Right
  1078. // to indicate the cursor offset is the end and render backwards
  1079. // Otherwise the cursor offset is the 'start' of the region or initial offset
  1080. enum class FlexRegionRenderCursor {
  1081. Left,
  1082. Right
  1083. };
  1084. auto flex_region_render_cursor = FlexRegionRenderCursor::Left;
  1085. if (auto_margins == 0) {
  1086. switch (flex_container().computed_values().justify_content()) {
  1087. case CSS::JustifyContent::Normal:
  1088. case CSS::JustifyContent::FlexStart:
  1089. case CSS::JustifyContent::Center:
  1090. case CSS::JustifyContent::SpaceAround:
  1091. case CSS::JustifyContent::SpaceBetween:
  1092. case CSS::JustifyContent::SpaceEvenly:
  1093. case CSS::JustifyContent::Stretch:
  1094. if (is_direction_reverse()) {
  1095. flex_region_render_cursor = FlexRegionRenderCursor::Right;
  1096. }
  1097. break;
  1098. case CSS::JustifyContent::End:
  1099. flex_region_render_cursor = FlexRegionRenderCursor::Right;
  1100. break;
  1101. case CSS::JustifyContent::FlexEnd:
  1102. if (!is_direction_reverse()) {
  1103. flex_region_render_cursor = FlexRegionRenderCursor::Right;
  1104. }
  1105. break;
  1106. default:
  1107. break;
  1108. }
  1109. }
  1110. CSSPixels cursor_offset = initial_offset;
  1111. auto place_item = [&](FlexItem& item) {
  1112. auto amount_of_main_size_used = item.main_size.value()
  1113. + item.margins.main_before
  1114. + item.borders.main_before
  1115. + item.padding.main_before
  1116. + item.margins.main_after
  1117. + item.borders.main_after
  1118. + item.padding.main_after
  1119. + space_between_items;
  1120. if (is_direction_reverse() && flex_region_render_cursor == FlexRegionRenderCursor::Right) {
  1121. item.main_offset = cursor_offset - item.main_size.value() - item.margins.main_after - item.borders.main_after - item.padding.main_after;
  1122. cursor_offset -= amount_of_main_size_used;
  1123. } else if (flex_region_render_cursor == FlexRegionRenderCursor::Right) {
  1124. cursor_offset -= amount_of_main_size_used;
  1125. item.main_offset = cursor_offset + item.margins.main_before + item.borders.main_before + item.padding.main_before;
  1126. } else {
  1127. item.main_offset = cursor_offset + item.margins.main_before + item.borders.main_before + item.padding.main_before;
  1128. cursor_offset += amount_of_main_size_used;
  1129. }
  1130. };
  1131. if (flex_region_render_cursor == FlexRegionRenderCursor::Right) {
  1132. for (ssize_t i = flex_line.items.size() - 1; i >= 0; --i) {
  1133. auto& item = flex_line.items[i];
  1134. place_item(item);
  1135. }
  1136. } else {
  1137. for (size_t i = 0; i < flex_line.items.size(); ++i) {
  1138. auto& item = flex_line.items[i];
  1139. place_item(item);
  1140. }
  1141. }
  1142. }
  1143. }
  1144. void FlexFormattingContext::dump_items() const
  1145. {
  1146. dbgln("\033[34;1mflex-container\033[0m {}, direction: {}, current-size: {}x{}", flex_container().debug_description(), is_row_layout() ? "row" : "column", m_flex_container_state.content_width(), m_flex_container_state.content_height());
  1147. for (size_t i = 0; i < m_flex_lines.size(); ++i) {
  1148. dbgln("{} flex-line #{}:", flex_container().debug_description(), i);
  1149. for (size_t j = 0; j < m_flex_lines[i].items.size(); ++j) {
  1150. auto& item = m_flex_lines[i].items[j];
  1151. dbgln("{} flex-item #{}: {} (main:{}, cross:{})", flex_container().debug_description(), j, item.box->debug_description(), item.main_size.value_or(-1), item.cross_size.value_or(-1));
  1152. }
  1153. }
  1154. }
  1155. CSS::AlignItems FlexFormattingContext::alignment_for_item(Box const& box) const
  1156. {
  1157. switch (box.computed_values().align_self()) {
  1158. case CSS::AlignSelf::Auto:
  1159. return flex_container().computed_values().align_items();
  1160. case CSS::AlignSelf::End:
  1161. return CSS::AlignItems::End;
  1162. case CSS::AlignSelf::Normal:
  1163. return CSS::AlignItems::Normal;
  1164. case CSS::AlignSelf::SelfStart:
  1165. return CSS::AlignItems::SelfStart;
  1166. case CSS::AlignSelf::SelfEnd:
  1167. return CSS::AlignItems::SelfEnd;
  1168. case CSS::AlignSelf::FlexStart:
  1169. return CSS::AlignItems::FlexStart;
  1170. case CSS::AlignSelf::FlexEnd:
  1171. return CSS::AlignItems::FlexEnd;
  1172. case CSS::AlignSelf::Center:
  1173. return CSS::AlignItems::Center;
  1174. case CSS::AlignSelf::Baseline:
  1175. return CSS::AlignItems::Baseline;
  1176. case CSS::AlignSelf::Start:
  1177. return CSS::AlignItems::Start;
  1178. case CSS::AlignSelf::Stretch:
  1179. return CSS::AlignItems::Stretch;
  1180. case CSS::AlignSelf::Safe:
  1181. return CSS::AlignItems::Safe;
  1182. case CSS::AlignSelf::Unsafe:
  1183. return CSS::AlignItems::Unsafe;
  1184. default:
  1185. VERIFY_NOT_REACHED();
  1186. }
  1187. }
  1188. void FlexFormattingContext::align_all_flex_items_along_the_cross_axis()
  1189. {
  1190. // FIXME: Take better care of margins
  1191. for (auto& flex_line : m_flex_lines) {
  1192. for (auto& item : flex_line.items) {
  1193. CSSPixels half_line_size = flex_line.cross_size / 2;
  1194. switch (alignment_for_item(item.box)) {
  1195. case CSS::AlignItems::Baseline:
  1196. // FIXME: Implement this
  1197. // Fallthrough
  1198. case CSS::AlignItems::Start:
  1199. case CSS::AlignItems::FlexStart:
  1200. case CSS::AlignItems::Stretch:
  1201. case CSS::AlignItems::Normal:
  1202. item.cross_offset = -half_line_size + item.margins.cross_before + item.borders.cross_before + item.padding.cross_before;
  1203. break;
  1204. case CSS::AlignItems::End:
  1205. case CSS::AlignItems::FlexEnd:
  1206. item.cross_offset = half_line_size - item.cross_size.value() - item.margins.cross_after - item.borders.cross_after - item.padding.cross_after;
  1207. break;
  1208. case CSS::AlignItems::Center:
  1209. item.cross_offset = -(item.cross_size.value() / 2);
  1210. break;
  1211. default:
  1212. break;
  1213. }
  1214. }
  1215. }
  1216. }
  1217. // https://www.w3.org/TR/css-flexbox-1/#algo-cross-container
  1218. void FlexFormattingContext::determine_flex_container_used_cross_size()
  1219. {
  1220. CSSPixels cross_size = 0;
  1221. if (has_definite_cross_size(m_flex_container_state)) {
  1222. // Flex container has definite cross size: easy-peasy.
  1223. cross_size = inner_cross_size(m_flex_container_state);
  1224. } else {
  1225. // Flex container has indefinite cross size.
  1226. auto cross_size_value = is_row_layout() ? flex_container().computed_values().height() : flex_container().computed_values().width();
  1227. if (cross_size_value.is_auto() || cross_size_value.contains_percentage()) {
  1228. // If a content-based cross size is needed, use the sum of the flex lines' cross sizes.
  1229. CSSPixels sum_of_flex_lines_cross_sizes = 0;
  1230. for (auto& flex_line : m_flex_lines) {
  1231. sum_of_flex_lines_cross_sizes += flex_line.cross_size;
  1232. }
  1233. cross_size = sum_of_flex_lines_cross_sizes;
  1234. if (cross_size_value.contains_percentage()) {
  1235. // FIXME: Handle percentage values here! Right now we're just treating them as "auto"
  1236. }
  1237. } else {
  1238. // Otherwise, resolve the indefinite size at this point.
  1239. cross_size = cross_size_value.to_px(flex_container(), inner_cross_size(m_state.get(*flex_container().containing_block())));
  1240. }
  1241. }
  1242. // AD-HOC: We don't apply min/max cross size constraints when sizing the flex container under an intrinsic sizing constraint.
  1243. if (!m_available_space_for_items->cross.is_intrinsic_sizing_constraint()) {
  1244. auto const& computed_min_size = this->computed_cross_min_size(flex_container());
  1245. auto const& computed_max_size = this->computed_cross_max_size(flex_container());
  1246. auto cross_min_size = (!computed_min_size.is_auto() && !computed_min_size.contains_percentage()) ? specified_cross_min_size(flex_container()) : 0;
  1247. auto cross_max_size = (!computed_max_size.is_none() && !computed_max_size.contains_percentage()) ? specified_cross_max_size(flex_container()) : CSSPixels::max();
  1248. set_cross_size(flex_container(), css_clamp(cross_size, cross_min_size, cross_max_size));
  1249. } else {
  1250. set_cross_size(flex_container(), cross_size);
  1251. }
  1252. }
  1253. // https://www.w3.org/TR/css-flexbox-1/#algo-line-align
  1254. void FlexFormattingContext::align_all_flex_lines()
  1255. {
  1256. if (m_flex_lines.is_empty())
  1257. return;
  1258. // FIXME: Support reverse
  1259. CSSPixels cross_size_of_flex_container = inner_cross_size(m_flex_container_state);
  1260. if (is_single_line()) {
  1261. // For single-line flex containers, we only need to center the line along the cross axis.
  1262. auto& flex_line = m_flex_lines[0];
  1263. CSSPixels center_of_line = cross_size_of_flex_container / 2;
  1264. for (auto& item : flex_line.items) {
  1265. item.cross_offset += center_of_line;
  1266. }
  1267. } else {
  1268. CSSPixels sum_of_flex_line_cross_sizes = 0;
  1269. for (auto& line : m_flex_lines)
  1270. sum_of_flex_line_cross_sizes += line.cross_size;
  1271. // CSS-FLEXBOX-2: Account for gap between flex lines.
  1272. sum_of_flex_line_cross_sizes += cross_gap() * (m_flex_lines.size() - 1);
  1273. CSSPixels start_of_current_line = 0;
  1274. CSSPixels gap_size = 0;
  1275. switch (flex_container().computed_values().align_content()) {
  1276. case CSS::AlignContent::FlexStart:
  1277. start_of_current_line = 0;
  1278. break;
  1279. case CSS::AlignContent::FlexEnd:
  1280. start_of_current_line = cross_size_of_flex_container - sum_of_flex_line_cross_sizes;
  1281. break;
  1282. case CSS::AlignContent::Center:
  1283. start_of_current_line = (cross_size_of_flex_container / 2) - (sum_of_flex_line_cross_sizes / 2);
  1284. break;
  1285. case CSS::AlignContent::SpaceBetween: {
  1286. start_of_current_line = 0;
  1287. auto leftover_free_space = cross_size_of_flex_container - sum_of_flex_line_cross_sizes;
  1288. auto leftover_flex_lines_size = m_flex_lines.size();
  1289. if (leftover_free_space >= 0 && leftover_flex_lines_size > 1) {
  1290. int gap_count = leftover_flex_lines_size - 1;
  1291. gap_size = leftover_free_space / gap_count;
  1292. }
  1293. break;
  1294. }
  1295. case CSS::AlignContent::SpaceAround: {
  1296. auto leftover_free_space = cross_size_of_flex_container - sum_of_flex_line_cross_sizes;
  1297. if (leftover_free_space < 0) {
  1298. // If the leftover free-space is negative this value is identical to center.
  1299. start_of_current_line = (cross_size_of_flex_container / 2) - (sum_of_flex_line_cross_sizes / 2);
  1300. break;
  1301. }
  1302. gap_size = leftover_free_space / m_flex_lines.size();
  1303. // The spacing between the first/last lines and the flex container edges is half the size of the spacing between flex lines.
  1304. start_of_current_line = gap_size / 2;
  1305. break;
  1306. }
  1307. case CSS::AlignContent::SpaceEvenly: {
  1308. auto leftover_free_space = cross_size_of_flex_container - sum_of_flex_line_cross_sizes;
  1309. if (leftover_free_space < 0) {
  1310. // If the leftover free-space is negative this value is identical to center.
  1311. start_of_current_line = (cross_size_of_flex_container / 2) - (sum_of_flex_line_cross_sizes / 2);
  1312. break;
  1313. }
  1314. gap_size = leftover_free_space / (m_flex_lines.size() + 1);
  1315. // The spacing between the first/last lines and the flex container edges is the size of the spacing between flex lines.
  1316. start_of_current_line = gap_size;
  1317. break;
  1318. }
  1319. case CSS::AlignContent::Normal:
  1320. case CSS::AlignContent::Stretch:
  1321. start_of_current_line = 0;
  1322. break;
  1323. }
  1324. for (auto& flex_line : m_flex_lines) {
  1325. CSSPixels center_of_current_line = start_of_current_line + (flex_line.cross_size / 2);
  1326. for (auto& item : flex_line.items) {
  1327. item.cross_offset += center_of_current_line;
  1328. }
  1329. start_of_current_line += flex_line.cross_size + gap_size;
  1330. // CSS-FLEXBOX-2: Account for gap between flex lines.
  1331. start_of_current_line += cross_gap();
  1332. }
  1333. }
  1334. }
  1335. void FlexFormattingContext::copy_dimensions_from_flex_items_to_boxes()
  1336. {
  1337. for (auto& item : m_flex_items) {
  1338. auto const& box = item.box;
  1339. item.used_values.padding_left = box->computed_values().padding().left().to_px(box, m_flex_container_state.content_width());
  1340. item.used_values.padding_right = box->computed_values().padding().right().to_px(box, m_flex_container_state.content_width());
  1341. item.used_values.padding_top = box->computed_values().padding().top().to_px(box, m_flex_container_state.content_width());
  1342. item.used_values.padding_bottom = box->computed_values().padding().bottom().to_px(box, m_flex_container_state.content_width());
  1343. item.used_values.margin_left = box->computed_values().margin().left().to_px(box, m_flex_container_state.content_width());
  1344. item.used_values.margin_right = box->computed_values().margin().right().to_px(box, m_flex_container_state.content_width());
  1345. item.used_values.margin_top = box->computed_values().margin().top().to_px(box, m_flex_container_state.content_width());
  1346. item.used_values.margin_bottom = box->computed_values().margin().bottom().to_px(box, m_flex_container_state.content_width());
  1347. item.used_values.border_left = box->computed_values().border_left().width;
  1348. item.used_values.border_right = box->computed_values().border_right().width;
  1349. item.used_values.border_top = box->computed_values().border_top().width;
  1350. item.used_values.border_bottom = box->computed_values().border_bottom().width;
  1351. set_main_size(box, item.main_size.value());
  1352. set_cross_size(box, item.cross_size.value());
  1353. set_offset(box, item.main_offset, item.cross_offset);
  1354. }
  1355. }
  1356. // https://drafts.csswg.org/css-flexbox-1/#intrinsic-sizes
  1357. void FlexFormattingContext::determine_intrinsic_size_of_flex_container()
  1358. {
  1359. if (m_available_space_for_items->main.is_intrinsic_sizing_constraint()) {
  1360. CSSPixels main_size = calculate_intrinsic_main_size_of_flex_container();
  1361. set_main_size(flex_container(), main_size);
  1362. }
  1363. if (m_available_space_for_items->cross.is_intrinsic_sizing_constraint()) {
  1364. CSSPixels cross_size = calculate_intrinsic_cross_size_of_flex_container();
  1365. set_cross_size(flex_container(), cross_size);
  1366. }
  1367. }
  1368. // https://drafts.csswg.org/css-flexbox-1/#intrinsic-main-sizes
  1369. CSSPixels FlexFormattingContext::calculate_intrinsic_main_size_of_flex_container()
  1370. {
  1371. // The min-content main size of a single-line flex container is calculated identically to the max-content main size,
  1372. // except that the flex items’ min-content contributions are used instead of their max-content contributions.
  1373. // However, for a multi-line container, it is simply the largest min-content contribution of all the non-collapsed flex items in the flex container.
  1374. if (!is_single_line() && m_available_space_for_items->main.is_min_content()) {
  1375. CSSPixels largest_contribution = 0;
  1376. for (auto const& item : m_flex_items) {
  1377. // FIXME: Skip collapsed flex items.
  1378. largest_contribution = max(largest_contribution, calculate_main_min_content_contribution(item));
  1379. }
  1380. return largest_contribution;
  1381. }
  1382. // The max-content main size of a flex container is, fundamentally, the smallest size the flex container
  1383. // can take such that when flex layout is run with that container size, each flex item ends up at least
  1384. // as large as its max-content contribution, to the extent allowed by the items’ flexibility.
  1385. // It is calculated, considering only non-collapsed flex items, by:
  1386. // 1. For each flex item, subtract its outer flex base size from its max-content contribution size.
  1387. // If that result is positive, divide it by the item’s flex grow factor if the flex grow factor is ≥ 1,
  1388. // or multiply it by the flex grow factor if the flex grow factor is < 1; if the result is negative,
  1389. // divide it by the item’s scaled flex shrink factor (if dividing by zero, treat the result as negative infinity).
  1390. // This is the item’s desired flex fraction.
  1391. for (auto& item : m_flex_items) {
  1392. CSSPixels contribution = 0;
  1393. if (m_available_space_for_items->main.is_min_content())
  1394. contribution = calculate_main_min_content_contribution(item);
  1395. else if (m_available_space_for_items->main.is_max_content())
  1396. contribution = calculate_main_max_content_contribution(item);
  1397. CSSPixels outer_flex_base_size = item.flex_base_size + item.margins.main_before + item.margins.main_after + item.borders.main_before + item.borders.main_after + item.padding.main_before + item.padding.main_after;
  1398. CSSPixels result = contribution - outer_flex_base_size;
  1399. if (result > 0) {
  1400. if (item.box->computed_values().flex_grow() >= 1) {
  1401. result.scale_by(1 / item.box->computed_values().flex_grow());
  1402. } else {
  1403. result.scale_by(item.box->computed_values().flex_grow());
  1404. }
  1405. } else if (result < 0) {
  1406. if (item.scaled_flex_shrink_factor == 0)
  1407. result = CSSPixels::min();
  1408. else
  1409. result.scale_by(1 / item.scaled_flex_shrink_factor);
  1410. }
  1411. item.desired_flex_fraction = result.to_double();
  1412. }
  1413. // 2. Place all flex items into lines of infinite length.
  1414. m_flex_lines.clear();
  1415. if (!m_flex_items.is_empty())
  1416. m_flex_lines.append(FlexLine {});
  1417. for (auto& item : m_flex_items) {
  1418. // FIXME: Honor breaking requests.
  1419. m_flex_lines.last().items.append(item);
  1420. }
  1421. // Within each line, find the greatest (most positive) desired flex fraction among all the flex items.
  1422. // This is the line’s chosen flex fraction.
  1423. for (auto& flex_line : m_flex_lines) {
  1424. float greatest_desired_flex_fraction = 0;
  1425. float sum_of_flex_grow_factors = 0;
  1426. float sum_of_flex_shrink_factors = 0;
  1427. for (auto& item : flex_line.items) {
  1428. greatest_desired_flex_fraction = max(greatest_desired_flex_fraction, item.desired_flex_fraction);
  1429. sum_of_flex_grow_factors += item.box->computed_values().flex_grow();
  1430. sum_of_flex_shrink_factors += item.box->computed_values().flex_shrink();
  1431. }
  1432. float chosen_flex_fraction = greatest_desired_flex_fraction;
  1433. // 3. If the chosen flex fraction is positive, and the sum of the line’s flex grow factors is less than 1,
  1434. // divide the chosen flex fraction by that sum.
  1435. if (chosen_flex_fraction > 0 && sum_of_flex_grow_factors < 1)
  1436. chosen_flex_fraction /= sum_of_flex_grow_factors;
  1437. // If the chosen flex fraction is negative, and the sum of the line’s flex shrink factors is less than 1,
  1438. // multiply the chosen flex fraction by that sum.
  1439. if (chosen_flex_fraction < 0 && sum_of_flex_shrink_factors < 1)
  1440. chosen_flex_fraction *= sum_of_flex_shrink_factors;
  1441. flex_line.chosen_flex_fraction = chosen_flex_fraction;
  1442. }
  1443. auto determine_main_size = [&]() -> CSSPixels {
  1444. CSSPixels largest_sum = 0;
  1445. for (auto& flex_line : m_flex_lines) {
  1446. // 4. Add each item’s flex base size to the product of its flex grow factor (scaled flex shrink factor, if shrinking)
  1447. // and the chosen flex fraction, then clamp that result by the max main size floored by the min main size.
  1448. CSSPixels sum = 0;
  1449. for (auto& item : flex_line.items) {
  1450. double product = 0;
  1451. if (item.desired_flex_fraction > 0)
  1452. product = flex_line.chosen_flex_fraction * static_cast<double>(item.box->computed_values().flex_grow());
  1453. else if (item.desired_flex_fraction < 0)
  1454. product = flex_line.chosen_flex_fraction * item.scaled_flex_shrink_factor;
  1455. auto result = item.flex_base_size + CSSPixels::nearest_value_for(product);
  1456. auto const& computed_min_size = this->computed_main_min_size(item.box);
  1457. auto const& computed_max_size = this->computed_main_max_size(item.box);
  1458. auto clamp_min = (!computed_min_size.is_auto() && !computed_min_size.contains_percentage()) ? specified_main_min_size(item.box) : automatic_minimum_size(item);
  1459. auto clamp_max = (!should_treat_main_max_size_as_none(item.box) && !computed_max_size.contains_percentage()) ? specified_main_max_size(item.box) : CSSPixels::max();
  1460. result = css_clamp(result, clamp_min, clamp_max);
  1461. // NOTE: The spec doesn't mention anything about the *outer* size here, but if we don't add the margin box,
  1462. // flex items with non-zero padding/border/margin in the main axis end up overflowing the container.
  1463. result = item.add_main_margin_box_sizes(result);
  1464. sum += result;
  1465. }
  1466. // CSS-FLEXBOX-2: Account for gap between flex items.
  1467. sum += main_gap() * (flex_line.items.size() - 1);
  1468. largest_sum = max(largest_sum, sum);
  1469. }
  1470. // 5. The flex container’s max-content size is the largest sum (among all the lines) of the afore-calculated sizes of all items within a single line.
  1471. return largest_sum;
  1472. };
  1473. auto main_size = determine_main_size();
  1474. set_main_size(flex_container(), main_size);
  1475. return main_size;
  1476. }
  1477. // https://drafts.csswg.org/css-flexbox-1/#intrinsic-cross-sizes
  1478. CSSPixels FlexFormattingContext::calculate_intrinsic_cross_size_of_flex_container()
  1479. {
  1480. // The min-content/max-content cross size of a single-line flex container
  1481. // is the largest min-content contribution/max-content contribution (respectively) of its flex items.
  1482. if (is_single_line()) {
  1483. auto calculate_largest_contribution = [&](bool resolve_percentage_min_max_sizes) {
  1484. CSSPixels largest_contribution = 0;
  1485. for (auto& item : m_flex_items) {
  1486. CSSPixels contribution = 0;
  1487. if (m_available_space_for_items->cross.is_min_content())
  1488. contribution = calculate_cross_min_content_contribution(item, resolve_percentage_min_max_sizes);
  1489. else if (m_available_space_for_items->cross.is_max_content())
  1490. contribution = calculate_cross_max_content_contribution(item, resolve_percentage_min_max_sizes);
  1491. largest_contribution = max(largest_contribution, contribution);
  1492. }
  1493. return largest_contribution;
  1494. };
  1495. auto first_pass_largest_contribution = calculate_largest_contribution(false);
  1496. set_cross_size(flex_container(), first_pass_largest_contribution);
  1497. auto second_pass_largest_contribution = calculate_largest_contribution(true);
  1498. return second_pass_largest_contribution;
  1499. }
  1500. if (is_row_layout()) {
  1501. // row multi-line flex container cross-size
  1502. // The min-content/max-content cross size is the sum of the flex line cross sizes resulting from
  1503. // sizing the flex container under a cross-axis min-content constraint/max-content constraint (respectively).
  1504. // NOTE: We fall through to the ad-hoc section below.
  1505. } else {
  1506. // column multi-line flex container cross-size
  1507. // The min-content cross size is the largest min-content contribution among all of its flex items.
  1508. if (m_available_space_for_items->cross.is_min_content()) {
  1509. auto calculate_largest_contribution = [&](bool resolve_percentage_min_max_sizes) {
  1510. CSSPixels largest_contribution = 0;
  1511. for (auto& item : m_flex_items) {
  1512. CSSPixels contribution = calculate_cross_min_content_contribution(item, resolve_percentage_min_max_sizes);
  1513. largest_contribution = max(largest_contribution, contribution);
  1514. }
  1515. return largest_contribution;
  1516. };
  1517. auto first_pass_largest_contribution = calculate_largest_contribution(false);
  1518. set_cross_size(flex_container(), first_pass_largest_contribution);
  1519. auto second_pass_largest_contribution = calculate_largest_contribution(true);
  1520. return second_pass_largest_contribution;
  1521. }
  1522. // The max-content cross size is the sum of the flex line cross sizes resulting from
  1523. // sizing the flex container under a cross-axis max-content constraint,
  1524. // using the largest max-content cross-size contribution among the flex items
  1525. // as the available space in the cross axis for each of the flex items during layout.
  1526. // NOTE: We fall through to the ad-hoc section below.
  1527. }
  1528. CSSPixels sum_of_flex_line_cross_sizes = 0;
  1529. for (auto& flex_line : m_flex_lines) {
  1530. sum_of_flex_line_cross_sizes += flex_line.cross_size;
  1531. }
  1532. // CSS-FLEXBOX-2: Account for gap between flex lines.
  1533. sum_of_flex_line_cross_sizes += cross_gap() * (m_flex_lines.size() - 1);
  1534. return sum_of_flex_line_cross_sizes;
  1535. }
  1536. // https://drafts.csswg.org/css-flexbox-1/#intrinsic-item-contributions
  1537. CSSPixels FlexFormattingContext::calculate_main_min_content_contribution(FlexItem const& item) const
  1538. {
  1539. // The main-size min-content contribution of a flex item is
  1540. // the larger of its outer min-content size and outer preferred size if that is not auto,
  1541. // clamped by its min/max main size.
  1542. auto larger_size = [&] {
  1543. auto inner_min_content_size = calculate_min_content_main_size(item);
  1544. if (computed_main_size(item.box).is_auto())
  1545. return inner_min_content_size;
  1546. auto inner_preferred_size = is_row_layout() ? get_pixel_width(item.box, computed_main_size(item.box)) : get_pixel_height(item.box, computed_main_size(item.box));
  1547. return max(inner_min_content_size, inner_preferred_size);
  1548. }();
  1549. auto clamp_min = has_main_min_size(item.box) ? specified_main_min_size(item.box) : automatic_minimum_size(item);
  1550. auto clamp_max = has_main_max_size(item.box) ? specified_main_max_size(item.box) : CSSPixels::max();
  1551. auto clamped_inner_size = css_clamp(larger_size, clamp_min, clamp_max);
  1552. return item.add_main_margin_box_sizes(clamped_inner_size);
  1553. }
  1554. // https://drafts.csswg.org/css-flexbox-1/#intrinsic-item-contributions
  1555. CSSPixels FlexFormattingContext::calculate_main_max_content_contribution(FlexItem const& item) const
  1556. {
  1557. // The main-size max-content contribution of a flex item is
  1558. // the larger of its outer max-content size and outer preferred size if that is not auto,
  1559. // clamped by its min/max main size.
  1560. auto larger_size = [&] {
  1561. auto inner_max_content_size = calculate_max_content_main_size(item);
  1562. if (computed_main_size(item.box).is_auto())
  1563. return inner_max_content_size;
  1564. auto inner_preferred_size = is_row_layout() ? get_pixel_width(item.box, computed_main_size(item.box)) : get_pixel_height(item.box, computed_main_size(item.box));
  1565. return max(inner_max_content_size, inner_preferred_size);
  1566. }();
  1567. auto clamp_min = has_main_min_size(item.box) ? specified_main_min_size(item.box) : automatic_minimum_size(item);
  1568. auto clamp_max = has_main_max_size(item.box) ? specified_main_max_size(item.box) : CSSPixels::max();
  1569. auto clamped_inner_size = css_clamp(larger_size, clamp_min, clamp_max);
  1570. return item.add_main_margin_box_sizes(clamped_inner_size);
  1571. }
  1572. bool FlexFormattingContext::should_treat_main_size_as_auto(Box const& box) const
  1573. {
  1574. if (is_row_layout())
  1575. return should_treat_width_as_auto(box, m_available_space_for_items->space);
  1576. return should_treat_height_as_auto(box, m_available_space_for_items->space);
  1577. }
  1578. bool FlexFormattingContext::should_treat_cross_size_as_auto(Box const& box) const
  1579. {
  1580. if (is_row_layout())
  1581. return should_treat_height_as_auto(box, m_available_space_for_items->space);
  1582. return should_treat_width_as_auto(box, m_available_space_for_items->space);
  1583. }
  1584. bool FlexFormattingContext::should_treat_main_max_size_as_none(Box const& box) const
  1585. {
  1586. if (is_row_layout())
  1587. return should_treat_max_width_as_none(box, m_available_space_for_items->space.width);
  1588. return should_treat_max_height_as_none(box, m_available_space_for_items->space.height);
  1589. }
  1590. bool FlexFormattingContext::should_treat_cross_max_size_as_none(Box const& box) const
  1591. {
  1592. if (is_row_layout())
  1593. return should_treat_max_height_as_none(box, m_available_space_for_items->space.height);
  1594. return should_treat_max_width_as_none(box, m_available_space_for_items->space.width);
  1595. }
  1596. CSSPixels FlexFormattingContext::calculate_cross_min_content_contribution(FlexItem const& item, bool resolve_percentage_min_max_sizes) const
  1597. {
  1598. auto size = [&] {
  1599. if (should_treat_cross_size_as_auto(item.box))
  1600. return calculate_min_content_cross_size(item);
  1601. return !is_row_layout() ? get_pixel_width(item.box, computed_cross_size(item.box)) : get_pixel_height(item.box, computed_cross_size(item.box));
  1602. }();
  1603. auto const& computed_min_size = this->computed_cross_min_size(item.box);
  1604. auto const& computed_max_size = this->computed_cross_max_size(item.box);
  1605. auto clamp_min = (!computed_min_size.is_auto() && (resolve_percentage_min_max_sizes || !computed_min_size.contains_percentage())) ? specified_cross_min_size(item.box) : 0;
  1606. auto clamp_max = (!should_treat_cross_max_size_as_none(item.box) && (resolve_percentage_min_max_sizes || !computed_max_size.contains_percentage())) ? specified_cross_max_size(item.box) : CSSPixels::max();
  1607. auto clamped_inner_size = css_clamp(size, clamp_min, clamp_max);
  1608. return item.add_cross_margin_box_sizes(clamped_inner_size);
  1609. }
  1610. CSSPixels FlexFormattingContext::calculate_cross_max_content_contribution(FlexItem const& item, bool resolve_percentage_min_max_sizes) const
  1611. {
  1612. auto size = [&] {
  1613. if (should_treat_cross_size_as_auto(item.box))
  1614. return calculate_max_content_cross_size(item);
  1615. return !is_row_layout() ? get_pixel_width(item.box, computed_cross_size(item.box)) : get_pixel_height(item.box, computed_cross_size(item.box));
  1616. }();
  1617. auto const& computed_min_size = this->computed_cross_min_size(item.box);
  1618. auto const& computed_max_size = this->computed_cross_max_size(item.box);
  1619. auto clamp_min = (!computed_min_size.is_auto() && (resolve_percentage_min_max_sizes || !computed_min_size.contains_percentage())) ? specified_cross_min_size(item.box) : 0;
  1620. auto clamp_max = (!should_treat_cross_max_size_as_none(item.box) && (resolve_percentage_min_max_sizes || !computed_max_size.contains_percentage())) ? specified_cross_max_size(item.box) : CSSPixels::max();
  1621. auto clamped_inner_size = css_clamp(size, clamp_min, clamp_max);
  1622. return item.add_cross_margin_box_sizes(clamped_inner_size);
  1623. }
  1624. CSSPixels FlexFormattingContext::calculate_width_to_use_when_determining_intrinsic_height_of_item(FlexItem const& item) const
  1625. {
  1626. auto const& box = *item.box;
  1627. auto computed_width = box.computed_values().width();
  1628. auto const& computed_min_width = box.computed_values().min_width();
  1629. auto const& computed_max_width = box.computed_values().max_width();
  1630. auto clamp_min = (!computed_min_width.is_auto() && (!computed_min_width.contains_percentage())) ? get_pixel_width(box, computed_min_width) : 0;
  1631. auto clamp_max = (!should_treat_max_width_as_none(box, m_available_space_for_items->space.width) && (!computed_max_width.contains_percentage())) ? get_pixel_width(box, computed_max_width) : CSSPixels::max();
  1632. CSSPixels width;
  1633. if (should_treat_width_as_auto(box, m_available_space_for_items->space) || computed_width.is_fit_content())
  1634. width = calculate_fit_content_width(box, m_available_space_for_items->space);
  1635. else if (computed_width.is_min_content())
  1636. width = calculate_min_content_width(box);
  1637. else if (computed_width.is_max_content())
  1638. width = calculate_max_content_width(box);
  1639. return css_clamp(width, clamp_min, clamp_max);
  1640. }
  1641. CSSPixels FlexFormattingContext::calculate_min_content_main_size(FlexItem const& item) const
  1642. {
  1643. if (is_row_layout()) {
  1644. return calculate_min_content_width(item.box);
  1645. }
  1646. auto available_space = item.used_values.available_inner_space_or_constraints_from(m_available_space_for_items->space);
  1647. if (available_space.width.is_indefinite()) {
  1648. available_space.width = AvailableSize::make_definite(calculate_width_to_use_when_determining_intrinsic_height_of_item(item));
  1649. }
  1650. return calculate_min_content_height(item.box, available_space.width.to_px_or_zero());
  1651. }
  1652. CSSPixels FlexFormattingContext::calculate_max_content_main_size(FlexItem const& item) const
  1653. {
  1654. if (is_row_layout()) {
  1655. return calculate_max_content_width(item.box);
  1656. }
  1657. auto available_space = item.used_values.available_inner_space_or_constraints_from(m_available_space_for_items->space);
  1658. if (available_space.width.is_indefinite()) {
  1659. available_space.width = AvailableSize::make_definite(calculate_width_to_use_when_determining_intrinsic_height_of_item(item));
  1660. }
  1661. return calculate_max_content_height(item.box, available_space.width.to_px_or_zero());
  1662. }
  1663. CSSPixels FlexFormattingContext::calculate_fit_content_main_size(FlexItem const& item) const
  1664. {
  1665. if (is_row_layout())
  1666. return calculate_fit_content_width(item.box, m_available_space_for_items->space);
  1667. return calculate_fit_content_height(item.box, m_available_space_for_items->space);
  1668. }
  1669. CSSPixels FlexFormattingContext::calculate_fit_content_cross_size(FlexItem const& item) const
  1670. {
  1671. if (!is_row_layout())
  1672. return calculate_fit_content_width(item.box, m_available_space_for_items->space);
  1673. return calculate_fit_content_height(item.box, m_available_space_for_items->space);
  1674. }
  1675. CSSPixels FlexFormattingContext::calculate_min_content_cross_size(FlexItem const& item) const
  1676. {
  1677. if (is_row_layout()) {
  1678. auto available_space = item.used_values.available_inner_space_or_constraints_from(m_available_space_for_items->space);
  1679. if (available_space.width.is_indefinite()) {
  1680. available_space.width = AvailableSize::make_definite(calculate_width_to_use_when_determining_intrinsic_height_of_item(item));
  1681. }
  1682. return calculate_min_content_height(item.box, available_space.width.to_px_or_zero());
  1683. }
  1684. return calculate_min_content_width(item.box);
  1685. }
  1686. CSSPixels FlexFormattingContext::calculate_max_content_cross_size(FlexItem const& item) const
  1687. {
  1688. if (is_row_layout()) {
  1689. auto available_space = item.used_values.available_inner_space_or_constraints_from(m_available_space_for_items->space);
  1690. if (available_space.width.is_indefinite()) {
  1691. available_space.width = AvailableSize::make_definite(calculate_width_to_use_when_determining_intrinsic_height_of_item(item));
  1692. }
  1693. return calculate_max_content_height(item.box, available_space.width.to_px_or_zero());
  1694. }
  1695. return calculate_max_content_width(item.box);
  1696. }
  1697. // https://drafts.csswg.org/css-flexbox-1/#stretched
  1698. bool FlexFormattingContext::flex_item_is_stretched(FlexItem const& item) const
  1699. {
  1700. auto alignment = alignment_for_item(item.box);
  1701. if (alignment != CSS::AlignItems::Stretch && alignment != CSS::AlignItems::Normal)
  1702. return false;
  1703. // If the cross size property of the flex item computes to auto, and neither of the cross-axis margins are auto, the flex item is stretched.
  1704. auto const& computed_cross_size = is_row_layout() ? item.box->computed_values().height() : item.box->computed_values().width();
  1705. return computed_cross_size.is_auto() && !item.margins.cross_before_is_auto && !item.margins.cross_after_is_auto;
  1706. }
  1707. CSS::Size const& FlexFormattingContext::computed_main_size(Box const& box) const
  1708. {
  1709. return is_row_layout() ? box.computed_values().width() : box.computed_values().height();
  1710. }
  1711. CSS::Size const& FlexFormattingContext::computed_main_min_size(Box const& box) const
  1712. {
  1713. return is_row_layout() ? box.computed_values().min_width() : box.computed_values().min_height();
  1714. }
  1715. CSS::Size const& FlexFormattingContext::computed_main_max_size(Box const& box) const
  1716. {
  1717. return is_row_layout() ? box.computed_values().max_width() : box.computed_values().max_height();
  1718. }
  1719. CSS::Size const& FlexFormattingContext::computed_cross_size(Box const& box) const
  1720. {
  1721. return !is_row_layout() ? box.computed_values().width() : box.computed_values().height();
  1722. }
  1723. CSS::Size const& FlexFormattingContext::computed_cross_min_size(Box const& box) const
  1724. {
  1725. return !is_row_layout() ? box.computed_values().min_width() : box.computed_values().min_height();
  1726. }
  1727. CSS::Size const& FlexFormattingContext::computed_cross_max_size(Box const& box) const
  1728. {
  1729. return !is_row_layout() ? box.computed_values().max_width() : box.computed_values().max_height();
  1730. }
  1731. // https://drafts.csswg.org/css-flexbox-1/#algo-cross-margins
  1732. void FlexFormattingContext::resolve_cross_axis_auto_margins()
  1733. {
  1734. for (auto& line : m_flex_lines) {
  1735. for (auto& item : line.items) {
  1736. // If a flex item has auto cross-axis margins:
  1737. if (!item.margins.cross_before_is_auto && !item.margins.cross_after_is_auto)
  1738. continue;
  1739. // If its outer cross size (treating those auto margins as zero) is less than the cross size of its flex line,
  1740. // distribute the difference in those sizes equally to the auto margins.
  1741. auto outer_cross_size = item.cross_size.value() + item.padding.cross_before + item.padding.cross_after + item.borders.cross_before + item.borders.cross_after;
  1742. if (outer_cross_size < line.cross_size) {
  1743. CSSPixels remainder = line.cross_size - outer_cross_size;
  1744. if (item.margins.cross_before_is_auto && item.margins.cross_after_is_auto) {
  1745. item.margins.cross_before = remainder / 2;
  1746. item.margins.cross_after = remainder / 2;
  1747. } else if (item.margins.cross_before_is_auto) {
  1748. item.margins.cross_before = remainder;
  1749. } else {
  1750. item.margins.cross_after = remainder;
  1751. }
  1752. } else {
  1753. // FIXME: Otherwise, if the block-start or inline-start margin (whichever is in the cross axis) is auto, set it to zero.
  1754. // Set the opposite margin so that the outer cross size of the item equals the cross size of its flex line.
  1755. }
  1756. }
  1757. }
  1758. }
  1759. // https://drafts.csswg.org/css-flexbox-1/#algo-line-stretch
  1760. void FlexFormattingContext::handle_align_content_stretch()
  1761. {
  1762. // If the flex container has a definite cross size,
  1763. if (!has_definite_cross_size(m_flex_container_state))
  1764. return;
  1765. // align-content is stretch,
  1766. if (flex_container().computed_values().align_content() != CSS::AlignContent::Stretch && flex_container().computed_values().align_content() != CSS::AlignContent::Normal)
  1767. return;
  1768. // and the sum of the flex lines' cross sizes is less than the flex container’s inner cross size,
  1769. CSSPixels sum_of_flex_line_cross_sizes = 0;
  1770. for (auto& line : m_flex_lines)
  1771. sum_of_flex_line_cross_sizes += line.cross_size;
  1772. // CSS-FLEXBOX-2: Account for gap between flex lines.
  1773. sum_of_flex_line_cross_sizes += cross_gap() * (m_flex_lines.size() - 1);
  1774. if (sum_of_flex_line_cross_sizes >= inner_cross_size(m_flex_container_state))
  1775. return;
  1776. // increase the cross size of each flex line by equal amounts
  1777. // such that the sum of their cross sizes exactly equals the flex container’s inner cross size.
  1778. CSSPixels remainder = inner_cross_size(m_flex_container_state) - sum_of_flex_line_cross_sizes;
  1779. CSSPixels extra_per_line = remainder / m_flex_lines.size();
  1780. for (auto& line : m_flex_lines)
  1781. line.cross_size += extra_per_line;
  1782. }
  1783. // https://drafts.csswg.org/css-flexbox-1/#abspos-items
  1784. CSSPixelPoint FlexFormattingContext::calculate_static_position(Box const& box) const
  1785. {
  1786. // The cross-axis edges of the static-position rectangle of an absolutely-positioned child
  1787. // of a flex container are the content edges of the flex container.
  1788. CSSPixels cross_offset = 0;
  1789. CSSPixels half_line_size = inner_cross_size(m_flex_container_state) / 2;
  1790. auto const& box_state = m_state.get(box);
  1791. CSSPixels cross_margin_before = is_row_layout() ? box_state.margin_top : box_state.margin_left;
  1792. CSSPixels cross_margin_after = is_row_layout() ? box_state.margin_bottom : box_state.margin_right;
  1793. CSSPixels cross_border_before = is_row_layout() ? box_state.border_top : box_state.border_left;
  1794. CSSPixels cross_border_after = is_row_layout() ? box_state.border_bottom : box_state.border_right;
  1795. CSSPixels cross_padding_after = is_row_layout() ? box_state.padding_bottom : box_state.padding_right;
  1796. CSSPixels main_border_before = is_row_layout() ? box_state.border_left : box_state.border_top;
  1797. CSSPixels main_border_after = is_row_layout() ? box_state.border_right : box_state.border_bottom;
  1798. switch (alignment_for_item(box)) {
  1799. case CSS::AlignItems::Baseline:
  1800. // FIXME: Implement this
  1801. // Fallthrough
  1802. case CSS::AlignItems::Start:
  1803. case CSS::AlignItems::FlexStart:
  1804. case CSS::AlignItems::SelfStart:
  1805. case CSS::AlignItems::Stretch:
  1806. case CSS::AlignItems::Normal:
  1807. cross_offset = -half_line_size + cross_margin_before;
  1808. break;
  1809. case CSS::AlignItems::End:
  1810. case CSS::AlignItems::SelfEnd:
  1811. case CSS::AlignItems::FlexEnd:
  1812. cross_offset = half_line_size - inner_cross_size(box_state) - cross_margin_after - cross_border_after - cross_padding_after;
  1813. break;
  1814. case CSS::AlignItems::Center:
  1815. cross_offset = -((inner_cross_size(box_state) + cross_border_before + cross_border_after) / 2);
  1816. break;
  1817. default:
  1818. break;
  1819. }
  1820. cross_offset += inner_cross_size(m_flex_container_state) / 2;
  1821. // The main-axis edges of the static-position rectangle are where the margin edges of the child
  1822. // would be positioned if it were the sole flex item in the flex container,
  1823. // assuming both the child and the flex container were fixed-size boxes of their used size.
  1824. // (For this purpose, auto margins are treated as zero.
  1825. bool pack_from_end = true;
  1826. CSSPixels main_offset = 0;
  1827. switch (flex_container().computed_values().justify_content()) {
  1828. case CSS::JustifyContent::Start:
  1829. pack_from_end = false;
  1830. break;
  1831. case CSS::JustifyContent::Stretch:
  1832. case CSS::JustifyContent::Normal:
  1833. case CSS::JustifyContent::FlexStart:
  1834. case CSS::JustifyContent::SpaceBetween:
  1835. pack_from_end = is_direction_reverse();
  1836. break;
  1837. case CSS::JustifyContent::End:
  1838. pack_from_end = true;
  1839. break;
  1840. case CSS::JustifyContent::FlexEnd:
  1841. pack_from_end = !is_direction_reverse();
  1842. break;
  1843. case CSS::JustifyContent::Center:
  1844. case CSS::JustifyContent::SpaceAround:
  1845. case CSS::JustifyContent::SpaceEvenly:
  1846. pack_from_end = false;
  1847. main_offset = (inner_main_size(m_flex_container_state) - inner_main_size(box_state) - main_border_before - main_border_after) / 2;
  1848. break;
  1849. }
  1850. if (pack_from_end)
  1851. main_offset += inner_main_size(m_flex_container_state) - inner_main_size(box_state) - main_border_before - main_border_after;
  1852. auto static_position_offset = is_row_layout() ? CSSPixelPoint { main_offset, cross_offset } : CSSPixelPoint { cross_offset, main_offset };
  1853. auto absolute_position_of_flex_container = absolute_content_rect(flex_container()).location();
  1854. auto absolute_position_of_abspos_containing_block = absolute_content_rect(*box.containing_block()).location();
  1855. auto diff = absolute_position_of_flex_container - absolute_position_of_abspos_containing_block;
  1856. return static_position_offset + diff;
  1857. }
  1858. double FlexFormattingContext::FlexLine::sum_of_flex_factor_of_unfrozen_items() const
  1859. {
  1860. double sum = 0;
  1861. for (auto const& item : items) {
  1862. if (!item.frozen)
  1863. sum += item.flex_factor.value();
  1864. }
  1865. return sum;
  1866. }
  1867. double FlexFormattingContext::FlexLine::sum_of_scaled_flex_shrink_factor_of_unfrozen_items() const
  1868. {
  1869. double sum = 0;
  1870. for (auto const& item : items) {
  1871. if (!item.frozen)
  1872. sum += item.scaled_flex_shrink_factor;
  1873. }
  1874. return sum;
  1875. }
  1876. CSSPixels FlexFormattingContext::main_gap() const
  1877. {
  1878. auto const& computed_values = flex_container().computed_values();
  1879. auto gap = is_row_layout() ? computed_values.column_gap() : computed_values.row_gap();
  1880. return gap.to_px(flex_container(), inner_main_size(m_flex_container_state));
  1881. }
  1882. CSSPixels FlexFormattingContext::cross_gap() const
  1883. {
  1884. auto const& computed_values = flex_container().computed_values();
  1885. auto gap = is_row_layout() ? computed_values.row_gap() : computed_values.column_gap();
  1886. return gap.to_px(flex_container(), inner_cross_size(m_flex_container_state));
  1887. }
  1888. }