1use crate::geometry::{AbsoluteAxis, AbstractAxis, InBothAbsAxis};
4use crate::geometry::{Line, Point, Rect, Size};
5use crate::style::{AlignContent, AlignItems, AlignSelf, AvailableSpace, Display, Overflow, Position};
6use crate::style_helpers::*;
7use crate::tree::{Layout, LayoutInput, LayoutOutput, RunMode, SizingMode};
8use crate::tree::{NodeId, PartialLayoutTree, PartialLayoutTreeExt};
9use crate::util::debug::debug_log;
10use crate::util::sys::{f32_max, GridTrackVec, Vec};
11use crate::util::MaybeMath;
12use crate::util::{MaybeResolve, ResolveOrZero};
13use alignment::{align_and_position_item, align_tracks};
14use explicit_grid::{compute_explicit_grid_size_in_axis, initialize_grid_tracks};
15use implicit_grid::compute_grid_size_estimate;
16use placement::place_grid_items;
17use track_sizing::{
18 determine_if_item_crosses_flexible_or_intrinsic_tracks, resolve_item_track_indexes, track_sizing_algorithm,
19};
20use types::{CellOccupancyMatrix, GridTrack};
21
22pub(crate) use types::{GridCoordinate, GridLine, OriginZeroLine};
23
24mod alignment;
25mod explicit_grid;
26mod implicit_grid;
27mod placement;
28mod track_sizing;
29mod types;
30mod util;
31
32pub fn compute_grid_layout(tree: &mut impl PartialLayoutTree, node: NodeId, inputs: LayoutInput) -> LayoutOutput {
39 let LayoutInput { known_dimensions, parent_size, available_space, run_mode, .. } = inputs;
40
41 let get_child_styles_iter = |node| tree.child_ids(node).map(|child_node: NodeId| tree.get_style(child_node));
42 let style = tree.get_style(node).clone();
43 let child_styles_iter = get_child_styles_iter(node);
44
45 let preferred_size = if inputs.sizing_mode == SizingMode::InherentSize {
46 style.size.maybe_resolve(parent_size).maybe_apply_aspect_ratio(style.aspect_ratio)
47 } else {
48 Size::NONE
49 };
50
51 let explicit_col_count = compute_explicit_grid_size_in_axis(&style, preferred_size, AbsoluteAxis::Horizontal);
54 let explicit_row_count = compute_explicit_grid_size_in_axis(&style, preferred_size, AbsoluteAxis::Vertical);
55
56 let (est_col_counts, est_row_counts) =
60 compute_grid_size_estimate(explicit_col_count, explicit_row_count, child_styles_iter);
61
62 let mut items = Vec::with_capacity(tree.child_count(node));
65 let mut cell_occupancy_matrix = CellOccupancyMatrix::with_track_counts(est_col_counts, est_row_counts);
66 let in_flow_children_iter = || {
67 tree.child_ids(node)
68 .enumerate()
69 .map(|(index, child_node)| (index, child_node, tree.get_style(child_node)))
70 .filter(|(_, _, style)| style.display != Display::None && style.position != Position::Absolute)
71 };
72 place_grid_items(
73 &mut cell_occupancy_matrix,
74 &mut items,
75 in_flow_children_iter,
76 style.grid_auto_flow,
77 style.align_items.unwrap_or(AlignItems::Stretch),
78 style.justify_items.unwrap_or(AlignItems::Stretch),
79 );
80
81 let final_col_counts = *cell_occupancy_matrix.track_counts(AbsoluteAxis::Horizontal);
83 let final_row_counts = *cell_occupancy_matrix.track_counts(AbsoluteAxis::Vertical);
84
85 let mut columns = GridTrackVec::new();
89 let mut rows = GridTrackVec::new();
90 initialize_grid_tracks(
91 &mut columns,
92 final_col_counts,
93 &style.grid_template_columns,
94 &style.grid_auto_columns,
95 style.gap.width,
96 |column_index| cell_occupancy_matrix.column_is_occupied(column_index),
97 );
98 initialize_grid_tracks(
99 &mut rows,
100 final_row_counts,
101 &style.grid_template_rows,
102 &style.grid_auto_rows,
103 style.gap.height,
104 |row_index| cell_occupancy_matrix.row_is_occupied(row_index),
105 );
106
107 let padding = style.padding.resolve_or_zero(parent_size.width);
110 let border = style.border.resolve_or_zero(parent_size.width);
111 let padding_border = padding + border;
112 let padding_border_size = padding_border.sum_axes();
113 let aspect_ratio = style.aspect_ratio;
114 let min_size = style.min_size.maybe_resolve(parent_size).maybe_apply_aspect_ratio(aspect_ratio);
115 let max_size = style.max_size.maybe_resolve(parent_size).maybe_apply_aspect_ratio(aspect_ratio);
116 let size = preferred_size;
117
118 let scrollbar_gutter = style.overflow.transpose().map(|overflow| match overflow {
122 Overflow::Scroll => style.scrollbar_width,
123 _ => 0.0,
124 });
125 let mut content_box_inset = padding_border;
127 content_box_inset.right += scrollbar_gutter.x;
128 content_box_inset.bottom += scrollbar_gutter.y;
129
130 let constrained_available_space = known_dimensions
131 .or(size)
132 .map(|size| size.map(AvailableSpace::Definite))
133 .unwrap_or(available_space)
134 .maybe_clamp(min_size, max_size)
135 .maybe_max(padding_border_size);
136
137 let available_grid_space = Size {
138 width: constrained_available_space
139 .width
140 .map_definite_value(|space| space - content_box_inset.horizontal_axis_sum()),
141 height: constrained_available_space
142 .height
143 .map_definite_value(|space| space - content_box_inset.vertical_axis_sum()),
144 };
145
146 let outer_node_size = known_dimensions.or(size).maybe_clamp(min_size, max_size).maybe_max(padding_border_size);
147 let mut inner_node_size = Size {
148 width: outer_node_size.width.map(|space| space - content_box_inset.horizontal_axis_sum()),
149 height: outer_node_size.height.map(|space| space - content_box_inset.vertical_axis_sum()),
150 };
151
152 debug_log!("parent_size", dbg:parent_size);
153 debug_log!("outer_node_size", dbg:outer_node_size);
154 debug_log!("inner_node_size", dbg:inner_node_size);
155
156 resolve_item_track_indexes(&mut items, final_col_counts, final_row_counts);
162
163 determine_if_item_crosses_flexible_or_intrinsic_tracks(&mut items, &columns, &rows);
166
167 let has_baseline_aligned_item = items.iter().any(|item| item.align_self == AlignSelf::Baseline);
169
170 track_sizing_algorithm(
172 tree,
173 AbstractAxis::Inline,
174 min_size.get(AbstractAxis::Inline),
175 max_size.get(AbstractAxis::Inline),
176 style.grid_align_content(AbstractAxis::Block),
177 available_grid_space,
178 inner_node_size,
179 &mut columns,
180 &mut rows,
181 &mut items,
182 |track: &GridTrack, parent_size: Option<f32>| track.max_track_sizing_function.definite_value(parent_size),
183 has_baseline_aligned_item,
184 );
185 let initial_column_sum = columns.iter().map(|track| track.base_size).sum::<f32>();
186 inner_node_size.width = inner_node_size.width.or_else(|| initial_column_sum.into());
187
188 items.iter_mut().for_each(|item| item.available_space_cache = None);
189
190 track_sizing_algorithm(
192 tree,
193 AbstractAxis::Block,
194 min_size.get(AbstractAxis::Block),
195 max_size.get(AbstractAxis::Block),
196 style.grid_align_content(AbstractAxis::Inline),
197 available_grid_space,
198 inner_node_size,
199 &mut rows,
200 &mut columns,
201 &mut items,
202 |track: &GridTrack, _| Some(track.base_size),
203 false, );
205 let initial_row_sum = rows.iter().map(|track| track.base_size).sum::<f32>();
206 inner_node_size.height = inner_node_size.height.or_else(|| initial_row_sum.into());
207
208 debug_log!("initial_column_sum", dbg:initial_column_sum);
209 debug_log!("initial_row_sum", dbg:initial_row_sum);
210
211 let resolved_style_size = known_dimensions.or(preferred_size);
213 let container_border_box = Size {
214 width: resolved_style_size
215 .get(AbstractAxis::Inline)
216 .unwrap_or_else(|| initial_column_sum + content_box_inset.horizontal_axis_sum())
217 .maybe_clamp(min_size.width, max_size.width)
218 .max(padding_border_size.width),
219 height: resolved_style_size
220 .get(AbstractAxis::Block)
221 .unwrap_or_else(|| initial_row_sum + content_box_inset.vertical_axis_sum())
222 .maybe_clamp(min_size.height, max_size.height)
223 .max(padding_border_size.height),
224 };
225 let container_content_box = Size {
226 width: f32_max(0.0, container_border_box.width - content_box_inset.horizontal_axis_sum()),
227 height: f32_max(0.0, container_border_box.height - content_box_inset.vertical_axis_sum()),
228 };
229
230 if run_mode == RunMode::ComputeSize {
232 return LayoutOutput::from_outer_size(container_border_box);
233 }
234
235 if !available_grid_space.width.is_definite() {
239 for column in &mut columns {
240 let min: Option<f32> =
241 column.min_track_sizing_function.resolved_percentage_size(container_content_box.width);
242 let max: Option<f32> =
243 column.max_track_sizing_function.resolved_percentage_size(container_content_box.width);
244 column.base_size = column.base_size.maybe_clamp(min, max);
245 }
246 }
247 if !available_grid_space.height.is_definite() {
248 for row in &mut rows {
249 let min: Option<f32> = row.min_track_sizing_function.resolved_percentage_size(container_content_box.height);
250 let max: Option<f32> = row.max_track_sizing_function.resolved_percentage_size(container_content_box.height);
251 row.base_size = row.base_size.maybe_clamp(min, max);
252 }
253 }
254
255 let mut rerun_column_sizing;
260
261 let has_percentage_column = columns.iter().any(|track| track.uses_percentage());
262 let parent_width_indefinite = !available_space.width.is_definite();
263 rerun_column_sizing = parent_width_indefinite && has_percentage_column;
264
265 if !rerun_column_sizing {
266 let min_content_contribution_changed = items
267 .iter_mut()
268 .filter(|item| item.crosses_intrinsic_column)
269 .map(|item| {
270 let available_space = item.available_space(
271 AbstractAxis::Inline,
272 &rows,
273 inner_node_size.height,
274 |track: &GridTrack, _| Some(track.base_size),
275 );
276 let new_min_content_contribution =
277 item.min_content_contribution(AbstractAxis::Inline, tree, available_space, inner_node_size);
278
279 let has_changed = Some(new_min_content_contribution) != item.min_content_contribution_cache.width;
280
281 item.available_space_cache = Some(available_space);
282 item.min_content_contribution_cache.width = Some(new_min_content_contribution);
283 item.max_content_contribution_cache.width = None;
284 item.minimum_contribution_cache.width = None;
285
286 has_changed
287 })
288 .any(|has_changed| has_changed);
289 rerun_column_sizing = min_content_contribution_changed;
290 } else {
291 items.iter_mut().for_each(|item| {
293 item.available_space_cache = None;
294 item.min_content_contribution_cache.width = None;
295 item.max_content_contribution_cache.width = None;
296 item.minimum_contribution_cache.width = None;
297 });
298 }
299
300 if rerun_column_sizing {
301 track_sizing_algorithm(
303 tree,
304 AbstractAxis::Inline,
305 min_size.get(AbstractAxis::Inline),
306 max_size.get(AbstractAxis::Inline),
307 style.grid_align_content(AbstractAxis::Block),
308 available_grid_space,
309 inner_node_size,
310 &mut columns,
311 &mut rows,
312 &mut items,
313 |track: &GridTrack, _| Some(track.base_size),
314 has_baseline_aligned_item,
315 );
316
317 let mut rerun_row_sizing;
322
323 let has_percentage_row = rows.iter().any(|track| track.uses_percentage());
324 let parent_height_indefinite = !available_space.height.is_definite();
325 rerun_row_sizing = parent_height_indefinite && has_percentage_row;
326
327 if !rerun_row_sizing {
328 let min_content_contribution_changed = items
329 .iter_mut()
330 .filter(|item| item.crosses_intrinsic_column)
331 .map(|item| {
332 let available_space = item.available_space(
333 AbstractAxis::Block,
334 &columns,
335 inner_node_size.width,
336 |track: &GridTrack, _| Some(track.base_size),
337 );
338 let new_min_content_contribution =
339 item.min_content_contribution(AbstractAxis::Block, tree, available_space, inner_node_size);
340
341 let has_changed = Some(new_min_content_contribution) != item.min_content_contribution_cache.height;
342
343 item.available_space_cache = Some(available_space);
344 item.min_content_contribution_cache.height = Some(new_min_content_contribution);
345 item.max_content_contribution_cache.height = None;
346 item.minimum_contribution_cache.height = None;
347
348 has_changed
349 })
350 .any(|has_changed| has_changed);
351 rerun_row_sizing = min_content_contribution_changed;
352 } else {
353 items.iter_mut().for_each(|item| {
354 item.available_space_cache = None;
356 item.min_content_contribution_cache.height = None;
357 item.max_content_contribution_cache.height = None;
358 item.minimum_contribution_cache.height = None;
359 });
360 }
361
362 if rerun_row_sizing {
363 track_sizing_algorithm(
365 tree,
366 AbstractAxis::Block,
367 min_size.get(AbstractAxis::Block),
368 max_size.get(AbstractAxis::Block),
369 style.grid_align_content(AbstractAxis::Inline),
370 available_grid_space,
371 inner_node_size,
372 &mut rows,
373 &mut columns,
374 &mut items,
375 |track: &GridTrack, _| Some(track.base_size),
376 false, );
378 }
379 }
380
381 align_tracks(
385 container_content_box.get(AbstractAxis::Inline),
386 Line { start: padding.left, end: padding.right },
387 Line { start: border.left, end: border.right },
388 &mut columns,
389 style.justify_content.unwrap_or(AlignContent::Stretch),
390 );
391 align_tracks(
393 container_content_box.get(AbstractAxis::Block),
394 Line { start: padding.top, end: padding.bottom },
395 Line { start: border.top, end: border.bottom },
396 &mut rows,
397 style.align_content.unwrap_or(AlignContent::Stretch),
398 );
399
400 #[cfg_attr(not(feature = "content_size"), allow(unused_mut))]
403 let mut item_content_size_contribution = Size::ZERO;
404
405 items.sort_by_key(|item| item.source_order);
407
408 let container_alignment_styles = InBothAbsAxis { horizontal: style.justify_items, vertical: style.align_items };
409
410 for (index, item) in items.iter_mut().enumerate() {
412 let grid_area = Rect {
413 top: rows[item.row_indexes.start as usize + 1].offset,
414 bottom: rows[item.row_indexes.end as usize].offset,
415 left: columns[item.column_indexes.start as usize + 1].offset,
416 right: columns[item.column_indexes.end as usize].offset,
417 };
418 #[cfg_attr(not(feature = "content_size"), allow(unused_variables))]
419 let (content_size_contribution, y_position, height) = align_and_position_item(
420 tree,
421 item.node,
422 index as u32,
423 grid_area,
424 container_alignment_styles,
425 item.baseline_shim,
426 );
427 item.y_position = y_position;
428 item.height = height;
429
430 #[cfg(feature = "content_size")]
431 {
432 item_content_size_contribution = item_content_size_contribution.f32_max(content_size_contribution);
433 }
434 }
435
436 let mut order = items.len() as u32;
438 (0..tree.child_count(node)).for_each(|index| {
439 let child = tree.get_child_id(node, index);
440 let child_style = tree.get_style(child);
441
442 if child_style.display == Display::None {
444 tree.set_unrounded_layout(child, &Layout::with_order(order));
445 tree.perform_child_layout(
446 child,
447 Size::NONE,
448 Size::NONE,
449 Size::MAX_CONTENT,
450 SizingMode::InherentSize,
451 Line::FALSE,
452 );
453 order += 1;
454 return;
455 }
456
457 if child_style.position == Position::Absolute {
459 let maybe_col_indexes = child_style
462 .grid_column
463 .into_origin_zero(final_col_counts.explicit)
464 .resolve_absolutely_positioned_grid_tracks()
465 .map(|maybe_grid_line| {
466 maybe_grid_line.map(|line: OriginZeroLine| line.into_track_vec_index(final_col_counts))
467 });
468 let maybe_row_indexes = child_style
471 .grid_row
472 .into_origin_zero(final_row_counts.explicit)
473 .resolve_absolutely_positioned_grid_tracks()
474 .map(|maybe_grid_line| {
475 maybe_grid_line.map(|line: OriginZeroLine| line.into_track_vec_index(final_row_counts))
476 });
477
478 let grid_area = Rect {
479 top: maybe_row_indexes.start.map(|index| rows[index].offset).unwrap_or(border.top),
480 bottom: maybe_row_indexes
481 .end
482 .map(|index| rows[index].offset)
483 .unwrap_or(container_border_box.height - border.bottom),
484 left: maybe_col_indexes.start.map(|index| columns[index].offset).unwrap_or(border.left),
485 right: maybe_col_indexes
486 .end
487 .map(|index| columns[index].offset)
488 .unwrap_or(container_border_box.width - border.right),
489 };
490 #[cfg_attr(not(feature = "content_size"), allow(unused_variables))]
492 let (content_size_contribution, _, _) =
493 align_and_position_item(tree, child, order, grid_area, container_alignment_styles, 0.0);
494 #[cfg(feature = "content_size")]
495 {
496 item_content_size_contribution = item_content_size_contribution.f32_max(content_size_contribution);
497 }
498
499 order += 1;
500 }
501 });
502
503 if items.is_empty() {
505 return LayoutOutput::from_outer_size(container_border_box);
506 }
507
508 let grid_container_baseline: f32 = {
510 items.sort_by_key(|item| item.row_indexes.start);
512
513 let first_row = items[0].row_indexes.start;
515
516 let first_row_items = &items[0..].split(|item| item.row_indexes.start != first_row).next().unwrap();
518
519 let row_has_baseline_item = first_row_items.iter().any(|item| item.align_self == AlignSelf::Baseline);
521
522 let item = if row_has_baseline_item {
523 first_row_items.iter().find(|item| item.align_self == AlignSelf::Baseline).unwrap()
524 } else {
525 &first_row_items[0]
526 };
527
528 item.y_position + item.baseline.unwrap_or(item.height)
529 };
530
531 LayoutOutput::from_sizes_and_baselines(
532 container_border_box,
533 item_content_size_contribution,
534 Point { x: None, y: Some(grid_container_baseline) },
535 )
536}