blackout/src/pathfinding.rs

274 lines
8.9 KiB
Rust
Raw Normal View History

2022-05-06 16:07:59 +00:00
use std::marker::PhantomData;
2021-05-13 17:25:45 +00:00
use bevy::{
2022-01-21 00:06:20 +00:00
ecs::entity::Entities,
prelude::*,
tasks::{prelude::*, Task},
};
use bevy_rapier2d::{
2022-05-06 16:07:59 +00:00
na::{Isometry2, Vector2},
prelude::*,
2022-07-12 17:43:43 +00:00
rapier::prelude::{ColliderHandle, ColliderSet, QueryPipeline, RigidBodySet},
};
use futures_lite::future;
2021-05-13 17:25:45 +00:00
use pathfinding::prelude::*;
use crate::{
commands::RunIfExistsExt,
2022-05-10 18:56:49 +00:00
core::PointLike,
map::{Map, MapObstruction},
2021-06-02 22:02:38 +00:00
navigation::{RotationSpeed, Speed},
2021-05-13 17:25:45 +00:00
};
2022-01-10 19:50:52 +00:00
#[derive(Component, Debug, Deref, DerefMut)]
struct Calculating(Task<Option<Path>>);
2022-01-10 19:50:52 +00:00
#[derive(Component, Clone, Copy, Debug, Default, Deref, DerefMut, Eq, Hash, PartialEq, Reflect)]
2021-05-13 17:25:45 +00:00
#[reflect(Component)]
pub struct Destination(pub (i32, i32));
impl_pointlike_for_tuple_component!(Destination);
impl_pointlike_for_tuple_component!(&Destination);
2021-05-13 17:25:45 +00:00
2022-01-10 19:50:52 +00:00
#[derive(Component, Clone, Debug, Default, Reflect)]
2021-07-27 16:17:06 +00:00
#[reflect(Component)]
pub struct NoPath;
2022-01-10 19:50:52 +00:00
#[derive(Component, Clone, Debug, Default, Deref, DerefMut, Reflect)]
2021-05-13 17:25:45 +00:00
#[reflect(Component)]
pub struct Path(pub Vec<(i32, i32)>);
2022-03-15 15:37:28 +00:00
pub fn find_path<D: 'static + Clone + Default + Send + Sync>(
2021-05-13 17:25:45 +00:00
start: &dyn PointLike,
destination: &dyn PointLike,
2022-03-15 15:37:28 +00:00
map: &Map<D>,
2021-05-13 17:25:45 +00:00
) -> Option<(Vec<(i32, i32)>, u32)> {
astar(
&start.into(),
|p| {
let mut successors: Vec<((i32, i32), u32)> = vec![];
2022-08-27 14:11:37 +00:00
if let Some(tile) = map.at(p.0 as usize, p.1 as usize) {
if tile.is_walkable() {
for tile in map.get_available_exits(p.0 as usize, p.1 as usize) {
successors.push(((tile.0 as i32, tile.1 as i32), (tile.2 * 100.) as u32));
}
2021-09-16 19:26:45 +00:00
}
2021-05-13 17:25:45 +00:00
}
successors
},
|p| (p.distance_squared(destination) * 100.) as u32,
|p| *p == destination.into(),
)
}
2022-03-15 15:37:28 +00:00
fn find_path_for_shape<D: 'static + Clone + Default + Send + Sync>(
2022-05-06 16:07:59 +00:00
initiator: ColliderHandle,
2022-05-10 18:56:49 +00:00
start: Transform,
destination: Destination,
2022-03-15 15:37:28 +00:00
map: Map<D>,
2022-05-06 16:07:59 +00:00
query_pipeline: QueryPipeline,
collider_set: ColliderSet,
2022-07-12 17:43:43 +00:00
rigid_body_set: RigidBodySet,
2022-05-06 16:07:59 +00:00
shape: Collider,
) -> Option<Path> {
let path = astar(
&start.i32(),
|p| {
let mut successors: Vec<((i32, i32), u32)> = vec![];
2022-08-27 14:11:37 +00:00
if let Some(tile) = map.at(p.0 as usize, p.1 as usize) {
if tile.is_walkable() {
for tile in map.get_available_exits(p.0 as usize, p.1 as usize) {
let mut should_push = true;
let shape_pos = Isometry2::new(
Vector2::new(tile.0 as f32 + 0.5, tile.1 as f32 + 0.5),
0.,
);
query_pipeline.intersections_with_shape(
&rigid_body_set,
&collider_set,
&shape_pos,
&*shape.raw,
bevy_rapier2d::rapier::pipeline::QueryFilter::new()
.predicate(&|h, _c| h != initiator),
|_handle| {
should_push = false;
false
},
);
if should_push {
successors
.push(((tile.0 as i32, tile.1 as i32), (tile.2 * 100.) as u32));
}
2021-09-16 19:26:45 +00:00
}
}
}
successors
},
|p| (p.distance_squared(&destination) * 100.) as u32,
|p| *p == destination.i32(),
);
if let Some(path) = path {
Some(Path(path.0))
} else {
None
}
}
2022-03-15 15:37:28 +00:00
fn calculate_path<D: 'static + Clone + Default + Send + Sync>(
2021-05-13 17:25:45 +00:00
mut commands: Commands,
2022-05-06 16:07:59 +00:00
rapier_context: Res<RapierContext>,
obstructions: Query<&RapierColliderHandle, With<MapObstruction>>,
2022-01-10 19:50:52 +00:00
query: Query<
2022-05-06 16:07:59 +00:00
(
Entity,
&RapierColliderHandle,
&Destination,
2022-05-10 18:56:49 +00:00
&Transform,
2022-05-06 16:07:59 +00:00
&Collider,
),
2022-01-10 19:50:52 +00:00
Changed<Destination>,
>,
2022-03-15 15:37:28 +00:00
map: Query<&Map<D>>,
2021-05-13 17:25:45 +00:00
) {
2022-05-06 16:07:59 +00:00
for (entity, handle, destination, coordinates, shape) in query.iter() {
2021-07-21 20:20:02 +00:00
if coordinates.i32() == **destination {
2021-07-27 16:17:06 +00:00
commands
.entity(entity)
.remove::<Path>()
2021-07-28 01:17:36 +00:00
.remove::<NoPath>()
.remove::<Calculating>()
2021-09-16 19:26:45 +00:00
.remove::<Destination>()
.remove::<Speed>();
2021-07-21 20:20:02 +00:00
continue;
}
2022-01-26 20:56:52 +00:00
if let Ok(map) = map.get_single() {
let coordinates_clone = *coordinates;
let destination_clone = *destination;
2022-05-06 16:07:59 +00:00
let query_pipeline = rapier_context.query_pipeline.clone();
2022-01-26 20:56:52 +00:00
let map_clone = map.clone();
2022-05-06 16:07:59 +00:00
let handle_clone = *handle;
let mut collider_set = rapier_context.colliders.clone();
let mut to_remove = vec![];
for handle in collider_set.iter() {
2022-05-07 09:53:21 +00:00
if !obstructions.iter().map(|v| v.0).any(|x| x == handle.0) {
to_remove.push(handle.0);
}
}
2022-07-12 17:43:43 +00:00
let mut bodies = rapier_context.bodies.clone();
if !to_remove.is_empty() {
let mut islands = rapier_context.islands.clone();
for handle in to_remove {
collider_set.remove(handle, &mut islands, &mut bodies, false);
}
}
2022-01-26 20:56:52 +00:00
let shape_clone = (*shape).clone();
let pool = AsyncComputeTaskPool::get();
2022-01-26 20:56:52 +00:00
let task = pool.spawn(async move {
find_path_for_shape(
2022-05-06 16:07:59 +00:00
handle_clone.0,
2022-01-26 20:56:52 +00:00
coordinates_clone,
destination_clone,
map_clone,
2022-05-06 16:07:59 +00:00
query_pipeline,
2022-01-26 20:56:52 +00:00
collider_set,
2022-07-12 17:43:43 +00:00
bodies,
2022-01-26 20:56:52 +00:00
shape_clone,
)
});
commands.run_if_exists(entity, |mut entity| {
entity.insert(Calculating(task));
entity.remove::<Path>();
entity.remove::<NoPath>();
});
2022-01-26 20:56:52 +00:00
}
}
}
2021-07-28 01:26:16 +00:00
fn poll_tasks(mut commands: Commands, mut query: Query<(Entity, &mut Calculating)>) {
for (entity, mut calculating) in query.iter_mut() {
2021-07-28 01:26:16 +00:00
if let Some(result) = future::block_on(future::poll_once(&mut **calculating)) {
if let Some(path) = result {
commands.entity(entity).insert(path);
} else {
commands.entity(entity).insert(NoPath);
}
commands.entity(entity).remove::<Calculating>();
2021-05-13 17:25:45 +00:00
}
}
}
2022-01-21 00:06:20 +00:00
fn remove_destination(
mut commands: Commands,
entities: &Entities,
removed: RemovedComponents<Destination>,
) {
2021-07-28 01:26:16 +00:00
for entity in removed.iter() {
2022-01-21 00:06:20 +00:00
if entities.contains(entity) {
commands.entity(entity).remove::<Calculating>();
}
2021-07-28 01:26:16 +00:00
}
}
2021-06-15 22:48:49 +00:00
fn negotiate_path(
2021-05-13 17:25:45 +00:00
mut commands: Commands,
mut query: Query<(
Entity,
&mut Path,
2022-05-06 16:07:59 +00:00
&mut Transform,
&mut Velocity,
2021-05-13 17:25:45 +00:00
&Speed,
Option<&RotationSpeed>,
)>,
) {
2022-05-06 16:07:59 +00:00
for (entity, mut path, mut transform, mut velocity, speed, rotation_speed) in query.iter_mut() {
let start_i32 = (transform.translation.x, transform.translation.y).i32();
2021-09-17 15:02:33 +00:00
let mut new_path = path
.iter()
.cloned()
.skip_while(|v| *v == start_i32)
.collect::<Vec<(i32, i32)>>();
new_path.retain(|v| *v != start_i32);
2021-06-15 22:48:49 +00:00
**path = new_path;
2021-09-17 15:02:33 +00:00
if let Some(next) = path.first() {
2022-05-06 16:07:59 +00:00
let start = Vec2::new(transform.translation.x, transform.translation.y);
let next = Vec2::new(next.0 as f32 + 0.5, next.1 as f32 + 0.5);
2021-09-17 15:02:33 +00:00
let mut direction = next - start;
direction = direction.normalize();
direction *= **speed;
2022-05-06 16:07:59 +00:00
velocity.linvel = direction;
2021-06-15 22:48:49 +00:00
if rotation_speed.is_some() {
let v = next - start;
let angle = v.y.atan2(v.x);
2022-05-06 16:07:59 +00:00
transform.rotation = Quat::from_rotation_z(angle);
2021-05-13 17:25:45 +00:00
}
2021-09-17 15:02:33 +00:00
continue;
2021-06-15 22:48:49 +00:00
} else {
2021-07-27 16:17:06 +00:00
commands
.entity(entity)
.remove::<Path>()
2021-09-16 19:26:45 +00:00
.remove::<NoPath>()
.remove::<Destination>()
.remove::<Speed>();
2022-05-06 16:07:59 +00:00
velocity.linvel = Vec2::ZERO;
2021-05-13 17:25:45 +00:00
}
}
2021-06-15 22:48:49 +00:00
}
2021-05-13 17:25:45 +00:00
2022-03-15 15:37:28 +00:00
pub struct PathfindingPlugin<D: 'static + Clone + Default + Send + Sync>(PhantomData<D>);
2021-05-13 17:25:45 +00:00
2022-03-15 15:37:28 +00:00
impl<D: 'static + Clone + Default + Send + Sync> Default for PathfindingPlugin<D> {
fn default() -> Self {
Self(Default::default())
}
}
impl<D: 'static + Clone + Default + Send + Sync> Plugin for PathfindingPlugin<D> {
2022-01-10 19:50:52 +00:00
fn build(&self, app: &mut App) {
2022-03-15 15:37:28 +00:00
app.add_system(calculate_path::<D>)
2022-01-10 19:50:52 +00:00
.add_system_to_stage(CoreStage::PostUpdate, remove_destination)
.add_system(poll_tasks)
2022-08-27 14:11:37 +00:00
.add_system(negotiate_path.after(crate::navigation::limit_speed));
2021-05-13 17:25:45 +00:00
}
}