jagua_rs/collision_detection/quadtree/
qt_hazard_vec.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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
use crate::collision_detection::hazard::HazardEntity;
use crate::collision_detection::hazard_helpers::HazardIgnorer;
use crate::collision_detection::quadtree::qt_hazard::QTHazPresence;
use crate::collision_detection::quadtree::qt_hazard::QTHazard;
use std::cmp::Ordering;

/// Vector of `QTHazard`s, which always remains sorted by activeness then presence.
/// <br>
/// This is a performance optimization to be able to quickly return the "strongest" hazard
/// Strongest meaning the first active hazard with the highest presence (`Entire` > `Partial` > `None`)
#[derive(Clone, Debug)]
pub struct QTHazardVec {
    hazards: Vec<QTHazard>,
    n_active: usize,
}

impl QTHazardVec {
    pub fn new() -> Self {
        QTHazardVec {
            hazards: Vec::new(),
            n_active: 0,
        }
    }

    pub fn add(&mut self, haz: QTHazard) {
        debug_assert!(
            self.hazards
                .iter()
                .filter(|other| other.entity == haz.entity
                    && matches!(haz.entity, HazardEntity::PlacedItem { .. }))
                .count()
                == 0,
            "More than one hazard from same item entity in the vector! (This should never happen!)"
        );
        match self
            .hazards
            .binary_search_by(|probe| order_by_descending_strength(probe, &haz))
        {
            Ok(pos) | Err(pos) => {
                self.n_active += haz.active as usize;
                self.hazards.insert(pos, haz);
            }
        }
    }

    pub fn remove(&mut self, haz_entity: HazardEntity) -> Option<QTHazard> {
        debug_assert!(
            self.hazards
                .iter()
                .filter(|ch| ch.entity == haz_entity
                    && matches!(ch.entity, HazardEntity::PlacedItem { .. }))
                .count()
                <= 1,
            "More than one hazard from same item entity in the vector! (This should never happen!)"
        );
        let pos = self.hazards.iter().position(|ch| ch.entity == haz_entity);
        match pos {
            Some(pos) => {
                let haz = self.hazards.remove(pos);
                self.n_active -= haz.active as usize;
                Some(haz)
            }
            None => None,
        }
    }

    #[inline(always)]
    /// Returns the strongest hazard (if any), meaning the first active hazard with the highest [QTHazPresence] (`Entire` > `Partial` > `None`)
    /// Ignores any hazard present in `irrelevant_hazards`.
    pub fn strongest(&self, h_i: &impl HazardIgnorer) -> Option<&QTHazard> {
        debug_assert!(
            self.hazards.iter().filter(|hz| hz.active).count() == self.n_active,
            "Active hazards count is not correct!"
        );
        debug_assert!(
            self.hazards
                .windows(2)
                .all(|w| order_by_descending_strength(&w[0], &w[1]) != Ordering::Greater),
            "Hazards are not sorted correctly!"
        );
        self.hazards[0..self.n_active]
            .iter()
            .find(|hz| !h_i.is_irrelevant(&hz.entity))
    }

    pub fn get(&self, entity: HazardEntity) -> Option<&QTHazard> {
        self.hazards
            .iter()
            .filter(|hz| hz.active)
            .find(|hz| hz.entity == entity)
    }

    pub fn activate_hazard(&mut self, entity: HazardEntity) -> bool {
        match self.hazards.iter_mut().position(|hz| hz.entity == entity) {
            Some(index) => {
                let mut hazard = self.hazards.remove(index);
                debug_assert!(!hazard.active);
                hazard.active = true;
                self.add(hazard);
                true
            }
            None => false,
        }
    }

    pub fn deactivate_hazard(&mut self, entity: HazardEntity) -> bool {
        match self.hazards.iter_mut().position(|hz| hz.entity == entity) {
            Some(index) => {
                let mut hazard = self.hazards.remove(index);
                debug_assert!(hazard.active);
                hazard.active = false;
                self.n_active -= 1;
                self.add(hazard);
                true
            }
            None => false,
        }
    }

    pub fn active_hazards(&self) -> &[QTHazard] {
        &self.hazards[0..self.n_active]
    }

    pub fn all_hazards(&self) -> &[QTHazard] {
        &self.hazards
    }

    pub fn is_empty(&self) -> bool {
        self.hazards.is_empty()
    }

    pub fn len(&self) -> usize {
        self.hazards.len()
    }

    pub fn has_only_entire_hazards(&self) -> bool {
        self.hazards
            .iter()
            .all(|hz| matches!(hz.presence, QTHazPresence::Entire))
    }
}

fn order_by_descending_strength(qth1: &QTHazard, qth2: &QTHazard) -> Ordering {
    //sort in descending order of active (true > false) then by presence (Entire > Partial > None)
    qth1.active
        .cmp(&qth2.active)
        .then({
            let pres1: u8 = (&qth1.presence).into();
            let pres2: u8 = (&qth2.presence).into();
            pres1.cmp(&pres2)
        })
        .reverse()
}