use rand::prelude::*; pub const CHUNK_SIZE: usize = 32; #[derive(Default, Debug, PartialEq, Clone, Copy)] pub struct Cell(pub u16); #[derive(Debug)] pub struct Dish { pub chunk: Chunk, pub rules: Vec, } #[derive(Debug)] pub struct Chunk { pub contents: Box<[[Cell; CHUNK_SIZE]; CHUNK_SIZE]>, } #[derive(Debug)] pub struct Rule { pub from: RulePattern, pub to: RulePattern, // probability // flip // rotate } impl Chunk { fn new() -> Self { Self { contents: vec![[Cell::EMPTY; CHUNK_SIZE]; CHUNK_SIZE] .into_boxed_slice() .try_into() .unwrap(), } } fn fill_random(mut self) -> Self { for col in self.contents.iter_mut() { for cell in col.iter_mut() { if random::() % 4 == 0 { *cell = Cell(1); } } } self } pub fn get_cell(&self, x: usize, y: usize) -> Cell { self.contents[x][y].clone() } fn set_cell(&mut self, x: usize, y: usize, cell: Cell) { self.contents[x][y] = cell } } impl Dish { pub fn new() -> Self { Self { chunk: Chunk::new().fill_random(), rules: vec![ Rule { from: RulePattern { width: 1, height: 2, contents: vec![Some(Cell(1)), Some(Cell(0))], }, to: RulePattern { width: 1, height: 2, contents: vec![Some(Cell(0)), Some(Cell(1))], }, }, Rule { from: RulePattern { width: 2, height: 2, contents: vec![Some(Cell(1)), None, Some(Cell(1)), Some(Cell(0))], }, to: RulePattern { width: 2, height: 2, contents: vec![Some(Cell(0)), None, Some(Cell(1)), Some(Cell(1))], }, }, Rule { from: RulePattern { width: 2, height: 2, contents: vec![None, Some(Cell(1)), Some(Cell(0)), Some(Cell(1))], }, to: RulePattern { width: 2, height: 2, contents: vec![None, Some(Cell(0)), Some(Cell(1)), Some(Cell(1))], }, }, ], } } pub fn fire_blindly(&mut self) { assert!(!self.rules.is_empty()); let x = random::() % CHUNK_SIZE; let y = random::() % CHUNK_SIZE; let rule = random::() % self.rules.len(); // let rule = &self.rules[rule]; self.fire_rule(rule, x, y); } fn fire_rule(&mut self, rule_index: usize, x: usize, y: usize) { let rule = &self.rules[rule_index]; let width = rule.to.width; let height = rule.to.height; // check is match for dx in 0..width { for dy in 0..height { let x = x + dx; let y = y + dy; if let Some(rule_cell) = rule.from.get(dx, dy) { if self.get_cell(x, y) != Some(rule_cell) { return; } } } } for dx in 0..width { for dy in 0..height { let x = x + dx; let y = y + dy; if let Some(rule_cell) = &self.rules[rule_index].to.get(dx, dy) { self.set_cell(x, y, rule_cell.clone()); } } } } //todo isize pub fn get_cell(&self, x: usize, y: usize) -> Option { if x >= CHUNK_SIZE || y >= CHUNK_SIZE { None } else { Some(self.chunk.get_cell(x, y)) } } //todo isize pub fn set_cell(&mut self, x: usize, y: usize, cell: Cell) { if x >= CHUNK_SIZE || y >= CHUNK_SIZE { return; } self.chunk.set_cell(x, y, cell) } } #[derive(Debug)] pub struct RulePattern { width: usize, height: usize, contents: Vec>, } impl RulePattern { pub fn get(&self, x: usize, y: usize) -> Option { if x >= self.width || y >= self.height { None } else { self.contents[x + self.width * y].clone() } } pub fn get_mut(&mut self, x: usize, y: usize) -> Option<&mut Cell> { if x >= self.width || y >= self.height { None } else { self.contents[x + self.width * y].as_mut() } } pub fn height(&self) -> usize { self.height } pub fn width(&self) -> usize { self.width } } impl Cell { pub const EMPTY: Self = Cell(0); pub fn id(&self) -> usize { self.0 as usize } }