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
use crate::components::{transform::*, volume::*, volume_overlap::*};
use core::{
    app::AppLifeCycle,
    ecs::{
        components::{Events, Tag},
        Comp, Entity, Universe, WorldRef,
    },
};
use std::collections::HashSet;

pub enum HaVolumeOverlapEvent {
    Begin(Entity),
    End(Entity),
}

#[derive(Debug, Default)]
pub struct HaVolumeOverlapSystemCache {
    active_overlaps: HashSet<(Entity, Entity)>,
}

pub type HaVolumeOverlapSystemResources<'a> = (
    WorldRef,
    &'a AppLifeCycle,
    &'a mut HaVolumeOverlapSystemCache,
    Comp<&'a Tag>,
    Comp<&'a HaTransform>,
    Comp<&'a HaVolume>,
    Comp<&'a mut HaVolumeOverlap>,
    Comp<&'a mut Events<HaVolumeOverlapEvent>>,
);

pub fn ha_volume_overlap_system(universe: &mut Universe) {
    let (world, lifecycle, mut cache, ..) =
        universe.query_resources::<HaVolumeOverlapSystemResources>();

    let dt = lifecycle.delta_time_seconds();

    for (entity_a, (transform_a, volume_a, overlap, events)) in world
        .query::<(
            &HaTransform,
            Option<&HaVolume>,
            &mut HaVolumeOverlap,
            &mut Events<HaVolumeOverlapEvent>,
        )>()
        .iter()
    {
        overlap.time += dt;
        if overlap.time < overlap.delay {
            continue;
        }
        overlap.time = 0.0;

        for (entity_b, (tag, transform_b, volume_b)) in
            world.query::<(&Tag, &HaTransform, &HaVolume)>().iter()
        {
            if entity_a != entity_b && overlap.filters.validate_tag(tag.0.as_ref()) {
                let overlaps = if let Some(volume_a) = volume_a {
                    HaVolume::world_space_overlaps(
                        volume_a,
                        &transform_a.world_matrix(),
                        volume_b,
                        &transform_b.world_matrix(),
                    )
                    .is_some()
                } else {
                    volume_b
                        .world_space_contains(
                            &transform_b.world_matrix(),
                            transform_a.get_translation(),
                        )
                        .is_some()
                };
                let active = cache.active_overlaps.contains(&(entity_a, entity_b));
                match (overlaps, active) {
                    (true, false) => {
                        events.send(HaVolumeOverlapEvent::Begin(entity_b));
                        cache.active_overlaps.insert((entity_a, entity_b));
                    }
                    (false, true) => {
                        events.send(HaVolumeOverlapEvent::End(entity_b));
                        cache.active_overlaps.remove(&(entity_a, entity_b));
                    }
                    _ => {}
                }
            }
        }
    }
}