1use super::types::{CellOccupancyMatrix, CellOccupancyState, GridItem};
4use super::{NamedLineResolver, OriginZeroLine};
5use crate::geometry::Line;
6use crate::geometry::{AbsoluteAxis, InBothAbsAxis};
7use crate::style::{AlignItems, GridAutoFlow, OriginZeroGridPlacement};
8use crate::tree::NodeId;
9use crate::util::sys::Vec;
10use crate::{CoreStyle, Direction, GridItemStyle};
11
12#[inline]
13fn axis_is_reversed(direction: Direction, axis: AbsoluteAxis) -> bool {
15 direction.is_rtl() && axis == AbsoluteAxis::Horizontal
16}
17
18#[inline]
19fn advance_position(position: OriginZeroLine, axis_is_reversed: bool) -> OriginZeroLine {
21 if axis_is_reversed {
22 OriginZeroLine(position.0 - 1)
23 } else {
24 OriginZeroLine(position.0 + 1)
25 }
26}
27
28#[inline]
29fn search_start_line(
31 grid_start_line: OriginZeroLine,
32 grid_end_line: OriginZeroLine,
33 axis_is_reversed: bool,
34) -> OriginZeroLine {
35 if axis_is_reversed {
36 grid_end_line - 1
37 } else {
38 grid_start_line
39 }
40}
41
42#[inline]
43fn resolve_indefinite_grid_span(position: OriginZeroLine, span: u16, axis_is_reversed: bool) -> Line<OriginZeroLine> {
45 if axis_is_reversed {
46 Line { start: (position - span) + 1, end: position + 1 }
47 } else {
48 Line { start: position, end: position + span }
49 }
50}
51
52#[inline]
53fn mirror_horizontal_span(span: Line<OriginZeroLine>, explicit_col_count: u16) -> Line<OriginZeroLine> {
55 let explicit_col_end_line = explicit_col_count as i16;
56 Line {
57 start: OriginZeroLine(explicit_col_end_line - span.end.0),
58 end: OriginZeroLine(explicit_col_end_line - span.start.0),
59 }
60}
61
62#[inline]
63fn maybe_mirror_span(
65 span: Line<OriginZeroLine>,
66 axis: AbsoluteAxis,
67 direction: Direction,
68 explicit_col_count: u16,
69) -> Line<OriginZeroLine> {
70 if axis == AbsoluteAxis::Horizontal && direction.is_rtl() {
71 mirror_horizontal_span(span, explicit_col_count)
72 } else {
73 span
74 }
75}
76
77#[allow(clippy::too_many_arguments)]
82pub(super) fn place_grid_items<'a, S, ChildIter>(
83 cell_occupancy_matrix: &mut CellOccupancyMatrix,
84 items: &mut Vec<GridItem>,
85 children_iter: impl Fn() -> ChildIter,
86 direction: Direction,
87 grid_auto_flow: GridAutoFlow,
88 align_items: AlignItems,
89 justify_items: AlignItems,
90 named_line_resolver: &NamedLineResolver<<S as CoreStyle>::CustomIdent>,
91) where
92 S: GridItemStyle + 'a,
93 ChildIter: Iterator<Item = (usize, NodeId, S)>,
94{
95 let primary_axis = grid_auto_flow.primary_axis();
96 let secondary_axis = primary_axis.other_axis();
97 let explicit_col_count = cell_occupancy_matrix.track_counts(AbsoluteAxis::Horizontal).explicit;
98
99 let map_child_style_to_origin_zero_placement = {
100 let explicit_row_count = cell_occupancy_matrix.track_counts(AbsoluteAxis::Vertical).explicit;
101 move |(index, node, style): (usize, NodeId, S)| -> (_, _, _, S) {
102 let origin_zero_placement = InBothAbsAxis {
103 horizontal: named_line_resolver
104 .resolve_column_names(&style.grid_column())
105 .map(|placement| placement.into_origin_zero_placement(explicit_col_count)),
106 vertical: named_line_resolver
107 .resolve_row_names(&style.grid_row())
108 .map(|placement| placement.into_origin_zero_placement(explicit_row_count)),
109 };
110 (index, node, origin_zero_placement, style)
111 }
112 };
113
114 let mut idx = 0;
116 children_iter()
117 .map(map_child_style_to_origin_zero_placement)
118 .filter(|(_, _, placement, _)| placement.horizontal.is_definite() && placement.vertical.is_definite())
119 .for_each(|(index, child_node, child_placement, style)| {
120 idx += 1;
121 #[cfg(test)]
122 println!("Definite Item {idx}\n==============");
123
124 let (row_span, col_span) =
125 place_definite_grid_item(child_placement, primary_axis, direction, explicit_col_count);
126 record_grid_placement(
127 cell_occupancy_matrix,
128 items,
129 child_node,
130 index,
131 style,
132 align_items,
133 justify_items,
134 primary_axis,
135 row_span,
136 col_span,
137 CellOccupancyState::DefinitelyPlaced,
138 );
139 });
140
141 let mut idx = 0;
143 children_iter()
144 .map(map_child_style_to_origin_zero_placement)
145 .filter(|(_, _, placement, _)| {
146 placement.get(secondary_axis).is_definite() && !placement.get(primary_axis).is_definite()
147 })
148 .for_each(|(index, child_node, child_placement, style)| {
149 idx += 1;
150 #[cfg(test)]
151 println!("Definite Secondary Item {idx}\n==============");
152
153 let (primary_span, secondary_span) = place_definite_secondary_axis_item(
154 &*cell_occupancy_matrix,
155 child_placement,
156 grid_auto_flow,
157 direction,
158 explicit_col_count,
159 );
160
161 record_grid_placement(
162 cell_occupancy_matrix,
163 items,
164 child_node,
165 index,
166 style,
167 align_items,
168 justify_items,
169 primary_axis,
170 primary_span,
171 secondary_span,
172 CellOccupancyState::AutoPlaced,
173 );
174 });
175
176 let primary_axis = grid_auto_flow.primary_axis();
196 let secondary_axis = primary_axis.other_axis();
197 let primary_axis_grid_start_line = cell_occupancy_matrix.track_counts(primary_axis).implicit_start_line();
198 let primary_axis_grid_end_line = cell_occupancy_matrix.track_counts(primary_axis).implicit_end_line();
199 let secondary_axis_grid_start_line = cell_occupancy_matrix.track_counts(secondary_axis).implicit_start_line();
200 let secondary_axis_grid_end_line = cell_occupancy_matrix.track_counts(secondary_axis).implicit_end_line();
201 let primary_axis_is_reversed = axis_is_reversed(direction, primary_axis);
202 let grid_start_position = (
203 search_start_line(primary_axis_grid_start_line, primary_axis_grid_end_line, primary_axis_is_reversed),
204 search_start_line(
205 secondary_axis_grid_start_line,
206 secondary_axis_grid_end_line,
207 axis_is_reversed(direction, secondary_axis),
208 ),
209 );
210 let mut grid_position = grid_start_position;
211 let mut idx = 0;
212 children_iter()
213 .map(map_child_style_to_origin_zero_placement)
214 .filter(|(_, _, placement, _)| !placement.get(secondary_axis).is_definite())
215 .for_each(|(index, child_node, child_placement, style)| {
216 idx += 1;
217 #[cfg(test)]
218 println!("\nAuto Item {idx}\n==============");
219
220 let (primary_span, secondary_span) = place_indefinitely_positioned_item(
222 &*cell_occupancy_matrix,
223 child_placement,
224 grid_auto_flow,
225 grid_position,
226 direction,
227 explicit_col_count,
228 );
229
230 record_grid_placement(
232 cell_occupancy_matrix,
233 items,
234 child_node,
235 index,
236 style,
237 align_items,
238 justify_items,
239 primary_axis,
240 primary_span,
241 secondary_span,
242 CellOccupancyState::AutoPlaced,
243 );
244
245 grid_position = match (grid_auto_flow.is_dense(), primary_axis_is_reversed) {
248 (true, _) => grid_start_position,
249 (false, false) => (primary_span.end, secondary_span.start),
250 (false, true) => (primary_span.start, secondary_span.start),
251 };
252 });
253}
254
255fn place_definite_grid_item(
258 placement: InBothAbsAxis<Line<OriginZeroGridPlacement>>,
259 primary_axis: AbsoluteAxis,
260 direction: Direction,
261 explicit_col_count: u16,
262) -> (Line<OriginZeroLine>, Line<OriginZeroLine>) {
263 let primary_span = maybe_mirror_span(
265 placement.get(primary_axis).resolve_definite_grid_lines(),
266 primary_axis,
267 direction,
268 explicit_col_count,
269 );
270 let secondary_span = maybe_mirror_span(
271 placement.get(primary_axis.other_axis()).resolve_definite_grid_lines(),
272 primary_axis.other_axis(),
273 direction,
274 explicit_col_count,
275 );
276
277 (primary_span, secondary_span)
278}
279
280fn place_definite_secondary_axis_item(
283 cell_occupancy_matrix: &CellOccupancyMatrix,
284 placement: InBothAbsAxis<Line<OriginZeroGridPlacement>>,
285 auto_flow: GridAutoFlow,
286 direction: Direction,
287 explicit_col_count: u16,
288) -> (Line<OriginZeroLine>, Line<OriginZeroLine>) {
289 let primary_axis = auto_flow.primary_axis();
290 let secondary_axis = primary_axis.other_axis();
291 let primary_axis_is_reversed = axis_is_reversed(direction, primary_axis);
292 let primary_axis_grid_start_line = cell_occupancy_matrix.track_counts(primary_axis).implicit_start_line();
293 let primary_axis_grid_end_line = cell_occupancy_matrix.track_counts(primary_axis).implicit_end_line();
294
295 let secondary_axis_placement = maybe_mirror_span(
296 placement.get(secondary_axis).resolve_definite_grid_lines(),
297 secondary_axis,
298 direction,
299 explicit_col_count,
300 );
301 let starting_position = match auto_flow.is_dense() {
302 true => search_start_line(primary_axis_grid_start_line, primary_axis_grid_end_line, primary_axis_is_reversed),
303 false => {
304 let lookup_result = if primary_axis_is_reversed {
305 cell_occupancy_matrix.first_of_type(
306 primary_axis,
307 secondary_axis_placement.start,
308 CellOccupancyState::AutoPlaced,
309 )
310 } else {
311 cell_occupancy_matrix.last_of_type(
312 primary_axis,
313 secondary_axis_placement.start,
314 CellOccupancyState::AutoPlaced,
315 )
316 };
317 lookup_result.unwrap_or(search_start_line(
318 primary_axis_grid_start_line,
319 primary_axis_grid_end_line,
320 primary_axis_is_reversed,
321 ))
322 }
323 };
324 let primary_axis_span = placement.get(primary_axis).indefinite_span();
325
326 let mut position: OriginZeroLine = starting_position;
327 loop {
328 let primary_axis_placement =
329 resolve_indefinite_grid_span(position, primary_axis_span, primary_axis_is_reversed);
330
331 let does_fit = cell_occupancy_matrix.line_area_is_unoccupied(
332 primary_axis,
333 primary_axis_placement,
334 secondary_axis_placement,
335 );
336
337 if does_fit {
338 return (primary_axis_placement, secondary_axis_placement);
339 } else {
340 position = advance_position(position, primary_axis_is_reversed);
341 }
342 }
343}
344
345fn place_indefinitely_positioned_item(
348 cell_occupancy_matrix: &CellOccupancyMatrix,
349 placement: InBothAbsAxis<Line<OriginZeroGridPlacement>>,
350 auto_flow: GridAutoFlow,
351 grid_position: (OriginZeroLine, OriginZeroLine),
352 direction: Direction,
353 explicit_col_count: u16,
354) -> (Line<OriginZeroLine>, Line<OriginZeroLine>) {
355 let primary_axis = auto_flow.primary_axis();
356 let secondary_axis = primary_axis.other_axis();
357 let primary_axis_is_reversed = axis_is_reversed(direction, primary_axis);
358 let secondary_axis_is_reversed = axis_is_reversed(direction, secondary_axis);
359
360 let primary_placement_style = placement.get(primary_axis);
361 let secondary_placement_style = placement.get(secondary_axis);
362
363 let secondary_span = secondary_placement_style.indefinite_span();
364 let has_definite_primary_axis_position = primary_placement_style.is_definite();
365 let primary_axis_grid_start_line = cell_occupancy_matrix.track_counts(primary_axis).implicit_start_line();
366 let primary_axis_grid_end_line = cell_occupancy_matrix.track_counts(primary_axis).implicit_end_line();
367 let secondary_axis_grid_start_line = cell_occupancy_matrix.track_counts(secondary_axis).implicit_start_line();
368 let secondary_axis_grid_end_line = cell_occupancy_matrix.track_counts(secondary_axis).implicit_end_line();
369 let primary_start_position =
370 search_start_line(primary_axis_grid_start_line, primary_axis_grid_end_line, primary_axis_is_reversed);
371 let secondary_start_position =
372 search_start_line(secondary_axis_grid_start_line, secondary_axis_grid_end_line, secondary_axis_is_reversed);
373
374 let line_area_is_occupied = |primary_span, secondary_span| {
375 !cell_occupancy_matrix.line_area_is_unoccupied(primary_axis, primary_span, secondary_span)
376 };
377
378 let (mut primary_idx, mut secondary_idx) = grid_position;
379
380 if has_definite_primary_axis_position {
381 let primary_span = maybe_mirror_span(
382 primary_placement_style.resolve_definite_grid_lines(),
383 primary_axis,
384 direction,
385 explicit_col_count,
386 );
387
388 secondary_idx = match auto_flow.is_dense() {
390 true => secondary_start_position,
392 false => {
393 let should_advance_secondary = if primary_axis_is_reversed {
394 primary_span.start > primary_idx
395 } else {
396 primary_span.start < primary_idx
397 };
398 if should_advance_secondary {
399 advance_position(secondary_idx, secondary_axis_is_reversed)
400 } else {
401 secondary_idx
402 }
403 }
404 };
405
406 loop {
409 let secondary_span =
410 resolve_indefinite_grid_span(secondary_idx, secondary_span, secondary_axis_is_reversed);
411
412 if line_area_is_occupied(primary_span, secondary_span) {
414 secondary_idx = advance_position(secondary_idx, secondary_axis_is_reversed);
415 continue;
416 }
417
418 return (primary_span, secondary_span);
420 }
421 } else {
422 let primary_span = primary_placement_style.indefinite_span();
423
424 loop {
428 let primary_span = resolve_indefinite_grid_span(primary_idx, primary_span, primary_axis_is_reversed);
429 let secondary_span =
430 resolve_indefinite_grid_span(secondary_idx, secondary_span, secondary_axis_is_reversed);
431
432 let primary_out_of_bounds = if primary_axis_is_reversed {
435 primary_span.start < primary_axis_grid_start_line
436 } else {
437 primary_span.end > primary_axis_grid_end_line
438 };
439 if primary_out_of_bounds {
440 secondary_idx = advance_position(secondary_idx, secondary_axis_is_reversed);
441 primary_idx = primary_start_position;
442 continue;
443 }
444
445 if line_area_is_occupied(primary_span, secondary_span) {
447 primary_idx = advance_position(primary_idx, primary_axis_is_reversed);
448 continue;
449 }
450
451 return (primary_span, secondary_span);
453 }
454 }
455}
456
457#[allow(clippy::too_many_arguments)]
460fn record_grid_placement<S: GridItemStyle>(
461 cell_occupancy_matrix: &mut CellOccupancyMatrix,
462 items: &mut Vec<GridItem>,
463 node: NodeId,
464 index: usize,
465 style: S,
466 parent_align_items: AlignItems,
467 parent_justify_items: AlignItems,
468 primary_axis: AbsoluteAxis,
469 primary_span: Line<OriginZeroLine>,
470 secondary_span: Line<OriginZeroLine>,
471 placement_type: CellOccupancyState,
472) {
473 #[cfg(test)]
474 println!("BEFORE placement:");
475 #[cfg(test)]
476 println!("{cell_occupancy_matrix:?}");
477
478 cell_occupancy_matrix.mark_area_as(primary_axis, primary_span, secondary_span, placement_type);
480
481 let (col_span, row_span) = match primary_axis {
483 AbsoluteAxis::Horizontal => (primary_span, secondary_span),
484 AbsoluteAxis::Vertical => (secondary_span, primary_span),
485 };
486 items.push(GridItem::new_with_placement_style_and_order(
487 node,
488 col_span,
489 row_span,
490 style,
491 parent_align_items,
492 parent_justify_items,
493 index as u16,
494 ));
495
496 #[cfg(test)]
497 println!("AFTER placement:");
498 #[cfg(test)]
499 println!("{cell_occupancy_matrix:?}");
500 #[cfg(test)]
501 println!("\n");
502}
503
504#[cfg(test)]
505mod tests {
506
507 mod test_placement_algorithm {
508 use crate::compute::grid::implicit_grid::compute_grid_size_estimate;
509 use crate::compute::grid::types::TrackCounts;
510 use crate::compute::grid::util::*;
511 use crate::compute::grid::CellOccupancyMatrix;
512 use crate::compute::grid::NamedLineResolver;
513 use crate::prelude::*;
514 use crate::style::GridAutoFlow;
515 use crate::Direction;
516
517 use super::super::place_grid_items;
518
519 type ExpectedPlacement = (i16, i16, i16, i16);
520
521 fn placement_test_runner(
522 explicit_col_count: u16,
523 explicit_row_count: u16,
524 children: Vec<(usize, Style, ExpectedPlacement)>,
525 expected_col_counts: TrackCounts,
526 expected_row_counts: TrackCounts,
527 flow: GridAutoFlow,
528 ) {
529 let children_iter = || children.iter().map(|(index, style, _)| (*index, NodeId::from(*index), style));
531 let child_styles_iter = children.iter().map(|(_, style, _)| style);
532 let estimated_sizes =
533 compute_grid_size_estimate(explicit_col_count, explicit_row_count, Direction::Ltr, child_styles_iter);
534 let mut items = Vec::new();
535 let mut cell_occupancy_matrix =
536 CellOccupancyMatrix::with_track_counts(estimated_sizes.0, estimated_sizes.1);
537 let mut name_resolver = NamedLineResolver::new(&Style::DEFAULT, 0, 0);
538 name_resolver.set_explicit_column_count(explicit_col_count);
539 name_resolver.set_explicit_row_count(explicit_row_count);
540
541 place_grid_items(
543 &mut cell_occupancy_matrix,
544 &mut items,
545 children_iter,
546 Direction::Ltr,
547 flow,
548 AlignSelf::Start,
549 AlignSelf::Start,
550 &name_resolver,
552 );
553
554 let mut sorted_children = children.clone();
556 sorted_children.sort_by_key(|child| child.0);
557 for (idx, ((id, _style, expected_placement), item)) in sorted_children.iter().zip(items.iter()).enumerate()
558 {
559 assert_eq!(item.node, NodeId::from(*id));
560 let actual_placement = (item.column.start, item.column.end, item.row.start, item.row.end);
561 assert_eq!(actual_placement, (*expected_placement).into_oz(), "Item {idx} (0-indexed)");
562 }
563
564 let actual_row_counts = *cell_occupancy_matrix.track_counts(crate::compute::grid::AbsoluteAxis::Vertical);
566 assert_eq!(actual_row_counts, expected_row_counts, "row track counts");
567 let actual_col_counts = *cell_occupancy_matrix.track_counts(crate::compute::grid::AbsoluteAxis::Horizontal);
568 assert_eq!(actual_col_counts, expected_col_counts, "column track counts");
569 }
570
571 #[test]
572 fn test_only_fixed_placement() {
573 let flow = GridAutoFlow::Row;
574 let explicit_col_count = 2;
575 let explicit_row_count = 2;
576 let children = {
577 vec![
578 (1, (line(1), auto(), line(1), auto()).into_grid_child(), (0, 1, 0, 1)),
580 (2, (line(-4), auto(), line(-3), auto()).into_grid_child(), (-1, 0, 0, 1)),
581 (3, (line(-3), auto(), line(-4), auto()).into_grid_child(), (0, 1, -1, 0)),
582 (4, (line(3), span(2), line(5), auto()).into_grid_child(), (2, 4, 4, 5)),
583 ]
584 };
585 let expected_cols = TrackCounts { negative_implicit: 1, explicit: 2, positive_implicit: 2 };
586 let expected_rows = TrackCounts { negative_implicit: 1, explicit: 2, positive_implicit: 3 };
587 placement_test_runner(explicit_col_count, explicit_row_count, children, expected_cols, expected_rows, flow);
588 }
589
590 #[test]
591 fn test_placement_spanning_origin() {
592 let flow = GridAutoFlow::Row;
593 let explicit_col_count = 2;
594 let explicit_row_count = 2;
595 let children = {
596 vec![
597 (1, (line(-1), line(-1), line(-1), line(-1)).into_grid_child(), (2, 3, 2, 3)),
599 (2, (line(-1), span(2), line(-1), span(2)).into_grid_child(), (2, 4, 2, 4)),
600 (3, (line(-4), line(-4), line(-4), line(-4)).into_grid_child(), (-1, 0, -1, 0)),
601 (4, (line(-4), span(2), line(-4), span(2)).into_grid_child(), (-1, 1, -1, 1)),
602 ]
603 };
604 let expected_cols = TrackCounts { negative_implicit: 1, explicit: 2, positive_implicit: 2 };
605 let expected_rows = TrackCounts { negative_implicit: 1, explicit: 2, positive_implicit: 2 };
606 placement_test_runner(explicit_col_count, explicit_row_count, children, expected_cols, expected_rows, flow);
607 }
608
609 #[test]
610 fn test_only_auto_placement_row_flow() {
611 let flow = GridAutoFlow::Row;
612 let explicit_col_count = 2;
613 let explicit_row_count = 2;
614 let children = {
615 let auto_child = (auto(), auto(), auto(), auto()).into_grid_child();
616 vec![
617 (1, auto_child.clone(), (0, 1, 0, 1)),
619 (2, auto_child.clone(), (1, 2, 0, 1)),
620 (3, auto_child.clone(), (0, 1, 1, 2)),
621 (4, auto_child.clone(), (1, 2, 1, 2)),
622 (5, auto_child.clone(), (0, 1, 2, 3)),
623 (6, auto_child.clone(), (1, 2, 2, 3)),
624 (7, auto_child.clone(), (0, 1, 3, 4)),
625 (8, auto_child.clone(), (1, 2, 3, 4)),
626 ]
627 };
628 let expected_cols = TrackCounts { negative_implicit: 0, explicit: 2, positive_implicit: 0 };
629 let expected_rows = TrackCounts { negative_implicit: 0, explicit: 2, positive_implicit: 2 };
630 placement_test_runner(explicit_col_count, explicit_row_count, children, expected_cols, expected_rows, flow);
631 }
632
633 #[test]
634 fn test_only_auto_placement_column_flow() {
635 let flow = GridAutoFlow::Column;
636 let explicit_col_count = 2;
637 let explicit_row_count = 2;
638 let children = {
639 let auto_child = (auto(), auto(), auto(), auto()).into_grid_child();
640 vec![
641 (1, auto_child.clone(), (0, 1, 0, 1)),
643 (2, auto_child.clone(), (0, 1, 1, 2)),
644 (3, auto_child.clone(), (1, 2, 0, 1)),
645 (4, auto_child.clone(), (1, 2, 1, 2)),
646 (5, auto_child.clone(), (2, 3, 0, 1)),
647 (6, auto_child.clone(), (2, 3, 1, 2)),
648 (7, auto_child.clone(), (3, 4, 0, 1)),
649 (8, auto_child.clone(), (3, 4, 1, 2)),
650 ]
651 };
652 let expected_cols = TrackCounts { negative_implicit: 0, explicit: 2, positive_implicit: 2 };
653 let expected_rows = TrackCounts { negative_implicit: 0, explicit: 2, positive_implicit: 0 };
654 placement_test_runner(explicit_col_count, explicit_row_count, children, expected_cols, expected_rows, flow);
655 }
656
657 #[test]
658 fn test_oversized_item() {
659 let flow = GridAutoFlow::Row;
660 let explicit_col_count = 2;
661 let explicit_row_count = 2;
662 let children = {
663 vec![
664 (1, (span(5), auto(), auto(), auto()).into_grid_child(), (0, 5, 0, 1)),
666 ]
667 };
668 let expected_cols = TrackCounts { negative_implicit: 0, explicit: 2, positive_implicit: 3 };
669 let expected_rows = TrackCounts { negative_implicit: 0, explicit: 2, positive_implicit: 0 };
670 placement_test_runner(explicit_col_count, explicit_row_count, children, expected_cols, expected_rows, flow);
671 }
672
673 #[test]
674 fn test_fixed_in_secondary_axis() {
675 let flow = GridAutoFlow::Row;
676 let explicit_col_count = 2;
677 let explicit_row_count = 2;
678 let children = {
679 vec![
680 (1, (span(2), auto(), line(1), auto()).into_grid_child(), (0, 2, 0, 1)),
682 (2, (auto(), auto(), line(2), auto()).into_grid_child(), (0, 1, 1, 2)),
683 (3, (auto(), auto(), line(1), auto()).into_grid_child(), (2, 3, 0, 1)),
684 (4, (auto(), auto(), line(4), auto()).into_grid_child(), (0, 1, 3, 4)),
685 ]
686 };
687 let expected_cols = TrackCounts { negative_implicit: 0, explicit: 2, positive_implicit: 1 };
688 let expected_rows = TrackCounts { negative_implicit: 0, explicit: 2, positive_implicit: 2 };
689 placement_test_runner(explicit_col_count, explicit_row_count, children, expected_cols, expected_rows, flow);
690 }
691
692 #[test]
693 fn test_definite_in_secondary_axis_with_fully_definite_negative() {
694 let flow = GridAutoFlow::Row;
695 let explicit_col_count = 2;
696 let explicit_row_count = 2;
697 let children = {
698 vec![
699 (2, (auto(), auto(), line(2), auto()).into_grid_child(), (0, 1, 1, 2)),
701 (1, (line(-4), auto(), line(2), auto()).into_grid_child(), (-1, 0, 1, 2)),
702 (3, (auto(), auto(), line(1), auto()).into_grid_child(), (-1, 0, 0, 1)),
703 ]
704 };
705 let expected_cols = TrackCounts { negative_implicit: 1, explicit: 2, positive_implicit: 0 };
706 let expected_rows = TrackCounts { negative_implicit: 0, explicit: 2, positive_implicit: 0 };
707 placement_test_runner(explicit_col_count, explicit_row_count, children, expected_cols, expected_rows, flow);
708 }
709
710 #[test]
711 fn test_dense_packing_algorithm() {
712 let flow = GridAutoFlow::RowDense;
713 let explicit_col_count = 4;
714 let explicit_row_count = 4;
715 let children = {
716 vec![
717 (1, (line(2), auto(), line(1), auto()).into_grid_child(), (1, 2, 0, 1)), (2, (span(2), auto(), auto(), auto()).into_grid_child(), (2, 4, 0, 1)), (3, (auto(), auto(), auto(), auto()).into_grid_child(), (0, 1, 0, 1)), ]
722 };
723 let expected_cols = TrackCounts { negative_implicit: 0, explicit: 4, positive_implicit: 0 };
724 let expected_rows = TrackCounts { negative_implicit: 0, explicit: 4, positive_implicit: 0 };
725 placement_test_runner(explicit_col_count, explicit_row_count, children, expected_cols, expected_rows, flow);
726 }
727
728 #[test]
729 fn test_sparse_packing_algorithm() {
730 let flow = GridAutoFlow::Row;
731 let explicit_col_count = 4;
732 let explicit_row_count = 4;
733 let children = {
734 vec![
735 (1, (auto(), span(3), auto(), auto()).into_grid_child(), (0, 3, 0, 1)), (2, (auto(), span(3), auto(), auto()).into_grid_child(), (0, 3, 1, 2)), (3, (auto(), span(1), auto(), auto()).into_grid_child(), (3, 4, 1, 2)), ]
740 };
741 let expected_cols = TrackCounts { negative_implicit: 0, explicit: 4, positive_implicit: 0 };
742 let expected_rows = TrackCounts { negative_implicit: 0, explicit: 4, positive_implicit: 0 };
743 placement_test_runner(explicit_col_count, explicit_row_count, children, expected_cols, expected_rows, flow);
744 }
745
746 #[test]
747 fn test_auto_placement_in_negative_tracks() {
748 let flow = GridAutoFlow::RowDense;
749 let explicit_col_count = 2;
750 let explicit_row_count = 2;
751 let children = {
752 vec![
753 (1, (line(-5), auto(), line(1), auto()).into_grid_child(), (-2, -1, 0, 1)), (2, (auto(), auto(), line(2), auto()).into_grid_child(), (-2, -1, 1, 2)), (3, (auto(), auto(), auto(), auto()).into_grid_child(), (-1, 0, 0, 1)), ]
758 };
759 let expected_cols = TrackCounts { negative_implicit: 2, explicit: 2, positive_implicit: 0 };
760 let expected_rows = TrackCounts { negative_implicit: 0, explicit: 2, positive_implicit: 0 };
761 placement_test_runner(explicit_col_count, explicit_row_count, children, expected_cols, expected_rows, flow);
762 }
763 }
764}