1use super::types::{GridItem, GridTrack, TrackCounts};
4use crate::geometry::{AbstractAxis, Line, Size};
5use crate::style::{AlignContent, AlignContentKeyword, AlignSelf, AvailableSpace};
6use crate::style_helpers::TaffyMinContent;
7use crate::tree::{LayoutPartialTree, LayoutPartialTreeExt, SizingMode};
8use crate::util::sys::{f32_max, f32_min, Vec};
9use crate::util::{MaybeMath, ResolveOrZero};
10use crate::CompactLength;
11use core::cmp::Ordering;
12
13struct ItemBatcher {
17 axis: AbstractAxis,
19 index_offset: usize,
21 current_span: u16,
23 current_is_flex: bool,
25}
26
27impl ItemBatcher {
28 #[inline(always)]
30 fn new(axis: AbstractAxis) -> Self {
31 ItemBatcher { index_offset: 0, axis, current_span: 1, current_is_flex: false }
32 }
33
34 #[inline]
37 fn next<'items>(&mut self, items: &'items mut [GridItem]) -> Option<(&'items mut [GridItem], bool)> {
38 if self.current_is_flex || self.index_offset >= items.len() {
39 return None;
40 }
41
42 let item = &items[self.index_offset];
43 self.current_span = item.span(self.axis);
44 self.current_is_flex = item.crosses_flexible_track(self.axis);
45
46 let next_index_offset = if self.current_is_flex {
47 items.len()
48 } else {
49 items
50 .iter()
51 .position(|item: &GridItem| {
52 item.crosses_flexible_track(self.axis) || item.span(self.axis) > self.current_span
53 })
54 .unwrap_or(items.len())
55 };
56
57 let batch_range = self.index_offset..next_index_offset;
58 self.index_offset = next_index_offset;
59
60 let batch = &mut items[batch_range];
61 Some((batch, self.current_is_flex))
62 }
63}
64
65struct IntrinsicSizeMeasurer<'tree, 'oat, Tree, EstimateFunction>
68where
69 Tree: LayoutPartialTree,
70 EstimateFunction: Fn(&GridTrack, Option<f32>, &Tree) -> Option<f32>,
71{
72 tree: &'tree mut Tree,
74 other_axis_tracks: &'oat [GridTrack],
76 get_track_size_estimate: EstimateFunction,
79 axis: AbstractAxis,
81 inner_node_size: Size<Option<f32>>,
83}
84
85impl<Tree, EstimateFunction> IntrinsicSizeMeasurer<'_, '_, Tree, EstimateFunction>
86where
87 Tree: LayoutPartialTree,
88 EstimateFunction: Fn(&GridTrack, Option<f32>, &Tree) -> Option<f32>,
89{
90 #[inline(always)]
95 fn grid_area_size(&self, item: &mut GridItem, axis_tracks: &[GridTrack]) -> Size<Option<f32>> {
96 item.grid_area_size_cached(
97 self.axis,
98 axis_tracks,
99 self.other_axis_tracks,
100 self.inner_node_size,
101 |track, basis| (self.get_track_size_estimate)(track, basis, self.tree),
102 &|val, basis| self.tree.calc(val, basis),
103 )
104 }
105
106 #[inline(always)]
109 fn margins_axis_sums_with_baseline_shims(&self, item: &GridItem, percentage_basis: Option<f32>) -> Size<f32> {
110 item.margins_axis_sums_with_baseline_shims(percentage_basis, self.tree)
111 }
112
113 #[inline(always)]
115 fn calc(&self, val: *const (), basis: f32) -> f32 {
116 self.tree.calc(val, basis)
117 }
118
119 #[inline(always)]
121 fn min_content_contribution(&mut self, item: &mut GridItem, axis_tracks: &[GridTrack]) -> f32 {
122 let grid_area_size = self.grid_area_size(item, axis_tracks);
123 let available_space = grid_area_size.with(self.axis, None);
124 let margin_axis_sums = self.margins_axis_sums_with_baseline_shims(item, available_space.width);
125 let contribution = item.min_content_contribution_cached(self.axis, self.tree, grid_area_size, available_space);
126 contribution + margin_axis_sums.get(self.axis)
127 }
128
129 #[inline(always)]
131 fn max_content_contribution(&mut self, item: &mut GridItem, axis_tracks: &[GridTrack]) -> f32 {
132 let grid_area_size = self.grid_area_size(item, axis_tracks);
133 let available_space = grid_area_size.with(self.axis, None);
134 let margin_axis_sums = self.margins_axis_sums_with_baseline_shims(item, available_space.width);
135 let contribution = item.max_content_contribution_cached(self.axis, self.tree, grid_area_size, available_space);
136 contribution + margin_axis_sums.get(self.axis)
137 }
138
139 #[inline(always)]
147 fn minimum_contribution(&mut self, item: &mut GridItem, axis_tracks: &[GridTrack]) -> f32 {
148 let grid_area_size = self.grid_area_size(item, axis_tracks);
149 let available_space = grid_area_size.with(self.axis, None);
150 let margin_axis_sums = self.margins_axis_sums_with_baseline_shims(item, available_space.width);
151 let contribution =
152 item.minimum_contribution_cached(self.tree, self.axis, axis_tracks, grid_area_size, self.inner_node_size);
153 contribution + margin_axis_sums.get(self.axis)
154 }
155}
156
157#[inline(always)]
160pub(super) fn cmp_by_cross_flex_then_span_then_start(
161 axis: AbstractAxis,
162) -> impl FnMut(&GridItem, &GridItem) -> Ordering {
163 move |item_a: &GridItem, item_b: &GridItem| -> Ordering {
164 match (item_a.crosses_flexible_track(axis), item_b.crosses_flexible_track(axis)) {
165 (false, true) => Ordering::Less,
166 (true, false) => Ordering::Greater,
167 _ => {
168 let placement_a = item_a.placement(axis);
169 let placement_b = item_b.placement(axis);
170 match placement_a.span().cmp(&placement_b.span()) {
171 Ordering::Less => Ordering::Less,
172 Ordering::Greater => Ordering::Greater,
173 Ordering::Equal => placement_a.start.cmp(&placement_b.start),
174 }
175 }
176 }
177 }
178}
179
180#[inline(always)]
184pub(super) fn compute_alignment_gutter_adjustment(
185 alignment: AlignContent,
186 axis_inner_node_size: Option<f32>,
187 get_track_size_estimate: impl Fn(&GridTrack, Option<f32>) -> Option<f32>,
188 tracks: &[GridTrack],
189) -> f32 {
190 if tracks.len() <= 1 {
191 return 0.0;
192 }
193
194 let outer_gutter_weight = match alignment.keyword() {
198 AlignContentKeyword::Start
199 | AlignContentKeyword::FlexStart
200 | AlignContentKeyword::End
201 | AlignContentKeyword::FlexEnd
202 | AlignContentKeyword::Center => 1,
203 AlignContentKeyword::Stretch => 0,
204 AlignContentKeyword::SpaceBetween => 0,
205 AlignContentKeyword::SpaceAround => 1,
206 AlignContentKeyword::SpaceEvenly => 1,
207 };
208
209 let inner_gutter_weight = match alignment.keyword() {
210 AlignContentKeyword::FlexStart
211 | AlignContentKeyword::Start
212 | AlignContentKeyword::FlexEnd
213 | AlignContentKeyword::End
214 | AlignContentKeyword::Center
215 | AlignContentKeyword::Stretch => 0,
216 AlignContentKeyword::SpaceBetween => 1,
217 AlignContentKeyword::SpaceAround => 2,
218 AlignContentKeyword::SpaceEvenly => 1,
219 };
220
221 if inner_gutter_weight == 0 {
222 return 0.0;
223 }
224
225 if let Some(axis_inner_node_size) = axis_inner_node_size {
226 let free_space = tracks
227 .iter()
228 .map(|track| get_track_size_estimate(track, Some(axis_inner_node_size)))
229 .sum::<Option<f32>>()
230 .map(|track_size_sum| f32_max(0.0, axis_inner_node_size - track_size_sum))
231 .unwrap_or(0.0);
232
233 let weighted_track_count =
234 (((tracks.len() - 3) / 2) * inner_gutter_weight as usize) + (2 * outer_gutter_weight as usize);
235
236 return (free_space / weighted_track_count as f32) * inner_gutter_weight as f32;
237 }
238
239 0.0
240}
241
242#[inline(always)]
244pub(super) fn resolve_item_track_indexes(items: &mut [GridItem], column_counts: TrackCounts, row_counts: TrackCounts) {
245 for item in items {
246 item.column_indexes = item.column.map(|line| line.into_track_vec_index(column_counts) as u16);
247 item.row_indexes = item.row.map(|line| line.into_track_vec_index(row_counts) as u16);
248 }
249}
250
251#[inline(always)]
253pub(super) fn determine_if_item_crosses_flexible_or_intrinsic_tracks(
254 items: &mut Vec<GridItem>,
255 columns: &[GridTrack],
256 rows: &[GridTrack],
257) {
258 for item in items {
259 item.crosses_flexible_column =
260 item.track_range_excluding_lines(AbstractAxis::Inline).any(|i| columns[i].is_flexible());
261 item.crosses_intrinsic_column =
262 item.track_range_excluding_lines(AbstractAxis::Inline).any(|i| columns[i].has_intrinsic_sizing_function());
263 item.crosses_flexible_row =
264 item.track_range_excluding_lines(AbstractAxis::Block).any(|i| rows[i].is_flexible());
265 item.crosses_intrinsic_row =
266 item.track_range_excluding_lines(AbstractAxis::Block).any(|i| rows[i].has_intrinsic_sizing_function());
267 }
268}
269
270#[allow(clippy::too_many_arguments)]
273pub(super) fn track_sizing_algorithm<Tree: LayoutPartialTree>(
274 tree: &mut Tree,
275 axis: AbstractAxis,
276 axis_min_size: Option<f32>,
277 axis_max_size: Option<f32>,
278 axis_alignment: AlignContent,
279 other_axis_alignment: AlignContent,
280 available_grid_space: Size<AvailableSpace>,
281 inner_node_size: Size<Option<f32>>,
282 axis_tracks: &mut [GridTrack],
283 other_axis_tracks: &mut [GridTrack],
284 items: &mut [GridItem],
285 get_track_size_estimate: fn(&GridTrack, Option<f32>, &Tree) -> Option<f32>,
286 has_baseline_aligned_item: bool,
287) {
288 let percentage_basis = inner_node_size.get(axis).or(axis_min_size);
291 initialize_track_sizes(tree, axis_tracks, percentage_basis);
292
293 if has_baseline_aligned_item {
295 resolve_item_baselines(tree, axis, items, inner_node_size);
296 }
297
298 if axis_tracks.iter().all(|track| track.base_size == track.growth_limit) {
301 return;
302 }
303
304 let gutter_alignment_adjustment = compute_alignment_gutter_adjustment(
309 other_axis_alignment,
310 inner_node_size.get(axis.other()),
311 |track, basis| get_track_size_estimate(track, basis, tree),
312 other_axis_tracks,
313 );
314 if other_axis_tracks.len() > 3 {
315 let len = other_axis_tracks.len();
316 let inner_gutter_tracks = other_axis_tracks[2..len].iter_mut().step_by(2);
317 for track in inner_gutter_tracks {
318 track.content_alignment_adjustment = gutter_alignment_adjustment;
319 }
320 }
321
322 resolve_intrinsic_track_sizes(
324 tree,
325 axis,
326 axis_tracks,
327 other_axis_tracks,
328 items,
329 available_grid_space.get(axis),
330 inner_node_size,
331 get_track_size_estimate,
332 );
333
334 maximise_tracks(axis_tracks, inner_node_size.get(axis), available_grid_space.get(axis));
337
338 let axis_available_space_for_expansion = if let Some(available_space) = inner_node_size.get(axis) {
343 AvailableSpace::Definite(available_space)
344 } else {
345 match available_grid_space.get(axis) {
346 AvailableSpace::MinContent => AvailableSpace::MinContent,
347 AvailableSpace::MaxContent | AvailableSpace::Definite(_) => AvailableSpace::MaxContent,
348 }
349 };
350
351 expand_flexible_tracks(
354 tree,
355 axis,
356 axis_tracks,
357 items,
358 axis_min_size,
359 axis_max_size,
360 axis_available_space_for_expansion,
361 );
362
363 if axis_alignment == AlignContent::STRETCH {
366 stretch_auto_tracks(axis_tracks, axis_min_size, axis_available_space_for_expansion);
367 }
368}
369
370#[derive(Copy, Clone, Debug, PartialEq, Eq)]
374enum IntrinsicContributionType {
375 Minimum,
377 Maximum,
379}
380
381#[inline(always)]
384fn flush_planned_base_size_increases(tracks: &mut [GridTrack]) {
385 for track in tracks {
386 track.base_size += track.base_size_planned_increase;
387 track.base_size_planned_increase = 0.0;
388 }
389}
390
391#[inline(always)]
394fn flush_planned_growth_limit_increases(tracks: &mut [GridTrack], set_infinitely_growable: bool) {
395 for track in tracks {
396 if track.growth_limit_planned_increase > 0.0 {
397 track.growth_limit = if track.growth_limit == f32::INFINITY {
398 track.base_size + track.growth_limit_planned_increase
399 } else {
400 track.growth_limit + track.growth_limit_planned_increase
401 };
402 track.infinitely_growable = set_infinitely_growable;
403 } else {
404 track.infinitely_growable = false;
405 }
406 track.growth_limit_planned_increase = 0.0
407 }
408}
409
410#[inline(always)]
413fn initialize_track_sizes(
414 tree: &impl LayoutPartialTree,
415 axis_tracks: &mut [GridTrack],
416 axis_inner_node_size: Option<f32>,
417) {
418 for track in axis_tracks.iter_mut() {
419 track.base_size = track
426 .min_track_sizing_function
427 .definite_value(axis_inner_node_size, |val, basis| tree.calc(val, basis))
428 .unwrap_or(0.0);
429
430 track.growth_limit = track
438 .max_track_sizing_function
439 .definite_value(axis_inner_node_size, |val, basis| tree.calc(val, basis))
440 .unwrap_or(f32::INFINITY);
441
442 if track.growth_limit < track.base_size {
444 track.growth_limit = track.base_size;
445 }
446 }
447}
448
449fn resolve_item_baselines(
451 tree: &mut impl LayoutPartialTree,
452 axis: AbstractAxis,
453 items: &mut [GridItem],
454 inner_node_size: Size<Option<f32>>,
455) {
456 let other_axis = axis.other();
459 items.sort_by_key(|item| item.placement(other_axis).start);
460
461 let mut remaining_items = &mut items[0..];
463 while !remaining_items.is_empty() {
464 let current_row = remaining_items[0].placement(other_axis).start;
466
467 let next_row_first_item =
469 remaining_items.iter().position(|item| item.placement(other_axis).start != current_row);
470
471 let row_items = if let Some(index) = next_row_first_item {
476 let (row_items, tail) = remaining_items.split_at_mut(index);
477 remaining_items = tail;
478 row_items
479 } else {
480 let row_items = remaining_items;
481 remaining_items = &mut [];
482 row_items
483 };
484
485 let row_baseline_item_count = row_items.iter().filter(|item| item.align_self == AlignSelf::BASELINE).count();
489 if row_baseline_item_count <= 1 {
490 continue;
491 }
492
493 for item in row_items.iter_mut() {
495 let measured_size_and_baselines = tree.perform_child_layout(
496 item.node,
497 Size::NONE,
498 inner_node_size,
499 Size::MIN_CONTENT,
500 SizingMode::InherentSize,
501 Line::FALSE,
502 );
503
504 let baseline = measured_size_and_baselines.first_baselines.y;
505 let height = measured_size_and_baselines.size.height;
506
507 item.baseline = Some(
508 baseline.unwrap_or(height)
509 + item.margin.top.resolve_or_zero(inner_node_size.width, |val, basis| tree.calc(val, basis)),
510 );
511 }
512
513 let row_max_baseline =
515 row_items.iter().map(|item| item.baseline.unwrap_or(0.0)).max_by(|a, b| a.total_cmp(b)).unwrap();
516
517 for item in row_items.iter_mut() {
519 item.baseline_shim = row_max_baseline - item.baseline.unwrap_or(0.0);
520 }
521 }
522}
523
524#[allow(clippy::too_many_arguments)]
526fn resolve_intrinsic_track_sizes<Tree: LayoutPartialTree>(
527 tree: &mut Tree,
528 axis: AbstractAxis,
529 axis_tracks: &mut [GridTrack],
530 other_axis_tracks: &[GridTrack],
531 items: &mut [GridItem],
532 axis_available_grid_space: AvailableSpace,
533 inner_node_size: Size<Option<f32>>,
534 get_track_size_estimate: impl Fn(&GridTrack, Option<f32>, &Tree) -> Option<f32>,
535) {
536 items.sort_by(cmp_by_cross_flex_then_span_then_start(axis));
546
547 let axis_inner_node_size = inner_node_size.get(axis);
559 let flex_factor_sum = axis_tracks.iter().map(|track| track.flex_factor()).sum::<f32>();
560 let mut item_sizer =
561 IntrinsicSizeMeasurer { tree, other_axis_tracks, axis, inner_node_size, get_track_size_estimate };
562
563 let mut batched_item_iterator = ItemBatcher::new(axis);
564 while let Some((batch, is_flex)) = batched_item_iterator.next(items) {
565 let batch_span = batch[0].placement(axis).span();
568 if !is_flex && batch_span == 1 {
569 for item in batch.iter_mut() {
570 let track_index = item.placement_indexes(axis).start + 1;
571 let track = &axis_tracks[track_index as usize];
572
573 let new_base_size = match track.min_track_sizing_function.0.tag() {
575 CompactLength::MIN_CONTENT_TAG => {
576 f32_max(track.base_size, item_sizer.min_content_contribution(item, axis_tracks))
577 }
578 CompactLength::PERCENT_TAG => {
581 if axis_inner_node_size.is_none() {
582 f32_max(track.base_size, item_sizer.min_content_contribution(item, axis_tracks))
583 } else {
584 track.base_size
585 }
586 }
587 CompactLength::MAX_CONTENT_TAG => {
588 f32_max(track.base_size, item_sizer.max_content_contribution(item, axis_tracks))
589 }
590 CompactLength::AUTO_TAG => {
591 let space = match axis_available_grid_space {
592 AvailableSpace::MinContent | AvailableSpace::MaxContent
601 if !item.overflow.get(axis).is_scroll_container() =>
602 {
603 let axis_minimum_size = item_sizer.minimum_contribution(item, axis_tracks);
604 let axis_min_content_size = item_sizer.min_content_contribution(item, axis_tracks);
605 let limit = track
606 .max_track_sizing_function
607 .definite_limit(axis_inner_node_size, |val, basis| item_sizer.calc(val, basis));
608 axis_min_content_size.maybe_min(limit).max(axis_minimum_size)
609 }
610 _ => item_sizer.minimum_contribution(item, axis_tracks),
611 };
612 f32_max(track.base_size, space)
613 }
614 CompactLength::LENGTH_TAG => {
615 track.base_size
617 }
618 #[cfg(feature = "calc")]
620 _ if track.min_track_sizing_function.0.is_calc() => {
621 if axis_inner_node_size.is_none() {
622 f32_max(track.base_size, item_sizer.min_content_contribution(item, axis_tracks))
623 } else {
624 track.base_size
625 }
626 }
627 _ => unreachable!(),
628 };
629 let growth_limit_min_content_contribution = if !item.overflow.get(axis).is_scroll_container() {
630 Some(item_sizer.min_content_contribution(item, axis_tracks))
631 } else {
632 None
633 };
634 let growth_limit_max_content_contribution = item_sizer.max_content_contribution(item, axis_tracks);
635 let growth_limit_intrinsic_min_content_contribution =
636 item_sizer.min_content_contribution(item, axis_tracks);
637 let track = &mut axis_tracks[track_index as usize];
638 track.base_size = new_base_size;
639
640 if track.max_track_sizing_function.is_fit_content() {
642 if let Some(min_content_contribution) = growth_limit_min_content_contribution {
645 track.growth_limit_planned_increase =
646 f32_max(track.growth_limit_planned_increase, min_content_contribution);
647 }
648
649 let fit_content_limit = track.fit_content_limit(axis_inner_node_size);
652 let max_content_contribution = f32_min(growth_limit_max_content_contribution, fit_content_limit);
653 track.growth_limit_planned_increase =
654 f32_max(track.growth_limit_planned_increase, max_content_contribution);
655 } else if track.max_track_sizing_function.is_max_content_alike()
656 || track.max_track_sizing_function.uses_percentage() && axis_inner_node_size.is_none()
657 {
658 track.growth_limit_planned_increase =
661 f32_max(track.growth_limit_planned_increase, growth_limit_max_content_contribution);
662 } else if track.max_track_sizing_function.is_intrinsic() {
663 track.growth_limit_planned_increase =
664 f32_max(track.growth_limit_planned_increase, growth_limit_intrinsic_min_content_contribution);
665 }
666 }
667
668 for track in axis_tracks.iter_mut() {
669 if track.growth_limit_planned_increase > 0.0 {
670 track.growth_limit = if track.growth_limit == f32::INFINITY {
671 track.growth_limit_planned_increase
672 } else {
673 f32_max(track.growth_limit, track.growth_limit_planned_increase)
674 };
675 }
676 track.infinitely_growable = false;
677 track.growth_limit_planned_increase = 0.0;
678 if track.growth_limit < track.base_size {
679 track.growth_limit = track.base_size;
680 }
681 }
682
683 continue;
684 }
685
686 let use_flex_factor_for_distribution = is_flex && flex_factor_sum != 0.0;
687
688 for item in batch.iter_mut().filter(|item| item.crosses_intrinsic_track(axis)) {
691 let space = match axis_available_grid_space {
701 AvailableSpace::MinContent | AvailableSpace::MaxContent
702 if !item.overflow.get(axis).is_scroll_container() =>
703 {
704 let axis_minimum_size = item_sizer.minimum_contribution(item, axis_tracks);
705 let axis_min_content_size = item_sizer.min_content_contribution(item, axis_tracks);
706 let limit = item.spanned_track_limit(axis, axis_tracks, axis_inner_node_size, &|val, basis| {
707 item_sizer.calc(val, basis)
708 });
709 axis_min_content_size.maybe_min(limit).max(axis_minimum_size)
710 }
711 _ => item_sizer.minimum_contribution(item, axis_tracks),
712 };
713 let tracks = &mut axis_tracks[item.track_range_excluding_lines(axis)];
714 if space > 0.0 {
715 let has_intrinsic_min_track_sizing_function = |track: &GridTrack| {
716 track
717 .min_track_sizing_function
718 .definite_value(axis_inner_node_size, |val, basis| item_sizer.calc(val, basis))
719 .is_none()
720 };
721 if item.overflow.get(axis).is_scroll_container() {
722 let fit_content_limit =
723 move |track: &GridTrack| track.fit_content_limited_growth_limit(axis_inner_node_size);
724 distribute_item_space_to_base_size(
725 is_flex,
726 use_flex_factor_for_distribution,
727 space,
728 tracks,
729 has_intrinsic_min_track_sizing_function,
730 fit_content_limit,
731 IntrinsicContributionType::Minimum,
732 );
733 } else {
734 distribute_item_space_to_base_size(
735 is_flex,
736 use_flex_factor_for_distribution,
737 space,
738 tracks,
739 has_intrinsic_min_track_sizing_function,
740 |track| track.growth_limit,
741 IntrinsicContributionType::Minimum,
742 );
743 }
744 }
745 }
746 flush_planned_base_size_increases(axis_tracks);
747
748 let has_min_or_max_content_min_track_sizing_function =
752 move |track: &GridTrack| track.min_track_sizing_function.is_min_or_max_content();
753 for item in batch.iter_mut() {
754 let space = item_sizer.min_content_contribution(item, axis_tracks);
755 let tracks = &mut axis_tracks[item.track_range_excluding_lines(axis)];
756 if space > 0.0 {
757 if item.overflow.get(axis).is_scroll_container() {
758 let fit_content_limit =
759 move |track: &GridTrack| track.fit_content_limited_growth_limit(axis_inner_node_size);
760 distribute_item_space_to_base_size(
761 is_flex,
762 use_flex_factor_for_distribution,
763 space,
764 tracks,
765 has_min_or_max_content_min_track_sizing_function,
766 fit_content_limit,
767 IntrinsicContributionType::Minimum,
768 );
769 } else {
770 distribute_item_space_to_base_size(
771 is_flex,
772 use_flex_factor_for_distribution,
773 space,
774 tracks,
775 has_min_or_max_content_min_track_sizing_function,
776 |track| track.growth_limit,
777 IntrinsicContributionType::Minimum,
778 );
779 }
780 }
781 }
782 flush_planned_base_size_increases(axis_tracks);
783
784 if axis_available_grid_space == AvailableSpace::MaxContent {
794 #[inline(always)]
803 fn has_auto_min_track_sizing_function(track: &GridTrack) -> bool {
804 track.min_track_sizing_function.is_auto() && !track.max_track_sizing_function.is_min_content()
805 }
806
807 #[inline(always)]
809 fn has_max_content_min_track_sizing_function(track: &GridTrack) -> bool {
810 track.min_track_sizing_function.is_max_content()
811 }
812
813 for item in batch.iter_mut() {
814 let axis_max_content_size = item_sizer.max_content_contribution(item, axis_tracks);
815 let limit = item.spanned_track_limit(axis, axis_tracks, axis_inner_node_size, &|val, basis| {
816 item_sizer.calc(val, basis)
817 });
818 let space = axis_max_content_size.maybe_min(limit);
819 let tracks = &mut axis_tracks[item.track_range_excluding_lines(axis)];
820 if space > 0.0 {
821 if tracks.iter().any(has_max_content_min_track_sizing_function) {
832 distribute_item_space_to_base_size(
833 is_flex,
834 use_flex_factor_for_distribution,
835 space,
836 tracks,
837 has_max_content_min_track_sizing_function,
838 |_| f32::INFINITY,
839 IntrinsicContributionType::Maximum,
840 );
841 } else {
842 let fit_content_limited_growth_limit =
843 move |track: &GridTrack| track.fit_content_limited_growth_limit(axis_inner_node_size);
844 distribute_item_space_to_base_size(
845 is_flex,
846 use_flex_factor_for_distribution,
847 space,
848 tracks,
849 has_auto_min_track_sizing_function,
850 fit_content_limited_growth_limit,
851 IntrinsicContributionType::Maximum,
852 );
853 }
854 }
855 }
856 flush_planned_base_size_increases(axis_tracks);
857 }
858
859 let has_max_content_min_track_sizing_function =
862 move |track: &GridTrack| track.min_track_sizing_function.is_max_content();
863 for item in batch.iter_mut() {
864 let axis_max_content_size = item_sizer.max_content_contribution(item, axis_tracks);
865 let space = axis_max_content_size;
866 let tracks = &mut axis_tracks[item.track_range_excluding_lines(axis)];
867 if space > 0.0 {
868 distribute_item_space_to_base_size(
869 is_flex,
870 use_flex_factor_for_distribution,
871 space,
872 tracks,
873 has_max_content_min_track_sizing_function,
874 |track| track.growth_limit,
875 IntrinsicContributionType::Maximum,
876 );
877 }
878 }
879 flush_planned_base_size_increases(axis_tracks);
880
881 for track in axis_tracks.iter_mut() {
883 if track.growth_limit < track.base_size {
884 track.growth_limit = track.base_size;
885 }
886 }
887
888 if !is_flex {
891 let has_intrinsic_max_track_sizing_function =
894 move |track: &GridTrack| !track.max_track_sizing_function.has_definite_value(axis_inner_node_size);
895 for item in batch.iter_mut() {
896 let axis_min_content_size = item_sizer.min_content_contribution(item, axis_tracks);
897 let space = axis_min_content_size;
898 let tracks = &mut axis_tracks[item.track_range_excluding_lines(axis)];
899 if space > 0.0 {
900 distribute_item_space_to_growth_limit(
901 space,
902 tracks,
903 has_intrinsic_max_track_sizing_function,
904 inner_node_size.get(axis),
905 );
906 }
907 }
908 flush_planned_growth_limit_increases(axis_tracks, true);
910
911 let has_max_content_max_track_sizing_function = |track: &GridTrack| {
915 track.max_track_sizing_function.is_max_content_alike()
916 || (track.max_track_sizing_function.uses_percentage() && axis_inner_node_size.is_none())
917 };
918 for item in batch.iter_mut() {
919 let axis_max_content_size = item_sizer.max_content_contribution(item, axis_tracks);
920 let space = axis_max_content_size;
921 let tracks = &mut axis_tracks[item.track_range_excluding_lines(axis)];
922 if space > 0.0 {
923 distribute_item_space_to_growth_limit(
924 space,
925 tracks,
926 has_max_content_max_track_sizing_function,
927 inner_node_size.get(axis),
928 );
929 }
930 }
931 flush_planned_growth_limit_increases(axis_tracks, false);
933 }
934 }
935
936 axis_tracks
940 .iter_mut()
941 .filter(|track| track.growth_limit == f32::INFINITY)
942 .for_each(|track| track.growth_limit = track.base_size);
943}
944
945#[inline(always)]
948fn distribute_item_space_to_base_size(
949 is_flex: bool,
950 use_flex_factor_for_distribution: bool,
951 space: f32,
952 tracks: &mut [GridTrack],
953 track_is_affected: impl Fn(&GridTrack) -> bool,
954 track_limit: impl Fn(&GridTrack) -> f32,
955 intrinsic_contribution_type: IntrinsicContributionType,
956) {
957 if is_flex {
958 let filter = |track: &GridTrack| track.is_flexible() && track_is_affected(track);
959 if use_flex_factor_for_distribution {
960 distribute_item_space_to_base_size_inner(
961 space,
962 tracks,
963 filter,
964 |track| track.flex_factor(),
965 track_limit,
966 intrinsic_contribution_type,
967 )
968 } else {
969 distribute_item_space_to_base_size_inner(
970 space,
971 tracks,
972 filter,
973 |_| 1.0,
974 track_limit,
975 intrinsic_contribution_type,
976 )
977 }
978 } else {
979 distribute_item_space_to_base_size_inner(
980 space,
981 tracks,
982 track_is_affected,
983 |_| 1.0,
984 track_limit,
985 intrinsic_contribution_type,
986 )
987 }
988
989 fn distribute_item_space_to_base_size_inner(
992 space: f32,
993 tracks: &mut [GridTrack],
994 track_is_affected: impl Fn(&GridTrack) -> bool,
995 track_distribution_proportion: impl Fn(&GridTrack) -> f32,
996 track_limit: impl Fn(&GridTrack) -> f32,
997 intrinsic_contribution_type: IntrinsicContributionType,
998 ) {
999 if space == 0.0 || !tracks.iter().any(&track_is_affected) {
1003 return;
1004 }
1005
1006 let get_base_size = |track: &GridTrack| track.base_size;
1009
1010 let track_sizes: f32 = tracks.iter().map(|track| track.base_size).sum();
1012 let extra_space: f32 = f32_max(0.0, space - track_sizes);
1013
1014 const THRESHOLD: f32 = 0.000001;
1022
1023 let extra_space = distribute_space_up_to_limits(
1024 extra_space,
1025 tracks,
1026 &track_is_affected,
1027 &track_distribution_proportion,
1028 get_base_size,
1029 &track_limit,
1030 );
1031
1032 if extra_space > THRESHOLD {
1034 let mut filter = match intrinsic_contribution_type {
1039 IntrinsicContributionType::Minimum => {
1040 (|track: &GridTrack| track.max_track_sizing_function.is_intrinsic()) as fn(&GridTrack) -> bool
1041 }
1042 IntrinsicContributionType::Maximum => {
1043 (|track: &GridTrack| {
1044 track.min_track_sizing_function.is_max_content()
1045 || track.max_track_sizing_function.is_max_or_fit_content()
1046 }) as fn(&GridTrack) -> bool
1047 }
1048 };
1049
1050 let number_of_tracks =
1052 tracks.iter().filter(|track| track_is_affected(track)).filter(|track| filter(track)).count();
1053 if number_of_tracks == 0 {
1054 filter = (|_| true) as fn(&GridTrack) -> bool;
1055 }
1056
1057 distribute_space_up_to_limits(
1058 extra_space,
1059 tracks,
1060 filter,
1061 &track_distribution_proportion,
1062 get_base_size,
1063 &track_limit, );
1065 }
1066
1067 for track in tracks.iter_mut() {
1070 if track.item_incurred_increase > track.base_size_planned_increase {
1071 track.base_size_planned_increase = track.item_incurred_increase;
1072 }
1073
1074 track.item_incurred_increase = 0.0;
1076 }
1077 }
1078}
1079
1080fn distribute_item_space_to_growth_limit(
1084 space: f32,
1085 tracks: &mut [GridTrack],
1086 track_is_affected: impl Fn(&GridTrack) -> bool,
1087 axis_inner_node_size: Option<f32>,
1088) {
1089 if space == 0.0 || tracks.iter().filter(|track| track_is_affected(track)).count() == 0 {
1093 return;
1094 }
1095
1096 let track_sizes: f32 = tracks
1098 .iter()
1099 .map(|track| if track.growth_limit == f32::INFINITY { track.base_size } else { track.growth_limit })
1100 .sum();
1101 let extra_space: f32 = f32_max(0.0, space - track_sizes);
1102
1103 let number_of_growable_tracks = tracks
1108 .iter()
1109 .filter(|track| track_is_affected(track))
1110 .filter(|track| {
1111 track.infinitely_growable || track.fit_content_limited_growth_limit(axis_inner_node_size) == f32::INFINITY
1112 })
1113 .count();
1114 if number_of_growable_tracks > 0 {
1115 let item_incurred_increase = extra_space / number_of_growable_tracks as f32;
1116 for track in tracks.iter_mut().filter(|track| track_is_affected(track)).filter(|track| {
1117 track.infinitely_growable || track.fit_content_limited_growth_limit(axis_inner_node_size) == f32::INFINITY
1118 }) {
1119 track.item_incurred_increase = item_incurred_increase;
1120 }
1121 } else {
1122 distribute_space_up_to_limits(
1126 extra_space,
1127 tracks,
1128 track_is_affected,
1129 |_| 1.0,
1130 |track| if track.growth_limit == f32::INFINITY { track.base_size } else { track.growth_limit },
1131 move |track| track.fit_content_limit(axis_inner_node_size),
1132 );
1133 };
1134
1135 for track in tracks.iter_mut() {
1138 if track.item_incurred_increase > track.growth_limit_planned_increase {
1139 track.growth_limit_planned_increase = track.item_incurred_increase;
1140 }
1141
1142 track.item_incurred_increase = 0.0;
1144 }
1145}
1146
1147#[inline(always)]
1150fn maximise_tracks(
1151 axis_tracks: &mut [GridTrack],
1152 axis_inner_node_size: Option<f32>,
1153 axis_available_grid_space: AvailableSpace,
1154) {
1155 let used_space: f32 = axis_tracks.iter().map(|track| track.base_size).sum();
1156 let free_space = axis_available_grid_space.compute_free_space(used_space);
1157 if free_space == f32::INFINITY {
1158 axis_tracks.iter_mut().for_each(|track| track.base_size = track.growth_limit);
1159 } else if free_space > 0.0 {
1160 distribute_space_up_to_limits(
1161 free_space,
1162 axis_tracks,
1163 |_| true,
1164 |_| 1.0,
1165 |track| track.base_size,
1166 move |track: &GridTrack| track.fit_content_limited_growth_limit(axis_inner_node_size),
1167 );
1168 for track in axis_tracks.iter_mut() {
1169 track.base_size += track.item_incurred_increase;
1170 track.item_incurred_increase = 0.0;
1171 }
1172 }
1173}
1174
1175#[allow(clippy::too_many_arguments)]
1178#[inline(always)]
1179fn expand_flexible_tracks(
1180 tree: &mut impl LayoutPartialTree,
1181 axis: AbstractAxis,
1182 axis_tracks: &mut [GridTrack],
1183 items: &mut [GridItem],
1184 axis_min_size: Option<f32>,
1185 axis_max_size: Option<f32>,
1186 axis_available_space_for_expansion: AvailableSpace,
1187) {
1188 let flex_fraction = match axis_available_space_for_expansion {
1190 AvailableSpace::Definite(available_space) => {
1196 let used_space: f32 = axis_tracks.iter().map(|track| track.base_size).sum();
1197 let free_space = available_space - used_space;
1198 if free_space <= 0.0 {
1199 0.0
1200 } else {
1201 find_size_of_fr(axis_tracks, available_space)
1202 }
1203 }
1204 AvailableSpace::MinContent => 0.0,
1206 AvailableSpace::MaxContent => {
1208 let flex_fraction = f32_max(
1210 axis_tracks
1213 .iter()
1214 .filter(|track| track.max_track_sizing_function.is_fr())
1215 .map(|track| {
1216 let flex_factor = track.flex_factor();
1217 if flex_factor > 1.0 {
1218 track.base_size / flex_factor
1219 } else {
1220 track.base_size
1221 }
1222 })
1223 .max_by(|a, b| a.total_cmp(b))
1224 .unwrap_or(0.0),
1225 items
1228 .iter_mut()
1229 .filter(|item| item.crosses_flexible_track(axis))
1230 .map(|item| {
1231 let tracks = &axis_tracks[item.track_range_excluding_lines(axis)];
1232 let max_content_contribution =
1234 item.max_content_contribution_cached(axis, tree, Size::NONE, Size::NONE);
1235 find_size_of_fr(tracks, max_content_contribution)
1236 })
1237 .max_by(|a, b| a.total_cmp(b))
1238 .unwrap_or(0.0),
1239 );
1240
1241 let hypothetical_grid_size: f32 = axis_tracks
1246 .iter()
1247 .map(|track| {
1248 if track.max_track_sizing_function.is_fr() {
1249 let track_flex_factor = track.max_track_sizing_function.0.value();
1250 f32_max(track.base_size, track_flex_factor * flex_fraction)
1251 } else {
1252 track.base_size
1253 }
1254 })
1255 .sum();
1256 let axis_min_size = axis_min_size.unwrap_or(0.0);
1257 let axis_max_size = axis_max_size.unwrap_or(f32::INFINITY);
1258 if hypothetical_grid_size < axis_min_size {
1259 find_size_of_fr(axis_tracks, axis_min_size)
1260 } else if hypothetical_grid_size > axis_max_size {
1261 find_size_of_fr(axis_tracks, axis_max_size)
1262 } else {
1263 flex_fraction
1264 }
1265 }
1266 };
1267
1268 for track in axis_tracks.iter_mut().filter(|track| track.max_track_sizing_function.is_fr()) {
1271 let track_flex_factor = track.max_track_sizing_function.0.value();
1272 track.base_size = f32_max(track.base_size, track_flex_factor * flex_fraction);
1273 }
1274}
1275
1276#[inline(always)]
1280fn find_size_of_fr(tracks: &[GridTrack], space_to_fill: f32) -> f32 {
1281 if space_to_fill == 0.0 {
1284 return 0.0;
1285 }
1286
1287 let mut hypothetical_fr_size = f32::INFINITY;
1292 let mut previous_iter_hypothetical_fr_size;
1293 loop {
1294 let mut used_space = 0.0;
1298 let mut naive_flex_factor_sum = 0.0;
1299 for track in tracks.iter() {
1300 if track.max_track_sizing_function.is_fr()
1302 && track.max_track_sizing_function.0.value() * hypothetical_fr_size >= track.base_size
1303 {
1304 naive_flex_factor_sum += track.max_track_sizing_function.0.value();
1305 } else {
1306 used_space += track.base_size;
1307 };
1308 }
1309 let leftover_space = space_to_fill - used_space;
1310 let flex_factor = f32_max(naive_flex_factor_sum, 1.0);
1311
1312 previous_iter_hypothetical_fr_size = hypothetical_fr_size;
1314 hypothetical_fr_size = leftover_space / flex_factor;
1315
1316 let hypothetical_fr_size_is_valid = tracks.iter().all(|track| {
1320 if track.max_track_sizing_function.is_fr() {
1321 let flex_factor = track.max_track_sizing_function.0.value();
1322 flex_factor * hypothetical_fr_size >= track.base_size
1323 || flex_factor * previous_iter_hypothetical_fr_size < track.base_size
1324 } else {
1325 true
1326 }
1327 });
1328 if hypothetical_fr_size_is_valid {
1329 break;
1330 }
1331 }
1332
1333 hypothetical_fr_size
1335}
1336
1337#[inline(always)]
1340fn stretch_auto_tracks(
1341 axis_tracks: &mut [GridTrack],
1342 axis_min_size: Option<f32>,
1343 axis_available_space_for_expansion: AvailableSpace,
1344) {
1345 let num_auto_tracks = axis_tracks.iter().filter(|track| track.max_track_sizing_function.is_auto()).count();
1346 if num_auto_tracks > 0 {
1347 let used_space: f32 = axis_tracks.iter().map(|track| track.base_size).sum();
1348
1349 let free_space = if axis_available_space_for_expansion.is_definite() {
1352 axis_available_space_for_expansion.compute_free_space(used_space)
1353 } else {
1354 match axis_min_size {
1355 Some(size) => size - used_space,
1356 None => 0.0,
1357 }
1358 };
1359 if free_space > 0.0 {
1360 let extra_space_per_auto_track = free_space / num_auto_tracks as f32;
1361 axis_tracks
1362 .iter_mut()
1363 .filter(|track| track.max_track_sizing_function.is_auto())
1364 .for_each(|track| track.base_size += extra_space_per_auto_track);
1365 }
1366 }
1367}
1368
1369#[inline(always)]
1372fn distribute_space_up_to_limits(
1373 space_to_distribute: f32,
1374 tracks: &mut [GridTrack],
1375 track_is_affected: impl Fn(&GridTrack) -> bool,
1376 track_distribution_proportion: impl Fn(&GridTrack) -> f32,
1377 track_affected_property: impl Fn(&GridTrack) -> f32,
1378 track_limit: impl Fn(&GridTrack) -> f32,
1379) -> f32 {
1380 const THRESHOLD: f32 = 0.01;
1383
1384 let mut space_to_distribute = space_to_distribute;
1385 while space_to_distribute > THRESHOLD {
1386 let track_distribution_proportion_sum: f32 = tracks
1387 .iter()
1388 .filter(|track| track_affected_property(track) + track.item_incurred_increase < track_limit(track))
1389 .filter(|track| track_is_affected(track))
1390 .map(&track_distribution_proportion)
1391 .sum();
1392
1393 if track_distribution_proportion_sum == 0.0 {
1394 break;
1395 }
1396
1397 let min_increase_limit = tracks
1399 .iter()
1400 .filter(|track| track_affected_property(track) + track.item_incurred_increase < track_limit(track))
1401 .filter(|track| track_is_affected(track))
1402 .map(|track| (track_limit(track) - track_affected_property(track)) / track_distribution_proportion(track))
1403 .min_by(|a, b| a.total_cmp(b))
1404 .unwrap(); let iteration_item_incurred_increase =
1406 f32_min(min_increase_limit, space_to_distribute / track_distribution_proportion_sum);
1407
1408 for track in tracks.iter_mut().filter(|track| track_is_affected(track)) {
1409 let increase = iteration_item_incurred_increase * track_distribution_proportion(track);
1410 if increase > 0.0 && track_affected_property(track) + increase <= track_limit(track) + THRESHOLD {
1411 track.item_incurred_increase += increase;
1412 space_to_distribute -= increase;
1413 }
1414 }
1415 }
1416
1417 space_to_distribute
1418}