mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
119 lines
3.2 KiB
Rust
119 lines
3.2 KiB
Rust
use serde::{Deserialize, Serialize};
|
|
use std::ops::{Index, IndexMut};
|
|
use vek::*;
|
|
|
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
pub struct Grid<T> {
|
|
cells: Vec<T>,
|
|
size: Vec2<i32>, // TODO: use u32
|
|
}
|
|
|
|
impl<T> Grid<T> {
|
|
pub fn from_raw(size: Vec2<i32>, raw: impl Into<Vec<T>>) -> Self {
|
|
let cells = raw.into();
|
|
assert_eq!(size.product() as usize, cells.len());
|
|
Self { cells, size }
|
|
}
|
|
|
|
pub fn populate_from(size: Vec2<i32>, mut f: impl FnMut(Vec2<i32>) -> T) -> Self {
|
|
Self {
|
|
cells: (0..size.y)
|
|
.flat_map(|y| (0..size.x).map(move |x| Vec2::new(x, y)))
|
|
.map(&mut f)
|
|
.collect(),
|
|
size,
|
|
}
|
|
}
|
|
|
|
pub fn new(size: Vec2<i32>, default_cell: T) -> Self
|
|
where
|
|
T: Clone,
|
|
{
|
|
Self {
|
|
cells: vec![default_cell; size.product() as usize],
|
|
size,
|
|
}
|
|
}
|
|
|
|
fn idx(&self, pos: Vec2<i32>) -> Option<usize> {
|
|
if pos.map2(self.size, |e, sz| e >= 0 && e < sz).reduce_and() {
|
|
Some((pos.y * self.size.x + pos.x) as usize)
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
pub fn size(&self) -> Vec2<i32> { self.size }
|
|
|
|
pub fn get(&self, pos: Vec2<i32>) -> Option<&T> { self.cells.get(self.idx(pos)?) }
|
|
|
|
pub fn get_mut(&mut self, pos: Vec2<i32>) -> Option<&mut T> {
|
|
let idx = self.idx(pos)?;
|
|
self.cells.get_mut(idx)
|
|
}
|
|
|
|
pub fn set(&mut self, pos: Vec2<i32>, cell: T) -> Option<T> {
|
|
let idx = self.idx(pos)?;
|
|
self.cells.get_mut(idx).map(|c| core::mem::replace(c, cell))
|
|
}
|
|
|
|
pub fn iter(&self) -> impl Iterator<Item = (Vec2<i32>, &T)> + '_ {
|
|
let w = self.size.x;
|
|
self.cells
|
|
.iter()
|
|
.enumerate()
|
|
.map(move |(i, cell)| (Vec2::new(i as i32 % w, i as i32 / w), cell))
|
|
}
|
|
|
|
pub fn iter_mut(&mut self) -> impl Iterator<Item = (Vec2<i32>, &mut T)> + '_ {
|
|
let w = self.size.x;
|
|
self.cells
|
|
.iter_mut()
|
|
.enumerate()
|
|
.map(move |(i, cell)| (Vec2::new(i as i32 % w, i as i32 / w), cell))
|
|
}
|
|
|
|
pub fn iter_area(
|
|
&self,
|
|
pos: Vec2<i32>,
|
|
size: Vec2<i32>,
|
|
) -> impl Iterator<Item = (Vec2<i32>, &T)> + '_ {
|
|
(0..size.x).flat_map(move |x| {
|
|
(0..size.y).flat_map(move |y| {
|
|
Some((
|
|
pos + Vec2::new(x, y),
|
|
&self.cells[self.idx(pos + Vec2::new(x, y))?],
|
|
))
|
|
})
|
|
})
|
|
}
|
|
|
|
pub fn raw(&self) -> &[T] { &self.cells }
|
|
}
|
|
|
|
impl<T> Index<Vec2<i32>> for Grid<T> {
|
|
type Output = T;
|
|
|
|
fn index(&self, index: Vec2<i32>) -> &Self::Output {
|
|
self.get(index).unwrap_or_else(|| {
|
|
panic!(
|
|
"Attempted to index grid of size {:?} with index {:?}",
|
|
self.size(),
|
|
index
|
|
)
|
|
})
|
|
}
|
|
}
|
|
|
|
impl<T> IndexMut<Vec2<i32>> for Grid<T> {
|
|
fn index_mut(&mut self, index: Vec2<i32>) -> &mut Self::Output {
|
|
let size = self.size();
|
|
self.get_mut(index).unwrap_or_else(|| {
|
|
panic!(
|
|
"Attempted to index grid of size {:?} with index {:?}",
|
|
size, index
|
|
)
|
|
})
|
|
}
|
|
}
|