New joint rect algorithm

This commit is contained in:
daa84 2017-04-21 18:26:06 +03:00
parent 7279047f53
commit 73e1980256

View File

@ -230,7 +230,40 @@ impl UiModel {
} }
} }
#[derive(Clone)] pub struct ModelRectVec {
list: Vec<ModelRect>,
}
impl ModelRectVec {
pub fn new(first: ModelRect) -> ModelRectVec {
ModelRectVec { list: vec![first] }
}
fn find_neighbor(&self, neighbor: &ModelRect) -> Option<usize> {
for (i, rect) in self.list.iter().enumerate() {
if (rect.top == neighbor.top - 1 || rect.bot == neighbor.bot + 1) &&
neighbor.in_horizontal(rect) {
return Some(i);
} else if (rect.left == neighbor.left - 1 || rect.right == neighbor.right + 1) &&
neighbor.in_vertical(rect) {
return Some(i);
} else if rect == neighbor {
return Some(i);
}
}
None
}
pub fn join(&mut self, other: ModelRect) {
match self.find_neighbor(&other) {
Some(i) => self.list.get_mut(i).unwrap().join(&other),
None => self.list.push(other),
}
}
}
#[derive(Clone, PartialEq)]
pub struct ModelRect { pub struct ModelRect {
pub top: usize, pub top: usize,
pub bot: usize, pub bot: usize,
@ -257,6 +290,18 @@ impl ModelRect {
} }
} }
#[inline]
fn in_horizontal(&self, other: &ModelRect) -> bool {
other.left >= self.left && other.left <= self.right ||
other.right >= self.left && other.right >= self.right
}
#[inline]
fn in_vertical(&self, other: &ModelRect) -> bool {
other.top >= self.top && other.top <= self.bot ||
other.bot >= self.top && other.bot <= self.bot
}
pub fn join(&mut self, rect: &ModelRect) { pub fn join(&mut self, rect: &ModelRect) {
self.top = if self.top < rect.top { self.top = if self.top < rect.top {
self.top self.top
@ -396,6 +441,20 @@ impl<'a> Iterator for ClipColIterator<'a> {
mod tests { mod tests {
use super::*; use super::*;
#[test]
fn test_model_vec_join() {
let mut list = ModelRectVec::new(ModelRect::point(5, 5));
let neighbor = ModelRect::point(6, 5);
let not_neighbor = ModelRect::point(6, 6);
list.join(neighbor);
assert_eq!(1, list.list.len());
list.join(not_neighbor);
assert_eq!(2, list.list.len());
}
#[test] #[test]
fn test_iterator_border() { fn test_iterator_border() {
let model = UiModel::new(10, 20); let model = UiModel::new(10, 20);