jagua_rs/collision_detection/
hazard_helpers.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
use crate::collision_detection::hazard::HazardEntity;
use crate::entities::placed_item::PItemKey;
use slotmap::SecondaryMap;

/// Trait for structs that can be used to filter out irrelevant hazards.
/// Basically only used in [`QTHazardVec::strongest()`](crate::collision_detection::quadtree::qt_hazard_vec::QTHazardVec::strongest).
pub trait HazardIgnorer {
    fn is_irrelevant(&self, haz: &HazardEntity) -> bool;
}

/// Trait for structs that can track and store already detected hazards.
/// Interface made to mimic a Vec of `HazardEntity`s.
pub trait HazardDetector: HazardIgnorer {
    fn contains(&self, haz: &HazardEntity) -> bool;

    fn push(&mut self, haz: HazardEntity);

    fn remove(&mut self, haz: &HazardEntity);

    fn is_empty(&self) -> bool {
        self.len() == 0
    }

    fn len(&self) -> usize;

    fn iter(&self) -> impl Iterator<Item = &HazardEntity>;
}

/// Datastructure to register which Hazards are detected during collision collection.
/// Hazards caused by placed items have instant lookups, the others are stored in a Vec.
/// It also stores an index for each hazard, which can be used to determine the order in which they were detected.
#[derive(Debug)]
pub struct DetectionMap {
    pi_hazards: SecondaryMap<PItemKey, (HazardEntity, usize)>,
    other: Vec<(HazardEntity, usize)>,
    idx_counter: usize,
}

impl DetectionMap {
    pub fn new() -> Self {
        DetectionMap {
            idx_counter: 0,
            pi_hazards: SecondaryMap::new(),
            other: Vec::new(),
        }
    }

    pub fn clear(&mut self) {
        self.idx_counter = 0;
        self.pi_hazards.clear();
        self.other.clear();
    }

    pub fn iter_with_index(&self) -> impl Iterator<Item = &(HazardEntity, usize)> {
        self.pi_hazards.values().chain(self.other.iter())
    }

    pub fn index_counter(&self) -> usize {
        self.idx_counter
    }
}

impl HazardDetector for DetectionMap {
    fn contains(&self, haz: &HazardEntity) -> bool {
        match haz {
            HazardEntity::PlacedItem { pk, .. } => self.pi_hazards.contains_key(*pk),
            _ => self.other.iter().find(|(h, _)| h == haz).is_some(),
        }
    }

    fn push(&mut self, haz: HazardEntity) {
        debug_assert!(!self.contains(&haz));
        match haz {
            HazardEntity::PlacedItem { pk, .. } => {
                self.pi_hazards.insert(pk, (haz, self.idx_counter));
            }
            _ => self.other.push((haz, self.idx_counter)),
        }
        self.idx_counter += 1;
    }

    fn remove(&mut self, haz: &HazardEntity) {
        match haz {
            HazardEntity::PlacedItem { pk, .. } => {
                self.pi_hazards.remove(*pk);
            }
            _ => self.other.retain(|(h, _)| h != haz),
        }
    }

    fn len(&self) -> usize {
        self.pi_hazards.len() + self.other.len()
    }

    fn iter(&self) -> impl Iterator<Item = &HazardEntity> {
        self.pi_hazards
            .iter()
            .map(|(_, (h, _))| h)
            .chain(self.other.iter().map(|(h, _)| h))
    }
}

impl HazardIgnorer for DetectionMap {
    fn is_irrelevant(&self, haz: &HazardEntity) -> bool {
        self.contains(haz)
    }
}

impl HazardIgnorer for &[HazardEntity] {
    fn is_irrelevant(&self, haz: &HazardEntity) -> bool {
        self.contains(&haz)
    }
}