1use crate::geometry::{AbsoluteAxis, AbstractAxis, InBothAbsAxis};
4use crate::geometry::{Line, Point, Rect, Size};
5use crate::style::{AlignItems, AlignSelf, AvailableSpace, Overflow, Position};
6use crate::tree::{Layout, LayoutInput, LayoutOutput, LayoutPartialTreeExt, NodeId, RunMode, SizingMode};
7use crate::util::debug::debug_log;
8use crate::util::sys::{f32_max, f32_min, GridTrackVec, Vec};
9use crate::util::MaybeMath;
10use crate::util::{MaybeResolve, ResolveOrZero};
11use crate::{
12 style_helpers::*, AlignContent, BoxGenerationMode, BoxSizing, CoreStyle, Direction, GridContainerStyle,
13 GridItemStyle, JustifyContent, LayoutGridContainer,
14};
15use alignment::{align_and_position_item, align_tracks};
16use explicit_grid::{compute_explicit_grid_size_in_axis, initialize_grid_tracks, AutoRepeatStrategy};
17use implicit_grid::compute_grid_size_estimate;
18use placement::place_grid_items;
19use track_sizing::{
20 determine_if_item_crosses_flexible_or_intrinsic_tracks, resolve_item_track_indexes, track_sizing_algorithm,
21};
22use types::{CellOccupancyMatrix, GridTrack, NamedLineResolver, TrackCounts};
23
24#[cfg(feature = "detailed_layout_info")]
25use types::{GridItem, GridTrackKind};
26
27pub(crate) use types::{GridCoordinate, GridLine, OriginZeroLine};
28
29mod alignment;
30mod explicit_grid;
31mod implicit_grid;
32mod placement;
33mod track_sizing;
34mod types;
35mod util;
36
37pub fn compute_grid_layout<Tree: LayoutGridContainer>(
44 tree: &mut Tree,
45 node: NodeId,
46 inputs: LayoutInput,
47) -> LayoutOutput {
48 let LayoutInput { known_dimensions, parent_size, available_space, run_mode, .. } = inputs;
49
50 let style = tree.get_grid_container_style(node);
51 let direction = style.direction();
52
53 let aspect_ratio = style.aspect_ratio();
56 let padding = style.padding().resolve_or_zero(parent_size.width, |val, basis| tree.calc(val, basis));
57 let border = style.border().resolve_or_zero(parent_size.width, |val, basis| tree.calc(val, basis));
58 let padding_border = padding + border;
59 let padding_border_size = padding_border.sum_axes();
60 let box_sizing_adjustment =
61 if style.box_sizing() == BoxSizing::ContentBox { padding_border_size } else { Size::ZERO };
62
63 let min_size = style
64 .min_size()
65 .maybe_resolve(parent_size, |val, basis| tree.calc(val, basis))
66 .maybe_apply_aspect_ratio(aspect_ratio)
67 .maybe_add(box_sizing_adjustment);
68 let max_size = style
69 .max_size()
70 .maybe_resolve(parent_size, |val, basis| tree.calc(val, basis))
71 .maybe_apply_aspect_ratio(aspect_ratio)
72 .maybe_add(box_sizing_adjustment);
73 let preferred_size = if inputs.sizing_mode == SizingMode::InherentSize {
74 style
75 .size()
76 .maybe_resolve(parent_size, |val, basis| tree.calc(val, basis))
77 .maybe_apply_aspect_ratio(style.aspect_ratio())
78 .maybe_add(box_sizing_adjustment)
79 } else {
80 Size::NONE
81 };
82
83 let scrollbar_gutter = style.overflow().transpose().map(|overflow| match overflow {
87 Overflow::Scroll => style.scrollbar_width(),
88 _ => 0.0,
89 });
90 let mut content_box_inset = padding_border;
91 content_box_inset.bottom += scrollbar_gutter.y;
92
93 match direction {
94 Direction::Ltr => content_box_inset.right += scrollbar_gutter.x,
95 Direction::Rtl => content_box_inset.left += scrollbar_gutter.x,
96 };
97
98 let align_content = style.align_content().unwrap_or(AlignContent::STRETCH);
99 let justify_content = style.justify_content().unwrap_or(JustifyContent::STRETCH);
100 let align_items = style.align_items();
101 let justify_items = style.justify_items();
102
103 let grid_template_columns = style.grid_template_columns();
106 let grid_template_rows = style.grid_template_rows();
107 let grid_auto_columns = style.grid_auto_columns();
108 let grid_auto_rows = style.grid_auto_rows();
109
110 let constrained_available_space = known_dimensions
111 .or(preferred_size)
112 .map(|size| size.map(AvailableSpace::Definite))
113 .unwrap_or(available_space)
114 .maybe_clamp(min_size, max_size)
115 .maybe_max(padding_border_size);
116
117 let available_grid_space = Size {
118 width: constrained_available_space
119 .width
120 .map_definite_value(|space| space - content_box_inset.horizontal_axis_sum()),
121 height: constrained_available_space
122 .height
123 .map_definite_value(|space| space - content_box_inset.vertical_axis_sum()),
124 };
125
126 let outer_node_size =
127 known_dimensions.or(preferred_size).maybe_clamp(min_size, max_size).maybe_max(padding_border_size);
128 let mut inner_node_size = Size {
129 width: outer_node_size.width.map(|space| space - content_box_inset.horizontal_axis_sum()),
130 height: outer_node_size.height.map(|space| space - content_box_inset.vertical_axis_sum()),
131 };
132
133 debug_log!("parent_size", dbg:parent_size);
134 debug_log!("outer_node_size", dbg:outer_node_size);
135 debug_log!("inner_node_size", dbg:inner_node_size);
136
137 if let (RunMode::ComputeSize, Some(width), Some(height)) = (run_mode, outer_node_size.width, outer_node_size.height)
138 {
139 return LayoutOutput::from_outer_size(Size { width, height });
140 }
141
142 let get_child_styles_iter =
143 |node| tree.child_ids(node).map(|child_node: NodeId| tree.get_grid_child_style(child_node));
144 let child_styles_iter = get_child_styles_iter(node);
145
146 let auto_fit_container_size = outer_node_size
151 .or(max_size)
152 .or(min_size)
153 .maybe_clamp(min_size, max_size)
154 .maybe_max(padding_border_size)
155 .maybe_sub(content_box_inset.sum_axes());
156
157 let auto_repeat_fit_strategy = outer_node_size.or(max_size).map(|val| match val {
164 Some(_) => AutoRepeatStrategy::MaxRepetitionsThatDoNotOverflow,
165 None => AutoRepeatStrategy::MinRepetitionsThatDoOverflow,
166 });
167
168 let (col_auto_repetition_count, grid_template_col_count) = compute_explicit_grid_size_in_axis(
171 &style,
172 auto_fit_container_size.width,
173 auto_repeat_fit_strategy.width,
174 |val, basis| tree.calc(val, basis),
175 AbsoluteAxis::Horizontal,
176 );
177 let (row_auto_repetition_count, grid_template_row_count) = compute_explicit_grid_size_in_axis(
178 &style,
179 auto_fit_container_size.height,
180 auto_repeat_fit_strategy.height,
181 |val, basis| tree.calc(val, basis),
182 AbsoluteAxis::Vertical,
183 );
184
185 let mut name_resolver = NamedLineResolver::new(&style, col_auto_repetition_count, row_auto_repetition_count);
187
188 let explicit_col_count = grid_template_col_count.max(name_resolver.area_column_count());
189 let explicit_row_count = grid_template_row_count.max(name_resolver.area_row_count());
190
191 name_resolver.set_explicit_column_count(explicit_col_count);
192 name_resolver.set_explicit_row_count(explicit_row_count);
193
194 let (est_col_counts, est_row_counts) =
198 compute_grid_size_estimate(explicit_col_count, explicit_row_count, direction, child_styles_iter);
199
200 let mut items = Vec::with_capacity(tree.child_count(node));
203 let mut cell_occupancy_matrix = CellOccupancyMatrix::with_track_counts(est_col_counts, est_row_counts);
204 let in_flow_children_iter = || {
205 tree.child_ids(node)
206 .enumerate()
207 .map(|(index, child_node)| (index, child_node, tree.get_grid_child_style(child_node)))
208 .filter(|(_, _, style)| {
209 style.box_generation_mode() != BoxGenerationMode::None && style.position() != Position::Absolute
210 })
211 };
212 place_grid_items(
213 &mut cell_occupancy_matrix,
214 &mut items,
215 in_flow_children_iter,
216 direction,
217 style.grid_auto_flow(),
218 align_items.unwrap_or(AlignItems::STRETCH),
219 justify_items.unwrap_or(AlignItems::STRETCH),
220 &name_resolver,
221 );
222
223 let final_col_counts = *cell_occupancy_matrix.track_counts(AbsoluteAxis::Horizontal);
225 let final_row_counts = *cell_occupancy_matrix.track_counts(AbsoluteAxis::Vertical);
226
227 let mut columns = GridTrackVec::new();
231 let mut rows = GridTrackVec::new();
232 let mut column_track_counts_for_init = final_col_counts;
233 if direction.is_rtl() && final_col_counts.explicit <= 1 {
234 column_track_counts_for_init.negative_implicit = final_col_counts.positive_implicit;
235 column_track_counts_for_init.positive_implicit = final_col_counts.negative_implicit;
236 }
237 initialize_grid_tracks(
238 &mut columns,
239 column_track_counts_for_init,
240 &style,
241 AbsoluteAxis::Horizontal,
242 |column_index| {
243 let occupancy_index = if direction.is_rtl() {
244 rtl_column_occupancy_index_for_initialization(column_index, final_col_counts)
245 } else {
246 column_index
247 };
248 cell_occupancy_matrix.column_is_occupied(occupancy_index)
249 },
250 );
251 initialize_grid_tracks(&mut rows, final_row_counts, &style, AbsoluteAxis::Vertical, |row_index| {
252 cell_occupancy_matrix.row_is_occupied(row_index)
253 });
254 if direction.is_rtl() {
255 reverse_non_gutter_tracks(&mut columns, final_col_counts);
256 }
257
258 drop(grid_template_rows);
259 drop(grid_template_columns);
260 drop(grid_auto_rows);
261 drop(grid_auto_columns);
262 drop(style);
263
264 resolve_item_track_indexes(&mut items, final_col_counts, final_row_counts);
270 determine_if_item_crosses_flexible_or_intrinsic_tracks(&mut items, &columns, &rows);
273
274 let has_baseline_aligned_item = items.iter().any(|item| item.align_self == AlignSelf::BASELINE);
276
277 track_sizing_algorithm(
279 tree,
280 AbstractAxis::Inline,
281 min_size.get(AbstractAxis::Inline),
282 max_size.get(AbstractAxis::Inline),
283 justify_content,
284 align_content,
285 available_grid_space,
286 inner_node_size,
287 &mut columns,
288 &mut rows,
289 &mut items,
290 |track: &GridTrack, parent_size: Option<f32>, tree: &Tree| {
291 track.max_track_sizing_function.definite_value(parent_size, |val, basis| tree.calc(val, basis))
292 },
293 has_baseline_aligned_item,
294 );
295 let initial_column_sum = columns.iter().map(|track| track.base_size).sum::<f32>();
296 inner_node_size.width = inner_node_size.width.or_else(|| initial_column_sum.into());
297
298 items.iter_mut().for_each(|item| item.grid_area_size_cache = None);
299
300 track_sizing_algorithm(
302 tree,
303 AbstractAxis::Block,
304 min_size.get(AbstractAxis::Block),
305 max_size.get(AbstractAxis::Block),
306 align_content,
307 justify_content,
308 available_grid_space,
309 inner_node_size,
310 &mut rows,
311 &mut columns,
312 &mut items,
313 |track: &GridTrack, _, _| Some(track.base_size),
314 false, );
316 let initial_row_sum = rows.iter().map(|track| track.base_size).sum::<f32>();
317 inner_node_size.height = inner_node_size.height.or_else(|| initial_row_sum.into());
318
319 debug_log!("initial_column_sum", dbg:initial_column_sum);
320 debug_log!(dbg: columns.iter().map(|track| track.base_size).collect::<Vec<_>>());
321 debug_log!("initial_row_sum", dbg:initial_row_sum);
322 debug_log!(dbg: rows.iter().map(|track| track.base_size).collect::<Vec<_>>());
323
324 let resolved_style_size = known_dimensions.or(preferred_size);
326 let mut container_border_box = Size {
327 width: resolved_style_size
328 .get(AbstractAxis::Inline)
329 .unwrap_or_else(|| initial_column_sum + content_box_inset.horizontal_axis_sum())
330 .maybe_clamp(min_size.width, max_size.width)
331 .max(padding_border_size.width),
332 height: resolved_style_size
333 .get(AbstractAxis::Block)
334 .unwrap_or_else(|| initial_row_sum + content_box_inset.vertical_axis_sum())
335 .maybe_clamp(min_size.height, max_size.height)
336 .max(padding_border_size.height),
337 };
338 let mut container_content_box = Size {
339 width: f32_max(0.0, container_border_box.width - content_box_inset.horizontal_axis_sum()),
340 height: f32_max(0.0, container_border_box.height - content_box_inset.vertical_axis_sum()),
341 };
342
343 if run_mode == RunMode::ComputeSize {
345 return LayoutOutput::from_outer_size(container_border_box);
346 }
347
348 if !available_grid_space.width.is_definite() {
352 for column in &mut columns {
353 let min: Option<f32> = column
354 .min_track_sizing_function
355 .resolved_percentage_size(container_content_box.width, |val, basis| tree.calc(val, basis));
356 let max: Option<f32> = column
357 .max_track_sizing_function
358 .resolved_percentage_size(container_content_box.width, |val, basis| tree.calc(val, basis));
359 column.base_size = column.base_size.maybe_clamp(min, max);
360 }
361 }
362 if !available_grid_space.height.is_definite() {
363 for row in &mut rows {
364 let min: Option<f32> = row
365 .min_track_sizing_function
366 .resolved_percentage_size(container_content_box.height, |val, basis| tree.calc(val, basis));
367 let max: Option<f32> = row
368 .max_track_sizing_function
369 .resolved_percentage_size(container_content_box.height, |val, basis| tree.calc(val, basis));
370 row.base_size = row.base_size.maybe_clamp(min, max);
371 }
372 }
373
374 let mut rerun_column_sizing;
379 let mut intrinsic_column_contribution_changed = false;
380
381 let has_percentage_column = columns.iter().any(|track| track.uses_percentage());
382 let has_percentage_row = rows.iter().any(|track| track.uses_percentage());
383 let parent_width_indefinite = !available_space.width.is_definite();
384 rerun_column_sizing = parent_width_indefinite && has_percentage_column;
385
386 if !rerun_column_sizing {
387 intrinsic_column_contribution_changed =
388 items.iter_mut().filter(|item| item.crosses_intrinsic_column).any(|item| {
389 let grid_area_size = item.grid_area_size(
390 AbstractAxis::Inline,
391 &columns,
392 &rows,
393 inner_node_size,
394 |track: &GridTrack, _| Some(track.base_size),
395 &|val, basis| tree.calc(val, basis),
396 );
397 let available_space = grid_area_size.with(AbstractAxis::Inline, None);
398 let new_min_content_contribution =
399 item.min_content_contribution(AbstractAxis::Inline, tree, grid_area_size, available_space);
400
401 let has_changed = Some(new_min_content_contribution) != item.min_content_contribution_cache.width;
402
403 item.grid_area_size_cache = Some(grid_area_size);
404 item.min_content_contribution_cache.width = Some(new_min_content_contribution);
405 item.max_content_contribution_cache.width = None;
406 item.minimum_contribution_cache.width = None;
407
408 has_changed
409 });
410 rerun_column_sizing = intrinsic_column_contribution_changed;
411 } else {
412 items.iter_mut().for_each(|item| {
414 item.grid_area_size_cache = None;
415 item.min_content_contribution_cache.width = None;
416 item.max_content_contribution_cache.width = None;
417 item.minimum_contribution_cache.width = None;
418 });
419 }
420
421 let mut intrinsic_row_contribution_changed = false;
422
423 if rerun_column_sizing {
424 track_sizing_algorithm(
426 tree,
427 AbstractAxis::Inline,
428 min_size.get(AbstractAxis::Inline),
429 max_size.get(AbstractAxis::Inline),
430 justify_content,
431 align_content,
432 available_grid_space,
433 inner_node_size,
434 &mut columns,
435 &mut rows,
436 &mut items,
437 |track: &GridTrack, _, _| Some(track.base_size),
438 has_baseline_aligned_item,
439 );
440
441 let mut rerun_row_sizing;
446
447 let parent_height_indefinite = !available_space.height.is_definite();
448 rerun_row_sizing = parent_height_indefinite && has_percentage_row;
449
450 if !rerun_row_sizing {
451 intrinsic_row_contribution_changed =
452 items.iter_mut().filter(|item| item.crosses_intrinsic_column).any(|item| {
453 let grid_area_size = item.grid_area_size(
454 AbstractAxis::Block,
455 &rows,
456 &columns,
457 inner_node_size,
458 |track: &GridTrack, _| Some(track.base_size),
459 &|val, basis| tree.calc(val, basis),
460 );
461 let available_space = grid_area_size.with(AbstractAxis::Block, None);
462 let new_min_content_contribution =
463 item.min_content_contribution(AbstractAxis::Block, tree, grid_area_size, available_space);
464
465 let has_changed = Some(new_min_content_contribution) != item.min_content_contribution_cache.height;
466
467 item.grid_area_size_cache = Some(grid_area_size);
468 item.min_content_contribution_cache.height = Some(new_min_content_contribution);
469 item.max_content_contribution_cache.height = None;
470 item.minimum_contribution_cache.height = None;
471
472 has_changed
473 });
474 rerun_row_sizing = intrinsic_row_contribution_changed;
475 } else {
476 items.iter_mut().for_each(|item| {
477 item.grid_area_size_cache = None;
479 item.min_content_contribution_cache.height = None;
480 item.max_content_contribution_cache.height = None;
481 item.minimum_contribution_cache.height = None;
482 });
483 }
484
485 if rerun_row_sizing {
486 track_sizing_algorithm(
488 tree,
489 AbstractAxis::Block,
490 min_size.get(AbstractAxis::Block),
491 max_size.get(AbstractAxis::Block),
492 align_content,
493 justify_content,
494 available_grid_space,
495 inner_node_size,
496 &mut rows,
497 &mut columns,
498 &mut items,
499 |track: &GridTrack, _, _| Some(track.base_size),
500 false, );
502 }
503 }
504
505 if (intrinsic_column_contribution_changed && !has_percentage_column)
506 || (intrinsic_row_contribution_changed && !has_percentage_row)
507 {
508 let final_column_sum = columns.iter().map(|track| track.base_size).sum::<f32>();
509 let final_row_sum = rows.iter().map(|track| track.base_size).sum::<f32>();
510
511 if intrinsic_column_contribution_changed && !has_percentage_column {
512 container_border_box.width = resolved_style_size
513 .get(AbstractAxis::Inline)
514 .unwrap_or_else(|| final_column_sum + content_box_inset.horizontal_axis_sum())
515 .maybe_clamp(min_size.width, max_size.width)
516 .max(padding_border_size.width);
517 container_content_box.width =
518 f32_max(0.0, container_border_box.width - content_box_inset.horizontal_axis_sum());
519 }
520
521 if intrinsic_row_contribution_changed && !has_percentage_row {
522 container_border_box.height = resolved_style_size
523 .get(AbstractAxis::Block)
524 .unwrap_or_else(|| final_row_sum + content_box_inset.vertical_axis_sum())
525 .maybe_clamp(min_size.height, max_size.height)
526 .max(padding_border_size.height);
527 container_content_box.height =
528 f32_max(0.0, container_border_box.height - content_box_inset.vertical_axis_sum());
529 }
530 }
531
532 if run_mode == RunMode::ComputeSize {
534 return LayoutOutput::from_outer_size(container_border_box);
535 }
536
537 let inline_size_without_scrollbar = f32_max(container_border_box.width - padding_border_size.width, 0.0);
541 let inline_scrollbar_gutter_for_alignment = f32_min(scrollbar_gutter.x, inline_size_without_scrollbar);
542 align_tracks(
543 container_content_box.get(AbstractAxis::Inline),
544 Line {
545 start: padding.left + if direction.is_rtl() { inline_scrollbar_gutter_for_alignment } else { 0.0 },
546 end: padding.right + if direction.is_rtl() { 0.0 } else { inline_scrollbar_gutter_for_alignment },
547 },
548 Line { start: border.left, end: border.right },
549 &mut columns,
550 justify_content,
551 direction.is_rtl(),
552 );
553 align_tracks(
555 container_content_box.get(AbstractAxis::Block),
556 Line { start: padding.top, end: padding.bottom },
557 Line { start: border.top, end: border.bottom },
558 &mut rows,
559 align_content,
560 false,
561 );
562
563 #[cfg_attr(not(feature = "content_size"), allow(unused_mut))]
566 let mut item_content_size_contribution = Size::ZERO;
567
568 items.sort_by_key(|item| item.source_order);
570
571 let container_alignment_styles = InBothAbsAxis { horizontal: justify_items, vertical: align_items };
572
573 for (index, item) in items.iter_mut().enumerate() {
575 let grid_area = Rect {
576 top: rows[item.row_indexes.start as usize + 1].offset,
577 bottom: rows[item.row_indexes.end as usize].offset,
578 left: columns[item.column_indexes.start as usize + 1].offset,
579 right: columns[item.column_indexes.end as usize].offset,
580 };
581 #[cfg_attr(not(feature = "content_size"), allow(unused_variables))]
582 let (content_size_contribution, y_position, height) = align_and_position_item(
583 tree,
584 item.node,
585 index as u32,
586 grid_area,
587 container_alignment_styles,
588 item.baseline_shim,
589 direction,
590 );
591 item.y_position = y_position;
592 item.height = height;
593
594 #[cfg(feature = "content_size")]
595 {
596 item_content_size_contribution = item_content_size_contribution.f32_max(content_size_contribution);
597 }
598 }
599
600 let mut order = items.len() as u32;
602 (0..tree.child_count(node)).for_each(|index| {
603 let child = tree.get_child_id(node, index);
604 let child_style = tree.get_grid_child_style(child);
605
606 if child_style.box_generation_mode() == BoxGenerationMode::None {
608 drop(child_style);
609 tree.set_unrounded_layout(child, &Layout::with_order(order));
610 tree.perform_child_layout(
611 child,
612 Size::NONE,
613 Size::NONE,
614 Size::MAX_CONTENT,
615 SizingMode::InherentSize,
616 Line::FALSE,
617 );
618 order += 1;
619 return;
620 }
621
622 if child_style.position() == Position::Absolute {
624 let maybe_col_indexes = name_resolver
627 .resolve_column_names(&child_style.grid_column())
628 .into_origin_zero(final_col_counts.explicit)
629 .resolve_absolutely_positioned_grid_tracks()
630 .map(|maybe_grid_line| {
631 maybe_grid_line
632 .map(|line: OriginZeroLine| {
633 if direction.is_rtl() {
634 OriginZeroLine(final_col_counts.explicit as i16 - line.0)
635 } else {
636 line
637 }
638 })
639 .and_then(|line| line.try_into_track_vec_index(final_col_counts))
640 });
641 let maybe_col_indexes = if direction.is_rtl() {
642 Line { start: maybe_col_indexes.end, end: maybe_col_indexes.start }
643 } else {
644 maybe_col_indexes
645 };
646 let maybe_row_indexes = name_resolver
649 .resolve_row_names(&child_style.grid_row())
650 .into_origin_zero(final_row_counts.explicit)
651 .resolve_absolutely_positioned_grid_tracks()
652 .map(|maybe_grid_line| {
653 maybe_grid_line.and_then(|line: OriginZeroLine| line.try_into_track_vec_index(final_row_counts))
654 });
655
656 let grid_area = Rect {
657 top: maybe_row_indexes.start.map(|index| rows[index].offset).unwrap_or(border.top),
658 bottom: maybe_row_indexes
659 .end
660 .map(|index| rows[index].offset)
661 .unwrap_or(container_border_box.height - border.bottom - scrollbar_gutter.y),
662 left: maybe_col_indexes.start.map(|index| columns[index].offset).unwrap_or_else(|| {
663 if direction.is_rtl() {
664 border.left + scrollbar_gutter.x
665 } else {
666 border.left
667 }
668 }),
669 right: maybe_col_indexes.end.map(|index| columns[index].offset).unwrap_or_else(|| {
670 if direction.is_rtl() {
671 container_border_box.width - border.right
672 } else {
673 container_border_box.width - border.right - scrollbar_gutter.x
674 }
675 }),
676 };
677 drop(child_style);
678
679 #[cfg_attr(not(feature = "content_size"), allow(unused_variables))]
681 let (content_size_contribution, _, _) =
682 align_and_position_item(tree, child, order, grid_area, container_alignment_styles, 0.0, direction);
683 #[cfg(feature = "content_size")]
684 {
685 item_content_size_contribution = item_content_size_contribution.f32_max(content_size_contribution);
686 }
687
688 order += 1;
689 }
690 });
691
692 #[cfg(feature = "detailed_layout_info")]
694 tree.set_detailed_grid_info(
695 node,
696 DetailedGridInfo {
697 rows: DetailedGridTracksInfo::from_grid_tracks_and_track_count(final_row_counts, rows),
698 columns: DetailedGridTracksInfo::from_grid_tracks_and_track_count(final_col_counts, columns),
699 items: items.iter().map(DetailedGridItemsInfo::from_grid_item).collect(),
700 },
701 );
702
703 if items.is_empty() {
705 return LayoutOutput::from_outer_size(container_border_box);
706 }
707
708 let grid_container_baseline: f32 = {
710 items.sort_by_key(|item| item.row_indexes.start);
712
713 let first_row = items[0].row_indexes.start;
715
716 let first_row_items = &items[0..].split(|item| item.row_indexes.start != first_row).next().unwrap();
718
719 let row_has_baseline_item = first_row_items.iter().any(|item| item.align_self == AlignSelf::BASELINE);
721
722 let item = if row_has_baseline_item {
723 first_row_items.iter().find(|item| item.align_self == AlignSelf::BASELINE).unwrap()
724 } else {
725 &first_row_items[0]
726 };
727
728 item.y_position + item.baseline.unwrap_or(item.height)
729 };
730
731 LayoutOutput::from_sizes_and_baselines(
732 container_border_box,
733 item_content_size_contribution,
734 Point { x: None, y: Some(grid_container_baseline) },
735 )
736}
737
738fn reverse_non_gutter_tracks(tracks: &mut [GridTrack], track_counts: TrackCounts) {
740 if track_counts.explicit <= 1 {
743 const MIN_TRACK_VEC_LEN_TO_REVERSE_COLUMNS: usize = 5;
744 if tracks.len() < MIN_TRACK_VEC_LEN_TO_REVERSE_COLUMNS {
745 return;
746 }
747 let mut left = 1;
748 let mut right = tracks.len() - 2;
749 while left < right {
750 tracks.swap(left, right);
751 left += 2;
752 right = right.saturating_sub(2);
753 }
754 return;
755 }
756
757 let explicit_track_count = track_counts.explicit as usize;
758 if explicit_track_count < 2 {
759 return;
760 }
761
762 let mut left = track_counts.negative_implicit as usize;
763 let mut right = left + explicit_track_count - 1;
764 while left < right {
765 tracks.swap((2 * left) + 1, (2 * right) + 1);
766 left += 1;
767 right = right.saturating_sub(1);
768 }
769}
770
771fn rtl_column_occupancy_index_for_initialization(column_index: usize, track_counts: TrackCounts) -> usize {
773 if track_counts.explicit <= 1 {
774 return track_counts.len() - column_index - 1;
775 }
776
777 let explicit_start = track_counts.negative_implicit as usize;
778 let explicit_end = explicit_start + track_counts.explicit as usize;
779 if (explicit_start..explicit_end).contains(&column_index) {
780 explicit_start + (explicit_end - column_index - 1)
781 } else {
782 column_index
783 }
784}
785
786#[derive(Debug, Clone, PartialEq)]
788#[cfg(feature = "detailed_layout_info")]
789pub struct DetailedGridInfo {
790 pub rows: DetailedGridTracksInfo,
792 pub columns: DetailedGridTracksInfo,
794 pub items: Vec<DetailedGridItemsInfo>,
796}
797
798#[derive(Debug, Clone, PartialEq)]
800#[cfg(feature = "detailed_layout_info")]
801pub struct DetailedGridTracksInfo {
802 pub negative_implicit_tracks: u16,
804 pub explicit_tracks: u16,
806 pub positive_implicit_tracks: u16,
808
809 pub gutters: Vec<f32>,
811 pub sizes: Vec<f32>,
813}
814
815#[cfg(feature = "detailed_layout_info")]
816impl DetailedGridTracksInfo {
817 #[inline(always)]
819 fn grid_track_base_size_of_kind(grid_tracks: &[GridTrack], kind: GridTrackKind) -> Vec<f32> {
820 grid_tracks
821 .iter()
822 .filter_map(|track| match track.kind == kind {
823 true => Some(track.base_size),
824 false => None,
825 })
826 .collect()
827 }
828
829 fn gutters_from_grid_track_layout(grid_tracks: &[GridTrack]) -> Vec<f32> {
831 DetailedGridTracksInfo::grid_track_base_size_of_kind(grid_tracks, GridTrackKind::Gutter)
832 }
833
834 fn sizes_from_grid_track_layout(grid_tracks: &[GridTrack]) -> Vec<f32> {
836 DetailedGridTracksInfo::grid_track_base_size_of_kind(grid_tracks, GridTrackKind::Track)
837 }
838
839 fn from_grid_tracks_and_track_count(track_count: TrackCounts, grid_tracks: Vec<GridTrack>) -> Self {
841 DetailedGridTracksInfo {
842 negative_implicit_tracks: track_count.negative_implicit,
843 explicit_tracks: track_count.explicit,
844 positive_implicit_tracks: track_count.positive_implicit,
845 gutters: DetailedGridTracksInfo::gutters_from_grid_track_layout(&grid_tracks),
846 sizes: DetailedGridTracksInfo::sizes_from_grid_track_layout(&grid_tracks),
847 }
848 }
849}
850
851#[derive(Debug, Clone, PartialEq)]
856#[cfg(feature = "detailed_layout_info")]
857pub struct DetailedGridItemsInfo {
858 pub row_start: u16,
860 pub row_end: u16,
862 pub column_start: u16,
864 pub column_end: u16,
866}
867
868#[cfg(feature = "detailed_layout_info")]
870impl DetailedGridItemsInfo {
871 #[inline(always)]
873 fn from_grid_item(grid_item: &GridItem) -> Self {
874 #[inline(always)]
876 fn to_one_indexed_grid_line(grid_track_index: u16) -> u16 {
877 grid_track_index / 2 + 1
878 }
879
880 DetailedGridItemsInfo {
881 row_start: to_one_indexed_grid_line(grid_item.row_indexes.start),
882 row_end: to_one_indexed_grid_line(grid_item.row_indexes.end),
883 column_start: to_one_indexed_grid_line(grid_item.column_indexes.start),
884 column_end: to_one_indexed_grid_line(grid_item.column_indexes.end),
885 }
886 }
887}