jagua_rs/collision_detection/quadtree/qt_node.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 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329
use crate::collision_detection::hazard::HazardEntity;
use crate::collision_detection::hazard_helpers::HazardDetector;
use crate::collision_detection::quadtree::qt_hazard::QTHazPresence;
use crate::collision_detection::quadtree::qt_hazard::QTHazard;
use crate::collision_detection::quadtree::qt_hazard_vec::QTHazardVec;
use crate::collision_detection::quadtree::qt_traits::QTQueryable;
use crate::geometry::geo_traits::CollidesWith;
use crate::geometry::primitives::aa_rectangle::AARectangle;
use tribool::Tribool;
/// A node in the quadtree
#[derive(Clone, Debug)]
pub struct QTNode {
/// The level of the node in the tree, 0 being the bottom-most level
pub level: u8,
/// The bounding box of the node
pub bbox: AARectangle,
/// The children of the node, if any
pub children: Option<Box<[QTNode; 4]>>,
/// The hazards present in the node
pub hazards: QTHazardVec,
}
impl QTNode {
pub fn new(level: u8, bbox: AARectangle) -> Self {
QTNode {
level,
bbox,
children: None,
hazards: QTHazardVec::new(),
}
}
pub fn register_hazard(&mut self, hazard: QTHazard) {
fn register_to_children(children: &mut Option<Box<[QTNode; 4]>>, hazard: &QTHazard) {
if let Some(children) = children.as_mut() {
let child_bboxes = [0, 1, 2, 3].map(|i| &children[i].bbox);
let c_hazards = hazard.constrict(child_bboxes);
for (i, c_hazard) in c_hazards.into_iter().enumerate() {
if let Some(c_hazard) = c_hazard {
children[i].register_hazard(c_hazard);
}
}
}
}
//If the hazard is of the partial type, and we are not at the max tree depth: generate children
if !self.has_children()
&& self.level > 0
&& matches!(hazard.presence, QTHazPresence::Partial(_))
{
self.generate_children();
//register all existing hazards to the newly created children
for hazard in self.hazards.all_hazards() {
register_to_children(&mut self.children, hazard);
}
}
register_to_children(&mut self.children, &hazard);
self.hazards.add(hazard);
}
pub fn deregister_hazard(&mut self, hazard_entity: HazardEntity) {
let removed_ch = self.hazards.remove(hazard_entity);
if removed_ch.is_some() && self.has_children() {
if self.hazards.is_empty() || self.hazards.has_only_entire_hazards() {
//If there are no hazards, or only entire hazards, drop the children
self.children = None;
} else {
//Otherwise, recursively deregister the entity from the children
self.children
.as_mut()
.unwrap()
.iter_mut()
.for_each(|child| child.deregister_hazard(hazard_entity));
}
}
}
pub fn activate_hazard(&mut self, entity: HazardEntity) {
let modified = self.hazards.activate_hazard(entity);
if modified {
if let Some(children) = &mut self.children {
children.iter_mut().for_each(|c| c.activate_hazard(entity))
}
}
}
pub fn deactivate_hazard(&mut self, entity: HazardEntity) {
let modified = self.hazards.deactivate_hazard(entity);
if modified {
if let Some(children) = &mut self.children {
children
.iter_mut()
.for_each(|c| c.deactivate_hazard(entity))
}
}
}
fn generate_children(&mut self) {
if self.level > 0 {
let quadrants = self.bbox.quadrants();
let children = quadrants.map(|q| QTNode::new(self.level - 1, q));
self.children = Some(Box::new(children));
}
}
pub fn get_number_of_children(&self) -> usize {
match &self.children {
Some(children) => {
4 + children
.iter()
.map(|x| x.get_number_of_children())
.sum::<usize>()
}
None => 0,
}
}
pub fn has_children(&self) -> bool {
self.children.is_some()
}
/// Used to detect collisions in a binary fashion: either there is a collision or there isn't.
/// Returns `None` if no collision between the entity and any hazard is detected,
/// otherwise the first encountered hazard that collides with the entity is returned.
pub fn collides<T>(
&self,
entity: &T,
irrelevant_hazards: &[HazardEntity],
) -> Option<&HazardEntity>
where
T: QTQueryable,
{
match self.hazards.strongest(&irrelevant_hazards) {
None => None,
Some(strongest_hazard) => match entity.collides_with(&self.bbox) {
false => None,
true => match strongest_hazard.presence {
QTHazPresence::None => None,
QTHazPresence::Entire => Some(&strongest_hazard.entity),
QTHazPresence::Partial(_) => match &self.children {
Some(children) => {
//Check if any of the children intersect with the entity
children
.iter()
.map(|child| child.collides(entity, irrelevant_hazards))
.find(|x| x.is_some())
.flatten()
}
None => {
//Check if any of the partially present (and active) hazards collide with the entity
let mut relevant_hazards = self
.hazards
.active_hazards()
.iter()
.filter(|hz| !irrelevant_hazards.contains(&hz.entity));
relevant_hazards
.find(|hz| match &hz.presence {
QTHazPresence::None => false,
QTHazPresence::Entire => {
unreachable!("should have been handled above")
}
QTHazPresence::Partial(p_haz) => p_haz.collides_with(entity),
})
.map(|hz| &hz.entity)
}
},
},
},
}
}
/// Gathers all hazards that collide with the entity and reports them to the `detector`.
/// All hazards already present in the `detector` are ignored.
pub fn collect_collisions<T>(&self, entity: &T, detector: &mut impl HazardDetector)
where
T: QTQueryable,
{
//TODO: This seems to be the fastest version of this function
// Check if the other collision functions can also be improved.
match entity.collides_with(&self.bbox) {
false => return, //Entity does not collide with the node
true => match self.hazards.strongest(detector) {
None => return, //Any hazards present are not relevant
Some(_) => match self.children.as_ref() {
Some(children) => {
//Do not perform any CD on this level, check the children
children.iter().for_each(|child| {
child.collect_collisions(entity, detector);
})
}
None => {
//No children, detect all Entire hazards and check the Partial ones
for hz in self.hazards.active_hazards().iter() {
match &hz.presence {
QTHazPresence::None => (),
QTHazPresence::Entire => {
if !detector.contains(&hz.entity) {
detector.push(hz.entity)
}
}
QTHazPresence::Partial(p_haz) => {
if !detector.contains(&hz.entity) && p_haz.collides_with(entity)
{
detector.push(hz.entity);
}
}
}
}
}
},
},
}
}
/// Used to detect collisions in a broad fashion:
/// Returns `Tribool::True` if the entity definitely collides with a hazard,
/// `Tribool::False` if the entity definitely does not collide with any hazard,
/// and `Tribool::Indeterminate` if it is not possible to determine whether the entity collides with any hazard.
pub fn definitely_collides<T>(&self, entity: &T, irrelevant_hazards: &[HazardEntity]) -> Tribool
where
T: CollidesWith<AARectangle>,
{
match self.hazards.strongest(&irrelevant_hazards) {
None => Tribool::False,
Some(hazard) => match (entity.collides_with(&self.bbox), &hazard.presence) {
(false, _) | (_, QTHazPresence::None) => Tribool::False,
(true, QTHazPresence::Entire) => Tribool::True,
(true, QTHazPresence::Partial(_)) => match &self.children {
Some(children) => {
//There is a partial hazard and the node has children, check all children
let mut result = Tribool::False; //Assume no collision
for i in 0..4 {
let child = &children[i];
match child.definitely_collides(entity, irrelevant_hazards) {
Tribool::True => return Tribool::True,
Tribool::Indeterminate => result = Tribool::Indeterminate,
Tribool::False => {}
}
}
result
}
None => Tribool::Indeterminate,
},
},
}
}
/// Used to detect collisions with a single hazard in a broad fashion:
/// Returns `Tribool::True` if the entity definitely collides with a hazard,
/// `Tribool::False` if the entity definitely does not collide with any hazard,
/// and `Tribool::Indeterminate` if it is not possible to determine whether the entity collides with any hazard.
pub fn definitely_collides_with<T>(&self, entity: &T, hazard_entity: HazardEntity) -> Tribool
where
T: CollidesWith<AARectangle>,
{
match self.hazards.get(hazard_entity) {
None => Tribool::False, //Node does not contain entity
Some(hazard) => match entity.collides_with(&self.bbox) {
false => Tribool::False, //Hazard present, but the point is fully outside the node
true => match hazard.presence {
QTHazPresence::None => Tribool::False, //The hazard is of type None, a collision is impossible
QTHazPresence::Entire => Tribool::True, //The hazard is of type Entire, a collision is guaranteed
QTHazPresence::Partial(_) => match &self.children {
Some(children) => {
//There is a partial hazard and the node has children, check all children
let mut result = Tribool::False; //Assume no collision
for i in 0..4 {
let child = &children[i];
match child.definitely_collides_with(entity, hazard_entity) {
Tribool::True => return Tribool::True, //If a child for sure collides, we can immediately return Yes
Tribool::Indeterminate => result = Tribool::Indeterminate, //If a child might collide, switch from to Maybe
Tribool::False => {} //If child does not collide, do nothing
}
}
result
}
None => Tribool::Indeterminate, //There are no children, so we can't be sure
},
},
},
}
}
/// Used to gather all hazards that within a given bounding box.
/// May overestimate the hazards that are present in the bounding box, since it is limited
/// by the resolution of the quadtree.
pub fn collect_potential_hazards_within(
&self,
bbox: &AARectangle,
detector: &mut impl HazardDetector,
) {
match bbox.collides_with(&self.bbox) {
false => return, //Entity does not collide with the node
true => match self.children.as_ref() {
Some(children) => {
//Do not perform any CD on this level, check the children
children.iter().for_each(|child| {
child.collect_potential_hazards_within(bbox, detector);
})
}
None => {
//No children, detect all Entire hazards and check the Partial ones
for hz in self.hazards.active_hazards().iter() {
match &hz.presence {
QTHazPresence::None => (),
QTHazPresence::Entire => {
if !detector.contains(&hz.entity) {
detector.push(hz.entity)
}
}
QTHazPresence::Partial(_) => {
// If the hazards is partially present, add it.
// We are limited by the resolution of the quadtree
if !detector.contains(&hz.entity) {
detector.push(hz.entity);
}
}
}
}
}
},
}
}
}