outfly/src/world.rs

524 lines
19 KiB
Rust
Raw Normal View History

use crate::{actor, audio, hud, nature, var};
2024-03-16 20:44:51 +00:00
use bevy::prelude::*;
2024-03-31 22:47:03 +00:00
use bevy::render::render_resource::{AsBindGroup, ShaderRef};
2024-04-02 03:12:53 +00:00
use bevy::math::{DVec3, I64Vec3};
use bevy::scene::{InstanceId, SceneInstance};
use bevy_xpbd_3d::prelude::*;
use bevy_xpbd_3d::plugins::sync::SyncConfig;
2024-04-02 03:12:53 +00:00
use std::collections::HashMap;
use std::f32::consts::PI;
use fastrand;
2024-03-16 20:44:51 +00:00
const ASTEROID_UPDATE_INTERVAL: f32 = 0.1; // seconds
const ASTEROID_SIZE_FACTOR: f32 = 10.0;
const RING_THICKNESS: f64 = 8.0e6;
const STARS_MAX_MAGNITUDE: f32 = 5.5; // max 7.0, see generate_starchart.py
2024-03-18 03:39:26 +00:00
2024-04-01 17:20:31 +00:00
const CENTER_WORLD_ON_PLAYER: bool = true;
2024-04-02 03:12:53 +00:00
const ASTEROID_SPAWN_STEP: f64 = 500.0;
2024-04-02 03:41:12 +00:00
const ASTEROID_VIEW_RADIUS: f64 = 3000.0;
2024-04-01 17:20:31 +00:00
2024-03-19 20:09:20 +00:00
const ASSET_ASTEROID1: &str = "models/asteroid.glb#Scene0";
const ASSET_ASTEROID2: &str = "models/asteroid2.glb#Scene0";
pub fn asset_name_to_path(name: &str) -> &'static str {
match name {
"suit" => "models/suit.glb#Scene0",
"suit_ar_chefhat" => "models/suit_ar_chefhat.glb#Scene0",
2024-03-20 20:03:22 +00:00
"asteroid1" => ASSET_ASTEROID1,
"asteroid2" => ASSET_ASTEROID2,
2024-03-21 02:15:00 +00:00
"moonlet" => "models/moonlet.glb#Scene0",
2024-03-29 13:19:42 +00:00
"monolith" => "models/monolith_neon.glb#Scene0",
"lightorb" => "models/lightorb.glb#Scene0",
2024-04-10 23:12:07 +00:00
"orb_busstop" => "models/orb_busstop.glb#Scene0",
"orb_busstop_dim" => "models/orb_busstop_dim.glb#Scene0",
2024-03-28 16:25:35 +00:00
"MeteorAceGT" => "models/MeteorAceGT.glb#Scene0",
2024-04-04 22:32:42 +00:00
"satellite" => "models/satellite.glb#Scene0",
2024-03-21 03:34:09 +00:00
"pizzeria" => "models/pizzeria2.glb#Scene0",
"pizzasign" => "models/pizzasign.glb#Scene0",
2024-04-05 20:16:01 +00:00
"selectagon" => "models/selectagon.glb#Scene0",
2024-04-10 15:37:06 +00:00
"clippy" => "models/clippy.glb#Scene0",
2024-04-10 19:03:30 +00:00
"clippy_ar" => "models/clippy_ar.glb#Scene0",
_ => "models/error.glb#Scene0",
}
}
2024-03-18 03:39:26 +00:00
2024-03-17 23:04:23 +00:00
pub struct WorldPlugin;
impl Plugin for WorldPlugin {
fn build(&self, app: &mut App) {
app.add_systems(Startup, setup);
2024-04-01 03:25:35 +00:00
app.add_systems(Update, handle_cheats);
app.add_systems(PostUpdate, handle_despawn);
app.add_systems(Update, spawn_despawn_asteroids);
app.add_plugins(PhysicsPlugins::default());
app.add_plugins(MaterialPlugin::<RingMaterial>::default());
2024-03-30 14:37:51 +00:00
//app.add_plugins(PhysicsDebugPlugin::default());
app.insert_resource(Gravity(DVec3::splat(0.0)));
app.insert_resource(AsteroidUpdateTimer(
Timer::from_seconds(ASTEROID_UPDATE_INTERVAL, TimerMode::Repeating)));
app.insert_resource(AsteroidDatabase(Vec::new()));
2024-04-02 03:12:53 +00:00
app.insert_resource(ActiveAsteroids(HashMap::new()));
app.add_event::<DespawnEvent>();
2024-04-01 17:20:31 +00:00
if CENTER_WORLD_ON_PLAYER {
// Disable bevy_xpbd's position->transform sync function
app.insert_resource(SyncConfig {
position_to_transform: false,
transform_to_position: false,
});
// Add own position->transform sync function
app.add_systems(PreUpdate, position_to_transform);
}
2024-03-17 23:04:23 +00:00
}
}
#[derive(Resource)] struct AsteroidUpdateTimer(Timer);
#[derive(Resource)] struct AsteroidDatabase(Vec<AsteroidData>);
2024-04-02 03:12:53 +00:00
#[derive(Resource)] struct ActiveAsteroids(HashMap<I64Vec3, AsteroidData>);
2024-04-07 16:35:19 +00:00
#[derive(Resource)] struct AsteroidModel1(Handle<Scene>);
#[derive(Resource)] struct AsteroidModel2(Handle<Scene>);
#[derive(Component)] struct Asteroid;
2024-04-05 00:58:02 +00:00
#[derive(Component)] pub struct DespawnOnPlayerDeath;
struct AsteroidData {
2024-04-02 03:12:53 +00:00
entity: Entity,
//viewdistance: f64,
}
#[derive(Event)]
2024-04-02 03:12:53 +00:00
pub struct DespawnEvent {
entity: Entity,
sceneinstance: InstanceId,
2024-04-02 03:12:53 +00:00
origin: I64Vec3,
}
2024-03-31 22:47:03 +00:00
#[derive(Asset, TypePath, AsBindGroup, Debug, Clone)]
pub struct RingMaterial {
alpha_mode: AlphaMode,
#[uniform(0)]
ring_radius: f32,
#[uniform(1)]
jupiter_radius: f32,
}
2024-03-31 22:47:03 +00:00
impl Material for RingMaterial {
2024-03-31 22:47:03 +00:00
fn fragment_shader() -> ShaderRef {
"shaders/jupiters_rings.wgsl".into()
}
fn alpha_mode(&self) -> AlphaMode {
self.alpha_mode
2024-03-31 22:47:03 +00:00
}
}
#[derive(Component)]
pub struct Star;
2024-03-16 20:44:51 +00:00
pub fn setup(
mut commands: Commands,
2024-03-16 22:11:56 +00:00
mut meshes: ResMut<Assets<Mesh>>,
mut materials: ResMut<Assets<StandardMaterial>>,
mut materials_custom: ResMut<Assets<RingMaterial>>,
2024-04-07 16:35:19 +00:00
asset_server: Res<AssetServer>,
2024-03-16 20:44:51 +00:00
) {
2024-04-07 16:35:19 +00:00
// Load assets
commands.insert_resource(AsteroidModel1(asset_server.load(ASSET_ASTEROID1)));
commands.insert_resource(AsteroidModel2(asset_server.load(ASSET_ASTEROID2)));
// Generate starmap
let sphere_handle = meshes.add(Circle::new(1.0));
let mut starcount = 0;
for star in nature::STARS {
let mag = star.3;
if mag > STARS_MAX_MAGNITUDE {
continue;
}
2024-04-07 23:14:08 +00:00
let is_sun = mag < -20.0;
let mag = mag.min(6.0);
let scale_color = {|color: f32|
2024-04-07 23:13:31 +00:00
if is_sun {
color * 13.0f32
2024-03-21 01:11:07 +00:00
} else {
color * (0.0659663 * mag * mag - 1.09862 * mag + 4.3)
2024-03-21 01:11:07 +00:00
}
};
let scale_size = {|mag: f32|
2024-04-07 23:13:31 +00:00
if is_sun {
40000.0f32
2024-03-21 01:11:07 +00:00
} else {
1000.0 * (0.230299 * mag * mag - 3.09013 * mag + 15.1782)
} * 100.0
2024-03-21 01:11:07 +00:00
};
let (r, g, b) = nature::star_color_index_to_rgb(star.4);
let star_color_handle = materials.add(StandardMaterial {
base_color: Color::rgb(scale_color(r), scale_color(g), scale_color(b)),
2024-03-21 01:11:07 +00:00
unlit: true,
..default()
});
2024-04-07 23:13:31 +00:00
let mesh_distance = 1e9;
let starchart_distance = if is_sun {
Some(nature::DIST_JUPTER_SUN)
} else if star.5 >= 100000.0 {
None
2024-04-07 23:13:31 +00:00
} else {
Some(nature::PARSEC2METER * star.5 as f64)
2024-04-07 23:13:31 +00:00
};
let name = if star.6.is_empty() {
"Uncharted Star".to_string()
} else {
star.6.to_string()
};
let translation = Vec3::new(
mesh_distance * star.0,
mesh_distance * star.1,
mesh_distance * star.2,
);
let rotation = Quat::from_rotation_arc(Vec3::Z, (-translation).normalize());
commands.spawn((
Star,
2024-04-07 22:39:57 +00:00
hud::IsClickable {
name: Some(name),
distance: starchart_distance,
2024-04-07 22:39:57 +00:00
},
PbrBundle {
mesh: sphere_handle.clone(),
2024-04-07 15:59:40 +00:00
material: star_color_handle,
transform: Transform {
translation,
rotation,
scale: Vec3::splat(scale_size(mag)),
},
..default()
}
));
starcount += 1;
}
info!("Generated {starcount} stars");
2024-03-17 13:16:25 +00:00
2024-03-31 22:47:03 +00:00
// Add shaded ring
let ring_radius = 229_000_000.0;
let jupiter_radius = 71_492_000.0;
commands.spawn((
MaterialMeshBundle {
mesh: meshes.add(Mesh::from(Cylinder::new(ring_radius, 1.0))),
material: materials_custom.add(RingMaterial {
alpha_mode: AlphaMode::Blend,
ring_radius: ring_radius,
jupiter_radius: jupiter_radius,
}),
..default()
},
Position::from_xyz(0.0, 0.0, 0.0),
Rotation::from(Quat::IDENTITY),
//Rotation::from(Quat::from_rotation_x(-0.3f32.to_radians())),
));
2024-03-16 20:44:51 +00:00
}
2024-04-01 03:25:35 +00:00
fn spawn_despawn_asteroids(
time: Res<Time>,
mut timer: ResMut<AsteroidUpdateTimer>,
mut commands: Commands,
q_player: Query<&Position, With<actor::PlayerCamera>>,
2024-04-02 03:12:53 +00:00
mut ew_despawn: EventWriter<DespawnEvent>,
mut db: ResMut<ActiveAsteroids>,
2024-04-07 16:35:19 +00:00
asteroid1_handle: Res<AsteroidModel1>,
asteroid2_handle: Res<AsteroidModel2>,
mut q_asteroid: Query<(Entity, &SceneInstance), With<Asteroid>>,
mut last_player_cell: Local<I64Vec3>,
) {
if !timer.0.tick(time.delta()).just_finished() || q_player.is_empty() {
//if q_player.is_empty() {
return;
}
let player = q_player.get_single().unwrap();
2024-04-02 03:12:53 +00:00
let player_cell = I64Vec3::new(
(player.x / ASTEROID_SPAWN_STEP).round() as i64,
(player.y / ASTEROID_SPAWN_STEP).round() as i64,
(player.z / ASTEROID_SPAWN_STEP).round() as i64,
);
if *last_player_cell == player_cell {
return;
}
*last_player_cell = player_cell;
2024-04-02 03:12:53 +00:00
// Parameters
let view_radius: f64 = ASTEROID_VIEW_RADIUS;
let step: f64 = ASTEROID_SPAWN_STEP;
let stepmax: i64 = (view_radius / step) as i64;
// Despawn far asteroids
2024-04-02 03:12:53 +00:00
let x_min = player_cell.x - stepmax;
let x_max = player_cell.x + stepmax;
let y_min = player_cell.y - stepmax;
let y_max = player_cell.y + stepmax;
let z_min = player_cell.z - stepmax;
let z_max = player_cell.z + stepmax;
for (origin, asteroid) in db.0.iter() {
if origin.x < x_min || origin.x > x_max
|| origin.y < y_min || origin.y > y_max
|| origin.z < z_min || origin.z > z_max
{
let mut despawning_worked = false;
for (ent, sceneinstance) in &mut q_asteroid {
2024-04-02 03:12:53 +00:00
if ent == asteroid.entity {
ew_despawn.send(DespawnEvent {
entity: asteroid.entity,
sceneinstance: **sceneinstance,
2024-04-02 03:12:53 +00:00
origin: origin.clone(),
});
despawning_worked = true;
break;
}
}
2024-04-02 03:12:53 +00:00
if !despawning_worked {
error!("Couldn't despawn asteroid:");
dbg!(origin);
}
}
2024-04-02 03:12:53 +00:00
}
// Density based on the radius alone
let radius_plane = (player.x * player.x + player.z * player.z).sqrt();
let density_r = nature::ring_density((radius_plane / 1e6) as f32);
if density_r < 0.001 {
return;
}
// Density based on radius and the vertical distance to the ring
let normalized_distance = player.y / (RING_THICKNESS / 2.0);
let density = density_r * (-4.0 * normalized_distance.powf(2.0)).exp() as f32;
if density < 0.001 {
return;
}
let mut rng = fastrand::Rng::new();
2024-04-02 03:12:53 +00:00
for x in -stepmax..=stepmax {
for y in -stepmax..=stepmax {
for z in -stepmax..=stepmax {
let origin = I64Vec3::new(
player_cell.x + x as i64,
player_cell.y + y as i64,
player_cell.z + z as i64,
);
if db.0.contains_key(&origin) {
// already in db, nothing to do
continue;
}
// Get a seed based on all of the origin axes
// Probably there's a faster way
rng.seed(origin.x as u64);
let seed = rng.u64(0..u64::MAX).wrapping_add(origin.y as u64);
rng.seed(seed);
let seed = rng.u64(0..u64::MAX).wrapping_add(origin.z as u64);
rng.seed(seed);
let rand_s = rng.f32();
let size_raw: f32 = (rand_s + 1e-4).powf(-0.5) as f32; // -> between ~1 and 100
let size_density = size_raw * density;
2024-04-02 03:41:12 +00:00
if size_density < 0.3 {
continue;
}
if size_raw < 20.0 {
2024-04-02 03:41:12 +00:00
let dist = player_cell.as_dvec3().distance(origin.as_dvec3());
if dist > 6.0 {
2024-04-02 03:41:12 +00:00
continue;
}
}
let size: f32 = ASTEROID_SIZE_FACTOR * size_density;
2024-04-02 03:41:12 +00:00
let rand_x = rng.f64();
let rand_y = rng.f64();
let rand_z = rng.f64();
let rand_c = rng.bool();
let class = if rand_c { 0 } else { 1 };
2024-04-02 03:12:53 +00:00
//let max_viewdist = ASTEROID_VIEW_RADIUS / ASTEROID_SPAWN_STEP;
let wobble = ASTEROID_SPAWN_STEP * 0.5;
2024-04-02 03:12:53 +00:00
let pos = DVec3::new(
2024-04-02 03:41:12 +00:00
origin.x as f64 * ASTEROID_SPAWN_STEP + wobble * rand_x * 2.0 - 1.0,
origin.y as f64 * ASTEROID_SPAWN_STEP + wobble * rand_y * 2.0 - 1.0,
origin.z as f64 * ASTEROID_SPAWN_STEP + wobble * rand_z * 2.0 - 1.0,
2024-04-02 03:12:53 +00:00
);
// Spawn
let mut entity_commands = commands.spawn((
actor::Actor::default(),
RigidBody::Dynamic,
AngularVelocity(DVec3::new(0.1, 0.1, 0.03)),
LinearVelocity(DVec3::new(0.0, 0.0, 0.35)),
Collider::sphere(1.0),
Rotation::from(Quat::from_rotation_y(-PI / 3.)),
Position::new(pos),
Asteroid,
));
2024-04-07 16:35:19 +00:00
let model = match class {
0 => asteroid1_handle.0.clone(),
_ => asteroid2_handle.0.clone(),
2024-04-02 03:12:53 +00:00
};
entity_commands.insert(SceneBundle {
2024-04-07 16:35:19 +00:00
scene: model,
2024-04-02 03:12:53 +00:00
transform: Transform {
scale: Vec3::splat(size),
2024-04-02 03:12:53 +00:00
..default()
},
..default()
2024-04-02 03:12:53 +00:00
});
db.0.insert(origin, AsteroidData {
entity: entity_commands.id(),
//viewdistance: 99999999.0,
});
}
}
}
}
fn handle_despawn(
mut commands: Commands,
mut er_despawn: EventReader<DespawnEvent>,
2024-04-02 03:12:53 +00:00
mut db: ResMut<ActiveAsteroids>,
mut scene_spawner: ResMut<SceneSpawner>,
) {
for despawn in er_despawn.read() {
2024-04-02 03:12:53 +00:00
commands.entity(despawn.entity).despawn();
scene_spawner.despawn_instance(despawn.sceneinstance);
2024-04-02 03:12:53 +00:00
db.0.remove(&despawn.origin);
}
}
2024-04-01 03:25:35 +00:00
fn handle_cheats(
key_input: Res<ButtonInput<KeyCode>>,
mut q_player: Query<(&Transform, &mut Position, &mut LinearVelocity), With<actor::PlayerCamera>>,
mut q_life: Query<(&mut actor::LifeForm, &mut actor::ExperiencesGForce), With<actor::Player>>,
2024-04-14 21:38:55 +00:00
q_target: Query<(&Transform, &Position, &LinearVelocity), (With<hud::IsTargeted>, Without<actor::PlayerCamera>)>,
mut ew_playerdies: EventWriter<actor::PlayerDiesEvent>,
mut settings: ResMut<var::Settings>,
2024-04-07 23:44:36 +00:00
mut ew_sfx: EventWriter<audio::PlaySfxEvent>,
2024-04-01 03:25:35 +00:00
) {
if q_player.is_empty() || q_life.is_empty() {
2024-04-01 03:25:35 +00:00
return;
}
let (trans, mut pos, mut v) = q_player.get_single_mut().unwrap();
let (mut lifeform, mut gforce) = q_life.get_single_mut().unwrap();
let boost = if key_input.pressed(KeyCode::ShiftLeft) {
1e6
} else {
1e3
};
2024-04-07 23:44:36 +00:00
if key_input.just_pressed(settings.key_cheat_god_mode) {
settings.god_mode ^= true;
if settings.god_mode {
ew_sfx.send(audio::PlaySfxEvent(audio::Sfx::EnterVehicle));
}
}
if !settings.god_mode && !settings.dev_mode {
return;
}
2024-04-01 03:25:35 +00:00
if key_input.just_pressed(settings.key_cheat_stop) {
gforce.ignore_gforce_seconds = 1.0;
v.0 = DVec3::ZERO;
2024-04-01 03:25:35 +00:00
}
if key_input.pressed(settings.key_cheat_speed) {
gforce.ignore_gforce_seconds = 1.0;
v.0 += DVec3::from(trans.rotation * Vec3::new(0.0, 0.0, boost));
2024-04-01 03:25:35 +00:00
}
if key_input.pressed(settings.key_cheat_speed_backward) {
gforce.ignore_gforce_seconds = 1.0;
v.0 += DVec3::from(trans.rotation * Vec3::new(0.0, 0.0, -boost));
}
2024-04-14 21:38:55 +00:00
if key_input.just_pressed(settings.key_cheat_teleport) {
if let Ok((transform, target_pos, target_v)) = q_target.get_single() {
let offset: DVec3 = 4.0 * (**pos - **target_pos).normalize() * transform.scale.as_dvec3();
pos.0 = **target_pos + offset;
*v = target_v.clone();
}
}
if !settings.dev_mode {
return;
}
if key_input.just_pressed(settings.key_cheat_pizza) {
2024-04-14 02:40:42 +00:00
pos.0 = DVec3::new(-121100218.0, 593057.0, -190818113.0);
gforce.ignore_gforce_seconds = 1.0;
}
if key_input.just_pressed(settings.key_cheat_farview1) {
2024-04-14 02:40:42 +00:00
pos.0 = DVec3::new(27643e3, -47e3, -124434e3);
gforce.ignore_gforce_seconds = 1.0;
}
if key_input.just_pressed(settings.key_cheat_farview2) {
2024-04-14 02:40:42 +00:00
pos.0 = DVec3::new(-184968e3, 149410e3, -134273e3);
gforce.ignore_gforce_seconds = 1.0;
}
2024-04-01 03:25:35 +00:00
if key_input.pressed(settings.key_cheat_adrenaline_zero) {
lifeform.adrenaline = 0.0;
}
if key_input.pressed(settings.key_cheat_adrenaline_mid) {
lifeform.adrenaline = 0.5;
}
if key_input.pressed(settings.key_cheat_adrenaline_max) {
lifeform.adrenaline = 1.0;
}
if key_input.just_pressed(settings.key_cheat_die) {
settings.god_mode = false;
ew_playerdies.send(actor::PlayerDiesEvent(actor::DamageType::Trauma));
}
2024-04-01 03:25:35 +00:00
}
// A variant of bevy_xpbd_3d::plugins::position_to_transform that adjusts
// the rendering position to center entities at the player camera.
// This avoids rendering glitches when very far away from the origin.
pub fn position_to_transform(
q_player: Query<&Position, With<actor::PlayerCamera>>,
mut q_trans: Query<(&'static mut Transform, &'static Position, &'static Rotation, Option<&'static Parent>)>,
parents: Query<(&'static GlobalTransform, Option<&'static Position>, Option<&'static Rotation>), With<Children>>,
) {
if let Ok(player_pos) = q_player.get_single() {
for (mut transform, pos, rot, parent) in &mut q_trans {
if let Some(parent) = parent {
if let Ok((parent_transform, parent_pos, parent_rot)) = parents.get(**parent) {
// Compute the global transform of the parent using its Position and Rotation
let parent_transform = parent_transform.compute_transform();
let parent_pos = parent_pos.map_or(parent_transform.translation, |pos| {
pos.as_vec3()
// NOTE: I commented out this because it turns a vec3 to a vec4,
// and I don't understand why bevy_xpbd would do that.
//.extend(parent_transform.translation.z)
});
let parent_rot = parent_rot.map_or(parent_transform.rotation, |rot| {
rot.as_quat()
});
let parent_scale = parent_transform.scale;
let parent_transform = Transform::from_translation(parent_pos)
.with_rotation(parent_rot)
.with_scale(parent_scale);
// The new local transform of the child body,
// computed from the its global transform and its parents global transform
let new_transform = GlobalTransform::from(
Transform::from_translation(
pos.as_vec3()
// NOTE: I commented out this because it turns a vec3 to a vec4,
// and I don't understand why bevy_xpbd would do that.
//.extend(parent_pos.z + transform.translation.z * parent_scale.z),
)
.with_rotation(rot.as_quat()),
)
.reparented_to(&GlobalTransform::from(parent_transform));
transform.translation = new_transform.translation;
transform.rotation = new_transform.rotation;
}
} else {
transform.translation = Vec3::new(
(pos.x - player_pos.x) as f32,
(pos.y - player_pos.y) as f32,
(pos.z - player_pos.z) as f32,
);
transform.rotation = rot.as_quat();
}
}
}
}