use super::Layout;
use crate::core::{ObjectId, Rect};
pub struct UniformGridLayout {
rows: u32,
cols: u32,
spacing: u32,
margin: u32,
cells: Vec<Option<ObjectId>>,
}
impl UniformGridLayout {
pub fn new(rows: u32, cols: u32, spacing: u32, margin: u32) -> Self {
let safe_rows = rows.max(1);
let safe_cols = cols.max(1);
Self {
rows: safe_rows,
cols: safe_cols,
spacing,
margin,
cells: vec![None; (safe_rows * safe_cols) as usize],
}
}
pub fn set_widget(&mut self, row: u32, col: u32, widget_id: ObjectId) {
if row < self.rows && col < self.cols {
self.cells[(row * self.cols + col) as usize] = Some(widget_id);
}
}
pub fn cell_count(&self) -> usize {
self.cells.iter().filter(|cell| cell.is_some()).count()
}
pub fn total_cells(&self) -> usize {
self.cells.len()
}
pub fn rows(&self) -> u32 {
self.rows
}
pub fn cols(&self) -> u32 {
self.cols
}
pub fn spacing(&self) -> u32 {
self.spacing
}
pub fn margin(&self) -> u32 {
self.margin
}
}
impl Layout for UniformGridLayout {
fn as_any(&self) -> &dyn std::any::Any {
self
}
fn as_any_mut(&mut self) -> &mut dyn std::any::Any {
self
}
fn add_widget(&mut self, widget_id: ObjectId, _stretch: u32) {
if let Some(slot) = self.cells.iter_mut().find(|cell| cell.is_none()) {
*slot = Some(widget_id);
}
}
fn remove_widget(&mut self, widget_id: ObjectId) {
for cell in &mut self.cells {
if *cell == Some(widget_id) {
*cell = None;
}
}
}
fn child_ids(&self) -> Vec<ObjectId> {
self.cells.iter().filter_map(|cell| *cell).collect()
}
fn has_child(&self, id: ObjectId) -> bool {
self.cells.contains(&Some(id))
}
fn clear(&mut self) {
for cell in &mut self.cells {
*cell = None;
}
}
fn update(&self, rect: Rect, widgets: &mut dyn FnMut(ObjectId, Rect)) {
if self.rows == 0 || self.cols == 0 {
return;
}
let spacing_h = (self.cols - 1) * self.spacing;
let spacing_v = (self.rows - 1) * self.spacing;
let available_w = rect.width.saturating_sub(self.margin * 2).saturating_sub(spacing_h);
let available_h = rect.height.saturating_sub(self.margin * 2).saturating_sub(spacing_v);
let cell_width = available_w / self.cols;
let cell_height = available_h / self.rows;
for row in 0..self.rows {
for col in 0..self.cols {
if let Some(widget_id) = self.cells[(row * self.cols + col) as usize] {
let x =
rect.x + self.margin as i32 + (col * (cell_width + self.spacing)) as i32;
let y =
rect.y + self.margin as i32 + (row * (cell_height + self.spacing)) as i32;
widgets(widget_id, Rect::new(x, y, cell_width, cell_height));
}
}
}
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn uniform_grid_has_correct_cell_count() {
let grid = UniformGridLayout::new(3, 4, 2, 1);
assert_eq!(grid.rows(), 3);
assert_eq!(grid.cols(), 4);
assert_eq!(grid.total_cells(), 12);
assert_eq!(grid.cell_count(), 0);
}
#[test]
fn uniform_grid_places_widgets_in_uniform_cells() {
let mut grid = UniformGridLayout::new(2, 2, 0, 0);
grid.set_widget(0, 0, 1);
grid.set_widget(1, 1, 2);
let mut rects = std::collections::HashMap::new();
grid.update(Rect::new(0, 0, 40, 20), &mut |id, rect| {
rects.insert(id, rect);
});
assert_eq!(rects.get(&1), Some(&Rect::new(0, 0, 20, 10)));
assert_eq!(rects.get(&2), Some(&Rect::new(20, 10, 20, 10)));
}
#[test]
fn uniform_grid_respects_margin_and_spacing() {
let mut grid = UniformGridLayout::new(2, 2, 4, 2);
grid.set_widget(0, 0, 1);
grid.set_widget(0, 1, 2);
let mut rects = std::collections::HashMap::new();
grid.update(Rect::new(0, 0, 100, 60), &mut |id, rect| {
rects.insert(id, rect);
});
assert_eq!(rects.get(&1), Some(&Rect::new(2, 2, 46, 26)));
assert_eq!(rects.get(&2), Some(&Rect::new(52, 2, 46, 26)));
}
#[test]
fn uniform_grid_add_widget_fills_empty_cell() {
let mut grid = UniformGridLayout::new(2, 2, 0, 0);
grid.add_widget(42, 1);
assert!(grid.has_child(42));
assert_eq!(grid.cell_count(), 1);
}
#[test]
fn uniform_grid_remove_widget_empties_cell() {
let mut grid = UniformGridLayout::new(2, 2, 0, 0);
grid.set_widget(0, 0, 1);
assert!(grid.has_child(1));
grid.remove_widget(1);
assert!(!grid.has_child(1));
assert_eq!(grid.cell_count(), 0);
}
#[test]
fn uniform_grid_clear_empties_all_cells() {
let mut grid = UniformGridLayout::new(2, 2, 0, 0);
grid.set_widget(0, 0, 1);
grid.set_widget(1, 1, 2);
assert_eq!(grid.cell_count(), 2);
grid.clear();
assert_eq!(grid.cell_count(), 0);
}
#[test]
fn uniform_grid_child_ids_returns_only_occupied() {
let mut grid = UniformGridLayout::new(3, 3, 0, 0);
grid.set_widget(1, 1, 10);
grid.set_widget(2, 0, 20);
let ids = grid.child_ids();
assert_eq!(ids.len(), 2);
assert!(ids.contains(&10));
assert!(ids.contains(&20));
}
#[test]
fn uniform_grid_out_of_bounds_set_widget_is_noop() {
let mut grid = UniformGridLayout::new(2, 2, 0, 0);
grid.set_widget(5, 5, 99); assert_eq!(grid.cell_count(), 0);
}
#[test]
fn uniform_grid_minimum_one_row_col() {
let grid = UniformGridLayout::new(0, 0, 0, 0);
assert_eq!(grid.rows(), 1);
assert_eq!(grid.cols(), 1);
assert_eq!(grid.total_cells(), 1);
}
#[test]
fn uniform_grid_all_cells_same_size() {
let mut grid = UniformGridLayout::new(3, 4, 2, 0);
for r in 0..3 {
for c in 0..4 {
grid.set_widget(r, c, (r * 4 + c + 1) as ObjectId);
}
}
let mut sizes = std::collections::HashSet::new();
grid.update(Rect::new(0, 0, 100, 80), &mut |_id, rect| {
sizes.insert((rect.width, rect.height));
});
assert_eq!(sizes.len(), 1, "all cells must be the same size");
}
}