2024-07-30 13:34:34 -07:00
use std ::collections ::{ HashMap , HashSet } ;
2024-01-29 22:37:05 -08:00
use crate ::model_physics ::{ self , PhysicsMesh , PhysicsMeshTransform , TransformedMesh , MeshQuery , PhysicsMeshId , PhysicsSubmeshId } ;
2024-01-29 16:19:41 -08:00
use strafesnet_common ::bvh ;
2024-01-29 22:37:05 -08:00
use strafesnet_common ::map ;
2024-08-01 09:29:13 -07:00
use strafesnet_common ::run ;
2024-01-29 16:19:41 -08:00
use strafesnet_common ::aabb ;
2024-01-29 22:37:05 -08:00
use strafesnet_common ::model ::{ MeshId , ModelId } ;
2024-02-16 00:19:44 -08:00
use strafesnet_common ::gameplay_attributes ::{ self , CollisionAttributesId } ;
use strafesnet_common ::gameplay_modes ::{ self , StageId } ;
2024-01-29 22:37:05 -08:00
use strafesnet_common ::gameplay_style ::{ self , StyleModifiers } ;
2024-02-16 00:19:44 -08:00
use strafesnet_common ::controls_bitflag ::Controls ;
2024-01-29 16:19:41 -08:00
use strafesnet_common ::instruction ::{ self , InstructionEmitter , InstructionConsumer , TimedInstruction } ;
2025-01-07 22:54:58 -08:00
use strafesnet_common ::integer ::{ self , vec3 , mat3 , Planar64 , Planar64Vec3 , Planar64Mat3 , Angle32 , Ratio64Vec2 } ;
pub use strafesnet_common ::physics ::{ Time , TimeInner } ;
2024-01-29 22:37:05 -08:00
use gameplay ::ModeState ;
2023-09-27 02:12:20 -07:00
2025-01-07 22:54:58 -08:00
pub type Body = crate ::body ::Body < TimeInner > ;
type MouseState = strafesnet_common ::mouse ::MouseState < TimeInner > ;
2024-08-06 11:10:43 -07:00
//external influence
//this is how you influence the physics from outside
use strafesnet_common ::physics ::Instruction as PhysicsInputInstruction ;
2024-08-01 11:47:20 -07:00
//internal influence
//when the physics asks itself what happens next, this is how it's represented
2023-09-18 18:34:48 -07:00
#[ derive(Debug) ]
2025-01-07 22:54:58 -08:00
pub enum PhysicsInternalInstruction {
2024-09-11 15:58:13 -07:00
CollisionStart ( Collision , model_physics ::GigaTime ) ,
CollisionEnd ( Collision , model_physics ::GigaTime ) ,
2023-09-08 17:00:50 -07:00
StrafeTick ,
2023-09-08 20:14:18 -07:00
ReachWalkTargetVelocity ,
2023-09-08 20:13:30 -07:00
// Water,
2024-08-01 11:47:20 -07:00
}
#[ derive(Debug) ]
2025-01-07 22:54:58 -08:00
pub enum PhysicsInstruction {
2024-08-01 11:47:20 -07:00
Internal ( PhysicsInternalInstruction ) ,
//InputInstructions conditionally activate RefreshWalkTarget
//(by doing what SetWalkTargetVelocity used to do and then flagging it)
Input ( PhysicsInputInstruction ) ,
2023-09-08 17:00:50 -07:00
}
2023-10-18 17:17:21 -07:00
2024-02-16 00:19:44 -08:00
#[ derive(Clone,Debug,Default) ]
pub struct InputState {
mouse :MouseState ,
next_mouse :MouseState ,
controls :strafesnet_common ::controls_bitflag ::Controls ,
}
impl InputState {
pub const fn get_next_mouse ( & self ) ->& MouseState {
& self . next_mouse
}
fn set_next_mouse ( & mut self , next_mouse :MouseState ) {
2024-07-30 19:36:03 -07:00
//I like your functions magic language
self . mouse = std ::mem ::replace ( & mut self . next_mouse , next_mouse ) ;
//equivalently:
//(self.next_mouse,self.mouse)=(next_mouse,self.next_mouse.clone());
2024-02-16 00:19:44 -08:00
}
fn replace_mouse ( & mut self , mouse :MouseState , next_mouse :MouseState ) {
( self . next_mouse , self . mouse ) = ( next_mouse , mouse ) ;
}
fn set_control ( & mut self , control :Controls , state :bool ) {
self . controls . set ( control , state )
}
fn time_delta ( & self ) ->Time {
self . next_mouse . time - self . mouse . time
}
fn mouse_delta ( & self ) ->glam ::IVec2 {
self . next_mouse . pos - self . mouse . pos
}
fn lerp_delta ( & self , time :Time ) ->glam ::IVec2 {
//these are deltas
let dm = self . mouse_delta ( ) . as_i64vec2 ( ) ;
let t = ( time - self . mouse . time ) . nanos ( ) ;
let dt = self . time_delta ( ) . nanos ( ) ;
( ( dm * t ) / dt ) . as_ivec2 ( )
}
}
#[ derive(Clone,Debug) ]
2023-11-30 01:51:17 -08:00
enum JumpDirection {
Exactly ( Planar64Vec3 ) ,
FromContactNormal ,
}
2024-02-16 00:19:44 -08:00
impl JumpDirection {
fn direction ( & self , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , contact :& ContactCollision ) ->Planar64Vec3 {
match self {
JumpDirection ::FromContactNormal = > contact_normal ( models , hitbox_mesh , contact ) ,
& JumpDirection ::Exactly ( dir ) = > dir ,
}
}
2023-09-18 23:36:14 -07:00
}
2024-02-16 00:19:44 -08:00
#[ derive(Clone,Debug) ]
enum TransientAcceleration {
Reached ,
Reachable {
acceleration :Planar64Vec3 ,
time :Time ,
} ,
//walk target will never be reached
Unreachable {
acceleration :Planar64Vec3 ,
}
2023-09-18 21:10:07 -07:00
}
2024-02-16 00:19:44 -08:00
#[ derive(Clone,Debug) ]
struct ContactMoveState {
2023-11-30 01:51:17 -08:00
jump_direction :JumpDirection ,
contact :ContactCollision ,
2024-02-16 00:19:44 -08:00
target :TransientAcceleration ,
}
impl TransientAcceleration {
fn with_target_diff ( target_diff :Planar64Vec3 , accel :Planar64 , time :Time ) ->Self {
2024-09-11 15:58:13 -07:00
if target_diff = = vec3 ::ZERO {
2024-02-16 00:19:44 -08:00
TransientAcceleration ::Reached
2023-10-17 00:17:54 -07:00
} else {
//normal friction acceleration is clippedAcceleration.dot(normal)*friction
2024-02-16 00:19:44 -08:00
TransientAcceleration ::Reachable {
2024-09-11 15:58:13 -07:00
acceleration :target_diff . with_length ( accel ) . divide ( ) . fix_1 ( ) ,
time :time + Time ::from ( ( target_diff . length ( ) / accel ) . divide ( ) . fix_1 ( ) )
2024-02-16 00:19:44 -08:00
}
}
}
fn ground ( walk_settings :& gameplay_style ::WalkSettings , body :& Body , gravity :Planar64Vec3 , target_velocity :Planar64Vec3 ) ->Self {
let target_diff = target_velocity - body . velocity ;
//precalculate accel
let accel = walk_settings . accel ( target_diff , gravity ) ;
Self ::with_target_diff ( target_diff , accel , body . time )
}
fn ladder ( ladder_settings :& gameplay_style ::LadderSettings , body :& Body , gravity :Planar64Vec3 , target_velocity :Planar64Vec3 ) ->Self {
let target_diff = target_velocity - body . velocity ;
let accel = ladder_settings . accel ( target_diff , gravity ) ;
Self ::with_target_diff ( target_diff , accel , body . time )
}
fn acceleration ( & self ) ->Planar64Vec3 {
match self {
2024-09-11 15:58:13 -07:00
TransientAcceleration ::Reached = > vec3 ::ZERO ,
2024-02-16 00:19:44 -08:00
& TransientAcceleration ::Reachable { acceleration , time :_ } = > acceleration ,
& TransientAcceleration ::Unreachable { acceleration } = > acceleration ,
2023-09-18 21:10:07 -07:00
}
}
}
2024-02-16 00:19:44 -08:00
impl ContactMoveState {
fn ground ( walk_settings :& gameplay_style ::WalkSettings , body :& Body , gravity :Planar64Vec3 , target_velocity :Planar64Vec3 , contact :ContactCollision ) ->Self {
Self {
target :TransientAcceleration ::ground ( walk_settings , body , gravity , target_velocity ) ,
2023-11-30 01:51:17 -08:00
contact ,
2024-09-11 15:58:13 -07:00
jump_direction :JumpDirection ::Exactly ( vec3 ::Y ) ,
2024-02-16 00:19:44 -08:00
}
2023-10-17 00:17:54 -07:00
}
2024-02-16 00:19:44 -08:00
fn ladder ( ladder_settings :& gameplay_style ::LadderSettings , body :& Body , gravity :Planar64Vec3 , target_velocity :Planar64Vec3 , contact :ContactCollision ) ->Self {
Self { //,style,velocity,normal,style.ladder_speed,style.ladder_accel
target :TransientAcceleration ::ladder ( ladder_settings , body , gravity , target_velocity ) ,
2023-11-30 01:51:17 -08:00
contact ,
jump_direction :JumpDirection ::FromContactNormal ,
2024-02-16 00:19:44 -08:00
}
2023-10-17 00:17:54 -07:00
}
}
2024-02-16 00:19:44 -08:00
fn ground_things ( walk_settings :& gameplay_style ::WalkSettings , contact :& ContactCollision , touching :& TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , style :& StyleModifiers , camera :& PhysicsCamera , input_state :& InputState ) ->( Planar64Vec3 , Planar64Vec3 ) {
let normal = contact_normal ( models , hitbox_mesh , contact ) ;
let gravity = touching . base_acceleration ( models , style , camera , input_state ) ;
let control_dir = style . get_y_control_dir ( camera , input_state . controls ) ;
let mut target_velocity = walk_settings . get_walk_target_velocity ( control_dir , normal ) ;
touching . constrain_velocity ( models , hitbox_mesh , & mut target_velocity ) ;
( gravity , target_velocity )
}
fn ladder_things ( ladder_settings :& gameplay_style ::LadderSettings , contact :& ContactCollision , touching :& TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , style :& StyleModifiers , camera :& PhysicsCamera , input_state :& InputState ) ->( Planar64Vec3 , Planar64Vec3 ) {
let normal = contact_normal ( models , hitbox_mesh , contact ) ;
let gravity = touching . base_acceleration ( models , style , camera , input_state ) ;
let control_dir = style . get_y_control_dir ( camera , input_state . controls ) ;
let mut target_velocity = ladder_settings . get_ladder_target_velocity ( control_dir , normal ) ;
touching . constrain_velocity ( models , hitbox_mesh , & mut target_velocity ) ;
( gravity , target_velocity )
}
2023-09-18 21:10:07 -07:00
2023-11-30 01:51:17 -08:00
#[ derive(Default) ]
2023-10-18 16:21:06 -07:00
struct PhysicsModels {
2024-01-29 22:37:05 -08:00
meshes :HashMap < PhysicsMeshId , PhysicsMesh > ,
2024-08-09 14:16:10 -07:00
contact_models :HashMap < ContactModelId , ContactModel > ,
intersect_models :HashMap < IntersectModelId , IntersectModel > ,
contact_attributes :HashMap < ContactAttributesId , gameplay_attributes ::ContactAttributes > ,
intersect_attributes :HashMap < IntersectAttributesId , gameplay_attributes ::IntersectAttributes > ,
2023-10-18 16:21:06 -07:00
}
impl PhysicsModels {
fn clear ( & mut self ) {
2023-11-30 01:51:17 -08:00
self . meshes . clear ( ) ;
2024-08-09 14:16:10 -07:00
self . contact_models . clear ( ) ;
self . intersect_models . clear ( ) ;
self . contact_attributes . clear ( ) ;
self . intersect_attributes . clear ( ) ;
2023-11-30 01:51:17 -08:00
}
2024-01-29 22:37:05 -08:00
fn mesh ( & self , convex_mesh_id :ConvexMeshId ) ->TransformedMesh {
2024-08-09 14:16:10 -07:00
let ( mesh_id , transform ) = match convex_mesh_id . model_id {
PhysicsModelId ::Contact ( model_id ) = > {
let model = & self . contact_models [ & model_id ] ;
( & model . mesh_id , & model . transform )
} ,
PhysicsModelId ::Intersect ( model_id ) = > {
let model = & self . intersect_models [ & model_id ] ;
( & model . mesh_id , & model . transform )
} ,
} ;
TransformedMesh ::new (
self . meshes [ mesh_id ] . submesh_view ( convex_mesh_id . submesh_id ) ,
transform
)
}
//it's a bit weird to have three functions, but it's always going to be one of these
fn contact_mesh ( & self , contact :& ContactCollision ) ->TransformedMesh {
let model = & self . contact_models [ & contact . model_id ] ;
2023-11-30 01:51:17 -08:00
TransformedMesh ::new (
2024-08-09 14:16:10 -07:00
self . meshes [ & model . mesh_id ] . submesh_view ( contact . submesh_id ) ,
2024-01-29 22:37:05 -08:00
& model . transform
2023-11-30 01:51:17 -08:00
)
}
2024-08-09 14:16:10 -07:00
fn intersect_mesh ( & self , intersect :& IntersectCollision ) ->TransformedMesh {
let model = & self . intersect_models [ & intersect . model_id ] ;
TransformedMesh ::new (
self . meshes [ & model . mesh_id ] . submesh_view ( intersect . submesh_id ) ,
& model . transform
)
2023-11-30 01:51:17 -08:00
}
2024-08-09 14:16:10 -07:00
fn get_model_transform ( & self , model_id :ModelId ) ->Option < & PhysicsMeshTransform > {
//ModelId can possibly be a decoration
2025-01-07 20:19:44 -08:00
match self . contact_models . get ( & ContactModelId ::new ( model_id . get ( ) ) ) {
Some ( model ) = > Some ( & model . transform ) ,
None = > self . intersect_models . get ( & IntersectModelId ::new ( model_id . get ( ) ) )
2024-08-09 14:16:10 -07:00
. map ( | model | & model . transform ) ,
2025-01-07 20:19:44 -08:00
}
2024-08-09 14:16:10 -07:00
}
fn contact_model ( & self , model_id :ContactModelId ) ->& ContactModel {
& self . contact_models [ & model_id ]
}
fn intersect_model ( & self , model_id :IntersectModelId ) ->& IntersectModel {
& self . intersect_models [ & model_id ]
}
fn contact_attr ( & self , model_id :ContactModelId ) ->& gameplay_attributes ::ContactAttributes {
& self . contact_attributes [ & self . contact_models [ & model_id ] . attr_id ]
}
fn intersect_attr ( & self , model_id :IntersectModelId ) ->& gameplay_attributes ::IntersectAttributes {
& self . intersect_attributes [ & self . intersect_models [ & model_id ] . attr_id ]
2023-10-18 16:21:06 -07:00
}
}
2024-02-16 00:19:44 -08:00
#[ derive(Clone,Copy,Debug) ]
2023-10-30 22:25:03 -07:00
pub struct PhysicsCamera {
2023-09-27 02:12:20 -07:00
//punch: Planar64Vec3,
//punch_velocity: Planar64Vec3,
sensitivity :Ratio64Vec2 , //dots to Angle32 ratios
clamped_mouse_pos :glam ::IVec2 , //angles are calculated from this cumulative value
//angle limits could be an enum + struct that defines whether it's limited and selects clamp or wrap depending
// enum AngleLimit{
// Unlimited,
// Limited{lower:Angle32,upper:Angle32},
// }
//pitch_limit:AngleLimit,
//yaw_limit:AngleLimit,
2023-09-19 17:53:29 -07:00
}
2024-01-29 22:37:05 -08:00
impl PhysicsCamera {
const ANGLE_PITCH_LOWER_LIMIT :Angle32 = Angle32 ::NEG_FRAC_PI_2 ;
const ANGLE_PITCH_UPPER_LIMIT :Angle32 = Angle32 ::FRAC_PI_2 ;
2024-02-16 00:19:44 -08:00
pub fn move_mouse ( & mut self , mouse_delta :glam ::IVec2 ) {
let mut unclamped_mouse_pos = self . clamped_mouse_pos + mouse_delta ;
2023-09-27 02:12:20 -07:00
unclamped_mouse_pos . y = unclamped_mouse_pos . y . clamp (
2024-01-29 22:37:05 -08:00
self . sensitivity . y . rhs_div_int ( Self ::ANGLE_PITCH_LOWER_LIMIT . get ( ) as i64 ) as i32 ,
self . sensitivity . y . rhs_div_int ( Self ::ANGLE_PITCH_UPPER_LIMIT . get ( ) as i64 ) as i32 ,
2023-09-27 02:12:20 -07:00
) ;
self . clamped_mouse_pos = unclamped_mouse_pos ;
2023-09-19 17:53:29 -07:00
}
2024-02-16 00:19:44 -08:00
pub fn simulate_move_angles ( & self , mouse_delta :glam ::IVec2 ) ->glam ::Vec2 {
let a = - self . sensitivity . mul_int ( ( self . clamped_mouse_pos + mouse_delta ) . as_i64vec2 ( ) ) ;
2023-09-27 02:12:20 -07:00
let ax = Angle32 ::wrap_from_i64 ( a . x ) ;
let ay = Angle32 ::clamp_from_i64 ( a . y )
//clamp to actual vertical cam limit
2024-01-29 22:37:05 -08:00
. clamp ( Self ::ANGLE_PITCH_LOWER_LIMIT , Self ::ANGLE_PITCH_UPPER_LIMIT ) ;
2023-09-27 02:12:20 -07:00
return glam ::vec2 ( ax . into ( ) , ay . into ( ) ) ;
}
2024-02-16 00:19:44 -08:00
#[ inline ]
fn get_rotation ( & self , mouse_pos :glam ::IVec2 ) ->Planar64Mat3 {
let a = - self . sensitivity . mul_int ( mouse_pos . as_i64vec2 ( ) ) ;
2023-10-16 23:20:38 -07:00
let ax = Angle32 ::wrap_from_i64 ( a . x ) ;
let ay = Angle32 ::clamp_from_i64 ( a . y )
//clamp to actual vertical cam limit
2024-01-29 22:37:05 -08:00
. clamp ( Self ::ANGLE_PITCH_LOWER_LIMIT , Self ::ANGLE_PITCH_UPPER_LIMIT ) ;
2024-09-11 15:58:13 -07:00
mat3 ::from_rotation_yx ( ax , ay )
2023-10-16 23:20:38 -07:00
}
2024-02-16 00:19:44 -08:00
fn rotation ( & self ) ->Planar64Mat3 {
self . get_rotation ( self . clamped_mouse_pos )
}
fn simulate_move_rotation ( & self , mouse_delta :glam ::IVec2 ) ->Planar64Mat3 {
self . get_rotation ( self . clamped_mouse_pos + mouse_delta )
}
fn get_rotation_y ( & self , mouse_pos_x :i32 ) ->Planar64Mat3 {
let ax = - self . sensitivity . x . mul_int ( mouse_pos_x as i64 ) ;
2024-09-11 15:58:13 -07:00
mat3 ::from_rotation_y ( Angle32 ::wrap_from_i64 ( ax ) )
2023-09-19 17:53:29 -07:00
}
2024-02-16 00:19:44 -08:00
fn rotation_y ( & self ) ->Planar64Mat3 {
self . get_rotation_y ( self . clamped_mouse_pos . x )
}
fn simulate_move_rotation_y ( & self , mouse_delta_x :i32 ) ->Planar64Mat3 {
self . get_rotation_y ( self . clamped_mouse_pos . x + mouse_delta_x )
}
2023-09-19 17:53:29 -07:00
}
2023-10-18 17:17:21 -07:00
impl std ::default ::Default for PhysicsCamera {
fn default ( ) ->Self {
Self {
sensitivity :Ratio64Vec2 ::ONE * 200_000 ,
clamped_mouse_pos :glam ::IVec2 ::ZERO ,
}
}
}
2024-01-29 22:37:05 -08:00
mod gameplay {
use super ::{ gameplay_modes , HashSet , HashMap , ModelId } ;
2025-01-06 21:22:40 -08:00
pub enum JumpIncrementResult {
Allowed ,
ExceededLimit ,
}
impl JumpIncrementResult {
pub const fn is_allowed ( self ) ->bool {
matches! ( self , JumpIncrementResult ::Allowed )
}
}
2024-02-16 00:19:44 -08:00
#[ derive(Clone,Debug) ]
2024-01-29 22:37:05 -08:00
pub struct ModeState {
mode_id :gameplay_modes ::ModeId ,
stage_id :gameplay_modes ::StageId ,
next_ordered_checkpoint_id :gameplay_modes ::CheckpointId , //which OrderedCheckpoint model_id you must pass next (if 0 you haven't passed OrderedCheckpoint0)
unordered_checkpoints :HashSet < ModelId > ,
jump_counts :HashMap < ModelId , u32 > , //model_id -> jump count
}
impl ModeState {
pub const fn get_mode_id ( & self ) ->gameplay_modes ::ModeId {
self . mode_id
}
pub const fn get_stage_id ( & self ) ->gameplay_modes ::StageId {
self . stage_id
}
pub const fn get_next_ordered_checkpoint_id ( & self ) ->gameplay_modes ::CheckpointId {
self . next_ordered_checkpoint_id
}
2025-01-06 21:22:40 -08:00
fn increment_jump_count ( & mut self , model_id :ModelId ) ->u32 {
* self . jump_counts . entry ( model_id ) . and_modify ( | c | * c + = 1 ) . or_insert ( 1 )
}
pub fn try_increment_jump_count ( & mut self , model_id :ModelId , jump_limit :Option < u8 > ) ->JumpIncrementResult {
match jump_limit {
Some ( jump_limit ) if ( jump_limit as u32 ) < self . increment_jump_count ( model_id ) = > JumpIncrementResult ::ExceededLimit ,
_ = > JumpIncrementResult ::Allowed ,
}
2024-01-29 22:37:05 -08:00
}
pub const fn ordered_checkpoint_count ( & self ) ->u32 {
self . next_ordered_checkpoint_id . get ( )
}
pub fn unordered_checkpoint_count ( & self ) ->u32 {
self . unordered_checkpoints . len ( ) as u32
}
pub fn set_mode_id ( & mut self , mode_id :gameplay_modes ::ModeId ) {
self . clear ( ) ;
self . mode_id = mode_id ;
}
pub fn set_stage_id ( & mut self , stage_id :gameplay_modes ::StageId ) {
self . clear_checkpoints ( ) ;
self . stage_id = stage_id ;
}
pub fn accumulate_ordered_checkpoint ( & mut self , stage :& gameplay_modes ::Stage , model_id :ModelId ) {
if stage . is_next_ordered_checkpoint ( self . get_next_ordered_checkpoint_id ( ) , model_id ) {
self . next_ordered_checkpoint_id = gameplay_modes ::CheckpointId ::new ( self . next_ordered_checkpoint_id . get ( ) + 1 ) ;
}
}
pub fn accumulate_unordered_checkpoint ( & mut self , stage :& gameplay_modes ::Stage , model_id :ModelId ) {
if stage . is_unordered_checkpoint ( model_id ) {
self . unordered_checkpoints . insert ( model_id ) ;
}
}
pub fn clear ( & mut self ) {
self . clear_jump_counts ( ) ;
self . clear_checkpoints ( ) ;
}
pub fn clear_jump_counts ( & mut self ) {
self . jump_counts . clear ( ) ;
}
pub fn clear_checkpoints ( & mut self ) {
self . next_ordered_checkpoint_id = gameplay_modes ::CheckpointId ::FIRST ;
self . unordered_checkpoints . clear ( ) ;
}
}
impl std ::default ::Default for ModeState {
fn default ( ) ->Self {
Self {
mode_id :gameplay_modes ::ModeId ::MAIN ,
stage_id :gameplay_modes ::StageId ::FIRST ,
next_ordered_checkpoint_id :gameplay_modes ::CheckpointId ::FIRST ,
unordered_checkpoints :HashSet ::new ( ) ,
jump_counts :HashMap ::new ( ) ,
}
2023-10-02 22:45:20 -07:00
}
}
}
2024-02-16 00:19:44 -08:00
#[ derive(Clone,Debug) ]
2023-10-17 16:18:55 -07:00
struct WorldState { }
2024-01-29 22:37:05 -08:00
struct HitboxMesh {
2023-11-30 01:51:17 -08:00
halfsize :Planar64Vec3 ,
mesh :PhysicsMesh ,
2024-01-29 22:37:05 -08:00
transform :PhysicsMeshTransform ,
2023-11-30 01:51:17 -08:00
}
2024-01-29 22:37:05 -08:00
impl HitboxMesh {
2024-01-29 16:19:41 -08:00
fn new ( mesh :PhysicsMesh , transform :integer ::Planar64Affine3 ) ->Self {
2023-11-30 01:51:17 -08:00
//calculate extents
2024-01-29 16:19:41 -08:00
let mut aabb = aabb ::Aabb ::default ( ) ;
2024-01-29 22:37:05 -08:00
let transform = PhysicsMeshTransform ::new ( transform ) ;
let transformed_mesh = TransformedMesh ::new ( mesh . complete_mesh_view ( ) , & transform ) ;
for vert in transformed_mesh . verts ( ) {
2024-09-11 15:58:13 -07:00
aabb . grow ( vert . fix_1 ( ) ) ;
2023-11-30 01:51:17 -08:00
}
Self {
2024-09-11 15:58:13 -07:00
halfsize :aabb . size ( ) > > 1 ,
2023-11-30 01:51:17 -08:00
mesh ,
transform ,
}
}
#[ inline ]
2024-08-19 17:01:49 -07:00
const fn transformed_mesh ( & self ) ->TransformedMesh {
2024-01-29 22:37:05 -08:00
TransformedMesh ::new ( self . mesh . complete_mesh_view ( ) , & self . transform )
2023-11-30 01:51:17 -08:00
}
}
2024-01-29 22:37:05 -08:00
trait StyleHelper {
2024-02-16 00:19:44 -08:00
fn get_control ( & self , control :Controls , controls :Controls ) ->bool ;
fn get_control_dir ( & self , controls :Controls ) ->Planar64Vec3 ;
fn get_y_control_dir ( & self , camera :& PhysicsCamera , controls :Controls ) ->Planar64Vec3 ;
fn get_propulsion_control_dir ( & self , camera :& PhysicsCamera , controls :Controls ) ->Planar64Vec3 ;
2024-01-29 22:37:05 -08:00
fn calculate_mesh ( & self ) ->HitboxMesh ;
}
impl StyleHelper for StyleModifiers {
2024-02-16 00:19:44 -08:00
fn get_control ( & self , control :Controls , controls :Controls ) ->bool {
controls . intersection ( self . controls_mask ) . contains ( control )
2023-10-31 14:11:39 -07:00
}
2024-02-16 00:19:44 -08:00
fn get_control_dir ( & self , controls :Controls ) ->Planar64Vec3 {
2023-10-03 16:52:45 -07:00
//don't get fancy just do it
2024-09-11 15:58:13 -07:00
let mut control_dir :Planar64Vec3 = vec3 ::ZERO ;
2023-10-03 16:52:45 -07:00
//Apply mask after held check so you can require non-allowed keys to be held for some reason
2024-02-16 00:19:44 -08:00
let controls = controls . intersection ( self . controls_mask ) ;
if controls . contains ( Controls ::MoveForward ) {
2023-10-03 16:52:45 -07:00
control_dir + = Self ::FORWARD_DIR ;
}
2024-02-16 00:19:44 -08:00
if controls . contains ( Controls ::MoveBackward ) {
2023-09-27 02:12:20 -07:00
control_dir - = Self ::FORWARD_DIR ;
2023-10-03 16:52:45 -07:00
}
2024-02-16 00:19:44 -08:00
if controls . contains ( Controls ::MoveLeft ) {
2023-09-27 02:12:20 -07:00
control_dir - = Self ::RIGHT_DIR ;
2023-10-03 16:52:45 -07:00
}
2024-02-16 00:19:44 -08:00
if controls . contains ( Controls ::MoveRight ) {
2023-10-03 16:52:45 -07:00
control_dir + = Self ::RIGHT_DIR ;
}
2024-02-16 00:19:44 -08:00
if controls . contains ( Controls ::MoveUp ) {
2023-10-03 16:52:45 -07:00
control_dir + = Self ::UP_DIR ;
}
2024-02-16 00:19:44 -08:00
if controls . contains ( Controls ::MoveDown ) {
2023-09-27 02:12:20 -07:00
control_dir - = Self ::UP_DIR ;
2023-10-03 16:52:45 -07:00
}
return control_dir
}
2023-10-14 14:51:13 -07:00
2024-02-16 00:19:44 -08:00
fn get_y_control_dir ( & self , camera :& PhysicsCamera , controls :Controls ) ->Planar64Vec3 {
2024-09-11 15:58:13 -07:00
( camera . rotation_y ( ) * self . get_control_dir ( controls ) ) . fix_1 ( )
2023-10-17 00:17:54 -07:00
}
2024-02-16 00:19:44 -08:00
fn get_propulsion_control_dir ( & self , camera :& PhysicsCamera , controls :Controls ) ->Planar64Vec3 {
//don't interpolate this! discrete mouse movement, constant acceleration
2024-09-11 15:58:13 -07:00
( camera . rotation ( ) * self . get_control_dir ( controls ) ) . fix_1 ( )
2023-10-14 14:51:13 -07:00
}
2024-01-29 22:37:05 -08:00
fn calculate_mesh ( & self ) ->HitboxMesh {
let mesh = match self . hitbox . mesh {
gameplay_style ::HitboxMesh ::Box = > PhysicsMesh ::unit_cube ( ) ,
gameplay_style ::HitboxMesh ::Cylinder = > PhysicsMesh ::unit_cylinder ( ) ,
} ;
2024-09-11 15:58:13 -07:00
let transform = integer ::Planar64Affine3 ::new (
mat3 ::from_diagonal ( self . hitbox . halfsize ) ,
vec3 ::ZERO
) ;
2024-01-29 22:37:05 -08:00
HitboxMesh ::new ( mesh , transform )
2023-11-30 01:51:17 -08:00
}
2023-10-03 16:52:45 -07:00
}
2024-02-16 00:19:44 -08:00
#[ derive(Clone,Debug) ]
2023-10-12 19:44:46 -07:00
enum MoveState {
Air ,
2024-02-16 00:19:44 -08:00
Walk ( ContactMoveState ) ,
Ladder ( ContactMoveState ) ,
2023-10-12 19:44:46 -07:00
Water ,
2024-02-16 00:19:44 -08:00
Fly ,
}
impl MoveState {
//call this after state.move_state is changed
fn apply_enum ( & self , body :& mut Body , touching :& TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , style :& StyleModifiers , camera :& PhysicsCamera , input_state :& InputState ) {
match self {
2024-09-11 15:58:13 -07:00
MoveState ::Fly = > body . acceleration = vec3 ::ZERO ,
2024-02-16 00:19:44 -08:00
MoveState ::Air = > {
//calculate base acceleration
let a = touching . base_acceleration ( models , style , camera , input_state ) ;
//set_acceleration clips according to contacts
set_acceleration ( body , touching , models , hitbox_mesh , a ) ;
} ,
_ = > ( ) ,
}
}
//function to coerce &mut self into &self
fn apply_to_body ( & self , body :& mut Body , touching :& TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , style :& StyleModifiers , camera :& PhysicsCamera , input_state :& InputState ) {
match self {
MoveState ::Air = > ( ) ,
MoveState ::Water = > ( ) ,
MoveState ::Fly = > {
//set velocity according to current control state
let v = style . get_propulsion_control_dir ( camera , input_state . controls ) * 80 ;
//set_velocity clips velocity according to current touching state
set_velocity ( body , touching , models , hitbox_mesh , v ) ;
} ,
MoveState ::Walk ( walk_state )
| MoveState ::Ladder ( walk_state )
= > {
//accelerate towards walk target or do nothing
let a = walk_state . target . acceleration ( ) ;
set_acceleration ( body , touching , models , hitbox_mesh , a ) ;
} ,
}
}
/// changes the move state
fn apply_input ( & mut self , body :& Body , touching :& TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , style :& StyleModifiers , camera :& PhysicsCamera , input_state :& InputState ) {
match self {
MoveState ::Fly
| MoveState ::Air
| MoveState ::Water = > ( ) ,
MoveState ::Walk ( ContactMoveState { target , contact , jump_direction :_ } ) = > {
if let Some ( walk_settings ) = & style . walk {
let ( gravity , target_velocity ) = ground_things ( walk_settings , contact , touching , models , hitbox_mesh , style , camera , input_state ) ;
* target = TransientAcceleration ::ground ( walk_settings , body , gravity , target_velocity ) ;
} else {
panic! ( " ContactMoveState exists in style which does not allow walking! " ) ;
}
} ,
MoveState ::Ladder ( ContactMoveState { target , contact , jump_direction :_ } ) = > {
if let Some ( ladder_settings ) = & style . ladder {
let ( gravity , target_velocity ) = ladder_things ( ladder_settings , contact , touching , models , hitbox_mesh , style , camera , input_state ) ;
* target = TransientAcceleration ::ladder ( ladder_settings , body , gravity , target_velocity ) ;
} else {
panic! ( " ContactMoveState exists in style which does not allow walking! " ) ;
}
} ,
}
}
fn get_walk_state ( & self ) ->Option < & ContactMoveState > {
match self {
MoveState ::Walk ( walk_state )
| MoveState ::Ladder ( walk_state )
= > Some ( walk_state ) ,
MoveState ::Air
| MoveState ::Water
| MoveState ::Fly
= > None ,
}
}
2025-01-07 22:54:58 -08:00
fn next_move_instruction ( & self , strafe :& Option < gameplay_style ::StrafeSettings > , time :Time ) ->Option < TimedInstruction < PhysicsInternalInstruction , TimeInner > > {
2024-02-16 00:19:44 -08:00
//check if you have a valid walk state and create an instruction
match self {
MoveState ::Walk ( walk_state ) | MoveState ::Ladder ( walk_state ) = > match & walk_state . target {
& TransientAcceleration ::Reachable { acceleration :_ , time } = > Some ( TimedInstruction {
time ,
2024-08-01 11:47:20 -07:00
instruction :PhysicsInternalInstruction ::ReachWalkTargetVelocity
2024-02-16 00:19:44 -08:00
} ) ,
TransientAcceleration ::Unreachable { acceleration :_ }
| TransientAcceleration ::Reached
= > None ,
}
MoveState ::Air = > strafe . as_ref ( ) . map ( | strafe | {
TimedInstruction {
time :strafe . next_tick ( time ) ,
//only poll the physics if there is a before and after mouse event
2024-08-01 11:47:20 -07:00
instruction :PhysicsInternalInstruction ::StrafeTick
2024-02-16 00:19:44 -08:00
}
} ) ,
MoveState ::Water = > None , //TODO
MoveState ::Fly = > None ,
}
}
2024-08-08 16:54:42 -07:00
//lmao idk this is convenient
fn apply_enum_and_input_and_body ( & mut self , body :& mut Body , touching :& TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , style :& StyleModifiers , camera :& PhysicsCamera , input_state :& InputState ) {
self . apply_enum ( body , touching , models , hitbox_mesh , style , camera , input_state ) ;
self . apply_input ( body , touching , models , hitbox_mesh , style , camera , input_state ) ;
self . apply_to_body ( body , touching , models , hitbox_mesh , style , camera , input_state ) ;
}
fn apply_enum_and_body ( & mut self , body :& mut Body , touching :& TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , style :& StyleModifiers , camera :& PhysicsCamera , input_state :& InputState ) {
self . apply_enum ( body , touching , models , hitbox_mesh , style , camera , input_state ) ;
self . apply_to_body ( body , touching , models , hitbox_mesh , style , camera , input_state ) ;
}
fn apply_input_and_body ( & mut self , body :& mut Body , touching :& TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , style :& StyleModifiers , camera :& PhysicsCamera , input_state :& InputState ) {
self . apply_input ( body , touching , models , hitbox_mesh , style , camera , input_state ) ;
self . apply_to_body ( body , touching , models , hitbox_mesh , style , camera , input_state ) ;
}
fn set_move_state ( & mut self , move_state :MoveState , body :& mut Body , touching :& TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , style :& StyleModifiers , camera :& PhysicsCamera , input_state :& InputState ) {
* self = move_state ;
//this function call reads the above state that was just set
self . apply_enum_and_body ( body , touching , models , hitbox_mesh , style , camera , input_state ) ;
}
fn cull_velocity ( & mut self , velocity :Planar64Vec3 , body :& mut Body , touching :& mut TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , style :& StyleModifiers , camera :& PhysicsCamera , input_state :& InputState ) {
//TODO: be more precise about contacts
if set_velocity_cull ( body , touching , models , hitbox_mesh , velocity ) {
//TODO do better
match self . get_walk_state ( ) {
//did you stop touching the thing you were walking on?
Some ( walk_state ) = > if ! touching . contacts . contains ( & walk_state . contact ) {
self . set_move_state ( MoveState ::Air , body , touching , models , hitbox_mesh , style , camera , input_state ) ;
} ,
None = > self . apply_enum_and_body ( body , touching , models , hitbox_mesh , style , camera , input_state ) ,
}
}
}
2023-10-12 19:44:46 -07:00
}
2023-11-30 01:51:17 -08:00
#[ derive(Clone,Hash,Eq,PartialEq) ]
2023-10-02 22:45:20 -07:00
enum PhysicsCollisionAttributes {
2024-08-09 14:16:10 -07:00
Contact ( gameplay_attributes ::ContactAttributes ) ,
Intersect ( gameplay_attributes ::IntersectAttributes ) ,
2023-10-02 22:45:20 -07:00
}
2023-11-30 01:51:17 -08:00
struct NonPhysicsError ;
2024-01-29 22:37:05 -08:00
impl TryFrom < & gameplay_attributes ::CollisionAttributes > for PhysicsCollisionAttributes {
2023-11-30 01:51:17 -08:00
type Error = NonPhysicsError ;
2024-01-29 22:37:05 -08:00
fn try_from ( value :& gameplay_attributes ::CollisionAttributes ) ->Result < Self , Self ::Error > {
2023-11-30 01:51:17 -08:00
match value {
2024-01-29 22:37:05 -08:00
gameplay_attributes ::CollisionAttributes ::Decoration = > Err ( NonPhysicsError ) ,
2024-08-09 14:16:10 -07:00
gameplay_attributes ::CollisionAttributes ::Contact ( attr ) = > Ok ( Self ::Contact ( attr . clone ( ) ) ) ,
gameplay_attributes ::CollisionAttributes ::Intersect ( attr ) = > Ok ( Self ::Intersect ( attr . clone ( ) ) ) ,
2023-11-30 01:51:17 -08:00
}
}
}
2024-01-29 22:37:05 -08:00
#[ derive(Clone,Copy,Hash,id::Id,Eq,PartialEq) ]
2024-08-09 14:16:10 -07:00
struct ContactAttributesId ( u32 ) ;
impl Into < CollisionAttributesId > for ContactAttributesId {
2024-01-29 22:37:05 -08:00
fn into ( self ) ->CollisionAttributesId {
CollisionAttributesId ::new ( self . 0 )
}
}
2024-08-09 14:16:10 -07:00
impl From < CollisionAttributesId > for ContactAttributesId {
2024-01-29 22:37:05 -08:00
fn from ( value :CollisionAttributesId ) ->Self {
Self ::new ( value . get ( ) )
}
}
2024-08-09 14:16:10 -07:00
#[ derive(Clone,Copy,Hash,id::Id,Eq,PartialEq) ]
struct IntersectAttributesId ( u32 ) ;
impl Into < CollisionAttributesId > for IntersectAttributesId {
fn into ( self ) ->CollisionAttributesId {
CollisionAttributesId ::new ( self . 0 )
}
2024-01-29 22:37:05 -08:00
}
2024-08-09 14:16:10 -07:00
impl From < CollisionAttributesId > for IntersectAttributesId {
fn from ( value :CollisionAttributesId ) ->Self {
Self ::new ( value . get ( ) )
}
}
#[ derive(Debug,Clone,Copy,Hash,id::Id,Eq,PartialEq) ]
struct ContactModelId ( u32 ) ;
impl Into < ModelId > for ContactModelId {
2024-01-29 22:37:05 -08:00
fn into ( self ) ->ModelId {
2024-08-09 14:16:10 -07:00
ModelId ::new ( self . get ( ) )
2024-01-29 22:37:05 -08:00
}
}
2024-08-09 14:16:10 -07:00
#[ derive(Debug,Clone,Copy,Hash,id::Id,Eq,PartialEq) ]
struct IntersectModelId ( u32 ) ;
impl Into < ModelId > for IntersectModelId {
fn into ( self ) ->ModelId {
ModelId ::new ( self . get ( ) )
2024-01-29 22:37:05 -08:00
}
}
2024-08-09 14:16:10 -07:00
#[ derive(Debug,Clone,Copy,Hash,Eq,PartialEq) ]
enum PhysicsModelId {
Contact ( ContactModelId ) ,
Intersect ( IntersectModelId ) ,
}
impl Into < ModelId > for PhysicsModelId {
fn into ( self ) ->ModelId {
ModelId ::new ( match self {
PhysicsModelId ::Contact ( model_id ) = > model_id . get ( ) ,
PhysicsModelId ::Intersect ( model_id ) = > model_id . get ( ) ,
} )
}
}
//unique physics meshes indexed by this
#[ derive(Debug,Clone,Copy,Eq,Hash,PartialEq) ]
struct ConvexMeshId {
model_id :PhysicsModelId ,
submesh_id :PhysicsSubmeshId ,
}
struct ContactModel {
2024-01-29 22:37:05 -08:00
mesh_id :PhysicsMeshId ,
2024-08-09 14:16:10 -07:00
attr_id :ContactAttributesId ,
2024-01-29 22:37:05 -08:00
transform :PhysicsMeshTransform ,
2023-09-08 15:54:43 -07:00
}
2024-08-09 14:16:10 -07:00
struct IntersectModel {
mesh_id :PhysicsMeshId ,
attr_id :IntersectAttributesId ,
transform :PhysicsMeshTransform ,
2023-09-08 15:54:43 -07:00
}
2024-02-16 00:19:44 -08:00
#[ derive(Debug,Clone,Copy,Eq,Hash,PartialEq) ]
2024-08-09 14:16:10 -07:00
struct ContactCollision {
2024-01-29 22:37:05 -08:00
face_id :model_physics ::MinkowskiFace ,
2024-08-09 14:16:10 -07:00
model_id :ContactModelId ,
submesh_id :PhysicsSubmeshId ,
2023-09-08 15:54:22 -07:00
}
2024-08-08 15:54:23 -07:00
#[ derive(Debug,Clone,Copy,Eq,Hash,PartialEq) ]
2024-08-09 14:16:10 -07:00
struct IntersectCollision {
model_id :IntersectModelId ,
submesh_id :PhysicsSubmeshId ,
2023-11-30 01:51:17 -08:00
}
#[ derive(Debug,Clone,Eq,Hash,PartialEq) ]
2024-08-09 14:16:10 -07:00
enum Collision {
2023-11-30 01:51:17 -08:00
Contact ( ContactCollision ) ,
Intersect ( IntersectCollision ) ,
}
impl Collision {
2024-08-09 14:16:10 -07:00
const fn new ( convex_mesh_id :ConvexMeshId , face_id :model_physics ::MinkowskiFace ) ->Self {
match convex_mesh_id . model_id {
PhysicsModelId ::Contact ( model_id ) = > Collision ::Contact ( ContactCollision { model_id , submesh_id :convex_mesh_id . submesh_id , face_id } ) ,
PhysicsModelId ::Intersect ( model_id ) = > Collision ::Intersect ( IntersectCollision { model_id , submesh_id :convex_mesh_id . submesh_id } ) ,
2023-11-30 01:51:17 -08:00
}
2023-09-08 15:54:22 -07:00
}
}
2024-02-16 00:19:44 -08:00
#[ derive(Clone,Debug,Default) ]
2023-10-12 19:44:46 -07:00
struct TouchingState {
2025-01-07 05:59:38 -08:00
contacts :HashSet < ContactCollision > ,
intersects :HashSet < IntersectCollision > ,
2023-10-12 19:44:46 -07:00
}
impl TouchingState {
fn clear ( & mut self ) {
self . contacts . clear ( ) ;
self . intersects . clear ( ) ;
}
2023-11-30 01:51:17 -08:00
fn insert ( & mut self , collision :Collision ) ->bool {
match collision {
Collision ::Contact ( collision ) = > self . contacts . insert ( collision ) ,
Collision ::Intersect ( collision ) = > self . intersects . insert ( collision ) ,
}
2023-10-12 19:44:46 -07:00
}
2023-11-30 01:51:17 -08:00
fn remove ( & mut self , collision :& Collision ) ->bool {
match collision {
Collision ::Contact ( collision ) = > self . contacts . remove ( collision ) ,
Collision ::Intersect ( collision ) = > self . intersects . remove ( collision ) ,
}
2023-10-12 19:44:46 -07:00
}
2024-02-16 00:19:44 -08:00
fn base_acceleration ( & self , models :& PhysicsModels , style :& StyleModifiers , camera :& PhysicsCamera , input_state :& InputState ) ->Planar64Vec3 {
2023-11-30 01:51:17 -08:00
let mut a = style . gravity ;
2024-02-16 00:19:44 -08:00
if let Some ( rocket_settings ) = & style . rocket {
a + = rocket_settings . acceleration ( style . get_propulsion_control_dir ( camera , input_state . controls ) ) ;
2023-11-30 01:51:17 -08:00
}
//add accelerators
for contact in & self . contacts {
2024-08-09 14:16:10 -07:00
if let Some ( accelerator ) = & models . contact_attr ( contact . model_id ) . general . accelerator {
a + = accelerator . acceleration ;
2023-11-30 01:51:17 -08:00
}
}
for intersect in & self . intersects {
2024-08-09 14:16:10 -07:00
if let Some ( accelerator ) = & models . intersect_attr ( intersect . model_id ) . general . accelerator {
a + = accelerator . acceleration ;
2023-11-30 01:51:17 -08:00
}
}
2024-02-16 00:19:44 -08:00
//TODO: add water
2023-11-30 01:51:17 -08:00
a
}
2024-01-29 22:37:05 -08:00
fn constrain_velocity ( & self , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , velocity :& mut Planar64Vec3 ) {
2024-08-21 15:17:36 -07:00
let contacts = self . contacts . iter ( ) . map ( | contact | {
2024-01-29 22:37:05 -08:00
let n = contact_normal ( models , hitbox_mesh , contact ) ;
2024-08-21 15:17:36 -07:00
crate ::push_solve ::Contact {
2025-01-08 19:07:13 -08:00
position :vec3 ::ZERO ,
2024-08-21 15:17:36 -07:00
velocity :n ,
normal :n ,
2023-10-12 19:44:46 -07:00
}
2024-08-21 15:17:36 -07:00
} ) . collect ( ) ;
2025-01-09 05:56:11 -08:00
* velocity = crate ::push_solve ::push_solve ( & contacts , * velocity ) ;
2023-10-12 19:44:46 -07:00
}
2024-01-29 22:37:05 -08:00
fn constrain_acceleration ( & self , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , acceleration :& mut Planar64Vec3 ) {
2024-08-21 15:17:36 -07:00
let contacts = self . contacts . iter ( ) . map ( | contact | {
2024-01-29 22:37:05 -08:00
let n = contact_normal ( models , hitbox_mesh , contact ) ;
2024-08-21 15:17:36 -07:00
crate ::push_solve ::Contact {
2025-01-08 19:07:13 -08:00
position :vec3 ::ZERO ,
2024-08-21 15:17:36 -07:00
velocity :n ,
normal :n ,
2023-10-12 19:44:46 -07:00
}
2024-08-21 15:17:36 -07:00
} ) . collect ( ) ;
2025-01-09 05:56:11 -08:00
* acceleration = crate ::push_solve ::push_solve ( & contacts , * acceleration ) ;
2023-10-12 19:44:46 -07:00
}
2025-01-07 22:54:58 -08:00
fn predict_collision_end ( & self , collector :& mut instruction ::InstructionCollector < PhysicsInternalInstruction , TimeInner > , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , body :& Body , time :Time ) {
let relative_body = crate ::body ::VirtualBody ::relative ( & Body ::ZERO , body ) . body ( time ) ;
2023-11-30 01:51:17 -08:00
for contact in & self . contacts {
//detect face slide off
2024-08-09 14:16:10 -07:00
let model_mesh = models . contact_mesh ( contact ) ;
2024-01-29 22:37:05 -08:00
let minkowski = model_physics ::MinkowskiMesh ::minkowski_sum ( model_mesh , hitbox_mesh . transformed_mesh ( ) ) ;
collector . collect ( minkowski . predict_collision_face_out ( & relative_body , collector . time ( ) , contact . face_id ) . map ( | ( _face , time ) | {
2023-11-30 01:51:17 -08:00
TimedInstruction {
2024-09-11 15:58:13 -07:00
time :relative_body . time + time . into ( ) ,
2024-08-01 11:47:20 -07:00
instruction :PhysicsInternalInstruction ::CollisionEnd (
2024-09-11 15:58:13 -07:00
Collision ::Contact ( * contact ) ,
time
2023-11-30 01:51:17 -08:00
) ,
}
} ) ) ;
}
for intersect in & self . intersects {
//detect model collision in reverse
2024-08-09 14:16:10 -07:00
let model_mesh = models . intersect_mesh ( intersect ) ;
2024-01-29 22:37:05 -08:00
let minkowski = model_physics ::MinkowskiMesh ::minkowski_sum ( model_mesh , hitbox_mesh . transformed_mesh ( ) ) ;
collector . collect ( minkowski . predict_collision_out ( & relative_body , collector . time ( ) ) . map ( | ( _face , time ) | {
2023-11-30 01:51:17 -08:00
TimedInstruction {
2024-09-11 15:58:13 -07:00
time :relative_body . time + time . into ( ) ,
2024-08-01 11:47:20 -07:00
instruction :PhysicsInternalInstruction ::CollisionEnd (
2024-09-11 15:58:13 -07:00
Collision ::Intersect ( * intersect ) ,
time
2023-11-30 01:51:17 -08:00
) ,
}
} ) ) ;
2023-10-12 19:44:46 -07:00
}
}
}
2024-02-16 00:19:44 -08:00
#[ derive(Clone,Debug) ]
2024-01-29 22:37:05 -08:00
pub struct PhysicsState {
time :Time ,
body :Body ,
2024-08-20 15:53:43 -07:00
_world :WorldState , //currently there is only one state the world can be in
2024-01-29 22:37:05 -08:00
touching :TouchingState ,
//camera must exist in state because wormholes modify the camera, also camera punch
camera :PhysicsCamera ,
//input_state
2024-02-16 00:19:44 -08:00
input_state :InputState ,
2024-01-29 22:37:05 -08:00
//style
style :StyleModifiers , //mode style with custom style updates applied
//gameplay_state
mode_state :ModeState ,
move_state :MoveState ,
2024-08-01 09:29:13 -07:00
//run is non optional: when you spawn in a run is created
//the run cannot be finished unless you start it by visiting
//a start zone. If you change mode, a new run is created.
run :run ::Run ,
2024-01-29 22:37:05 -08:00
}
//random collection of contextual data that doesn't belong in PhysicsState
pub struct PhysicsData {
//permanent map data
bvh :bvh ::BvhNode < ConvexMeshId > ,
//transient map/environment data (open world loads/unloads parts of this data)
models :PhysicsModels ,
//semi-transient data
modes :gameplay_modes ::Modes ,
//cached calculations
hitbox_mesh :HitboxMesh ,
}
2023-10-04 20:04:04 -07:00
impl Default for PhysicsState {
2023-10-30 22:25:03 -07:00
fn default ( ) ->Self {
2023-10-04 20:04:04 -07:00
Self {
2024-09-11 15:58:13 -07:00
body :Body ::new ( vec3 ::int ( 0 , 50 , 0 ) , vec3 ::int ( 0 , 0 , 0 ) , vec3 ::int ( 0 , - 100 , 0 ) , Time ::ZERO ) ,
2023-10-30 22:25:03 -07:00
time :Time ::ZERO ,
2023-10-04 20:04:04 -07:00
style :StyleModifiers ::default ( ) ,
2023-10-12 19:44:46 -07:00
touching :TouchingState ::default ( ) ,
2024-08-01 09:29:13 -07:00
move_state :MoveState ::Air ,
2023-10-30 22:25:03 -07:00
camera :PhysicsCamera ::default ( ) ,
2024-02-16 00:19:44 -08:00
input_state :InputState ::default ( ) ,
2024-08-20 15:53:43 -07:00
_world :WorldState { } ,
2024-01-29 22:37:05 -08:00
mode_state :ModeState ::default ( ) ,
2024-08-01 09:29:13 -07:00
run :run ::Run ::new ( ) ,
2024-01-29 22:37:05 -08:00
}
}
}
impl Default for PhysicsData {
fn default ( ) ->Self {
Self {
bvh :bvh ::BvhNode ::default ( ) ,
models :Default ::default ( ) ,
modes :Default ::default ( ) ,
hitbox_mesh :StyleModifiers ::default ( ) . calculate_mesh ( ) ,
2023-10-04 20:04:04 -07:00
}
}
}
2024-08-01 11:47:20 -07:00
impl PhysicsState {
2024-02-16 00:19:44 -08:00
fn clear ( & mut self ) {
2023-10-12 19:44:46 -07:00
self . touching . clear ( ) ;
2023-10-04 20:04:04 -07:00
}
2024-08-01 12:57:58 -07:00
fn reset_to_default ( & mut self ) {
let mut new_state = Self ::default ( ) ;
new_state . camera . sensitivity = self . camera . sensitivity ;
* self = new_state ;
}
2025-01-07 22:54:58 -08:00
fn next_move_instruction ( & self ) ->Option < TimedInstruction < PhysicsInternalInstruction , TimeInner > > {
2024-02-16 00:19:44 -08:00
self . move_state . next_move_instruction ( & self . style . strafe , self . time )
}
2024-08-08 16:54:42 -07:00
fn cull_velocity ( & mut self , data :& PhysicsData , velocity :Planar64Vec3 ) {
self . move_state . cull_velocity ( velocity , & mut self . body , & mut self . touching , & data . models , & data . hitbox_mesh , & self . style , & self . camera , & self . input_state ) ;
2024-02-16 00:19:44 -08:00
}
fn set_move_state ( & mut self , data :& PhysicsData , move_state :MoveState ) {
2024-08-08 16:54:42 -07:00
self . move_state . set_move_state ( move_state , & mut self . body , & self . touching , & data . models , & data . hitbox_mesh , & self . style , & self . camera , & self . input_state ) ;
2024-02-16 00:19:44 -08:00
}
2024-08-08 16:54:42 -07:00
fn apply_input_and_body ( & mut self , data :& PhysicsData ) {
self . move_state . apply_input_and_body ( & mut self . body , & self . touching , & data . models , & data . hitbox_mesh , & self . style , & self . camera , & self . input_state ) ;
2023-09-08 11:33:20 -07:00
}
2023-09-08 20:13:30 -07:00
//state mutated on collision:
//Accelerator
//stair step-up
//state mutated on instruction
//change fly acceleration (fly_sustain)
//change fly velocity
//generic event emmiters
//PlatformStandTime
//walk/swim/air/ladder sounds
//VState?
//falling under the map
// fn next_respawn_instruction(&self) -> Option<TimedInstruction<PhysicsInstruction>> {
// if self.body.position<self.world.min_y {
// return Some(TimedInstruction{
// time:self.time,
// instruction:PhysicsInstruction::Trigger(None)
// });
// }
// }
// fn next_water_instruction(&self) -> Option<TimedInstruction<PhysicsInstruction>> {
// return Some(TimedInstruction{
// time:(self.time*self.strafe_tick_num/self.strafe_tick_den+1)*self.strafe_tick_den/self.strafe_tick_num,
// //only poll the physics if there is a before and after mouse event
// instruction:PhysicsInstruction::Water
// });
// }
2023-09-08 11:33:20 -07:00
}
2024-02-16 00:19:44 -08:00
2024-01-29 22:37:05 -08:00
#[ derive(Default) ]
pub struct PhysicsContext {
2024-02-16 00:19:44 -08:00
state :PhysicsState , //this captures the entire state of the physics.
2024-01-29 22:37:05 -08:00
data :PhysicsData , //data currently loaded into memory which is needded for physics to run, but is not part of the state.
}
2024-08-01 11:47:20 -07:00
//the physics consumes the generic PhysicsInstruction, but can only emit the more narrow PhysicsInternalInstruction
2025-01-07 22:54:58 -08:00
impl instruction ::InstructionConsumer for PhysicsContext {
type Instruction = PhysicsInstruction ;
type TimeInner = TimeInner ;
fn process_instruction ( & mut self , ins :TimedInstruction < PhysicsInstruction , TimeInner > ) {
2024-01-29 22:37:05 -08:00
atomic_state_update ( & mut self . state , & self . data , ins )
}
}
2025-01-07 22:54:58 -08:00
impl instruction ::InstructionEmitter for PhysicsContext {
type Instruction = PhysicsInternalInstruction ;
type TimeInner = TimeInner ;
2025-01-07 05:59:31 -08:00
//this little next instruction function could cache its return value and invalidate the cached value by watching the State.
2025-01-07 22:54:58 -08:00
fn next_instruction ( & self , time_limit :Time ) ->Option < TimedInstruction < PhysicsInternalInstruction , TimeInner > > {
2024-08-01 11:47:20 -07:00
next_instruction_internal ( & self . state , & self . data , time_limit )
2024-01-29 22:37:05 -08:00
}
}
impl PhysicsContext {
2024-08-20 16:24:28 -07:00
pub fn camera_body ( & self ) ->Body {
Body {
position :self . state . body . position + self . state . style . camera_offset ,
.. self . state . body
2024-02-16 00:19:44 -08:00
}
}
2024-08-20 16:24:28 -07:00
pub const fn camera ( & self ) ->PhysicsCamera {
self . state . camera
}
2024-02-16 00:19:44 -08:00
pub const fn get_next_mouse ( & self ) ->& MouseState {
self . state . input_state . get_next_mouse ( )
}
2024-08-06 11:26:27 -07:00
/// use with caution, this is the only non-instruction way to mess with physics
2024-01-29 22:37:05 -08:00
pub fn generate_models ( & mut self , map :& map ::CompleteMap ) {
2024-08-06 11:26:27 -07:00
self . state . clear ( ) ;
2024-08-09 14:16:10 -07:00
let mut modes = map . modes . clone ( ) ;
for mode in & mut modes . modes {
2024-08-01 09:29:09 -07:00
mode . denormalize_data ( ) ;
}
2024-08-09 14:16:10 -07:00
let mut used_contact_attributes = Vec ::new ( ) ;
let mut used_intersect_attributes = Vec ::new ( ) ;
//temporary type for type safety lol
#[ derive(Clone,Copy,Hash,Eq,PartialEq) ]
enum PhysicsAttributesId {
Contact ( ContactAttributesId ) ,
Intersect ( IntersectAttributesId ) ,
}
let mut contact_models = HashMap ::new ( ) ;
let mut intersect_models = HashMap ::new ( ) ;
2024-01-29 22:37:05 -08:00
let mut physics_attr_id_from_model_attr_id = HashMap ::< CollisionAttributesId , PhysicsAttributesId > ::new ( ) ;
let mut used_meshes = Vec ::new ( ) ;
let mut physics_mesh_id_from_model_mesh_id = HashMap ::< MeshId , PhysicsMeshId > ::new ( ) ;
2024-08-09 14:16:10 -07:00
for ( model_id , model ) in map . models . iter ( ) . enumerate ( ) {
2024-02-16 06:04:24 -08:00
//TODO: use .entry().or_insert_with(||{
2024-01-29 22:37:05 -08:00
let attr_id = if let Some ( & attr_id ) = physics_attr_id_from_model_attr_id . get ( & model . attributes ) {
attr_id
} else {
//check if it's real
match map . attributes . get ( model . attributes . get ( ) as usize ) . and_then ( | m_attr | {
PhysicsCollisionAttributes ::try_from ( m_attr ) . map_or ( None , | p_attr | {
2024-08-09 14:16:10 -07:00
let attr_id = match p_attr {
PhysicsCollisionAttributes ::Contact ( attr ) = > {
let attr_id = ContactAttributesId ::new ( used_contact_attributes . len ( ) as u32 ) ;
used_contact_attributes . push ( attr ) ;
PhysicsAttributesId ::Contact ( attr_id )
} ,
PhysicsCollisionAttributes ::Intersect ( attr ) = > {
let attr_id = IntersectAttributesId ::new ( used_intersect_attributes . len ( ) as u32 ) ;
used_intersect_attributes . push ( attr ) ;
PhysicsAttributesId ::Intersect ( attr_id )
} ,
} ;
2024-01-29 22:37:05 -08:00
physics_attr_id_from_model_attr_id . insert ( model . attributes , attr_id ) ;
Some ( attr_id )
} )
} ) {
Some ( attr_id ) = > attr_id ,
2024-08-09 14:16:10 -07:00
None = > continue ,
2024-01-29 22:37:05 -08:00
}
} ;
let mesh_id = if let Some ( & mesh_id ) = physics_mesh_id_from_model_mesh_id . get ( & model . mesh ) {
mesh_id
} else {
match map . meshes . get ( model . mesh . get ( ) as usize ) . and_then ( | mesh | {
2024-02-13 23:30:05 -08:00
match PhysicsMesh ::try_from ( mesh ) {
Ok ( physics_mesh ) = > {
let mesh_id = PhysicsMeshId ::new ( used_meshes . len ( ) as u32 ) ;
used_meshes . push ( physics_mesh ) ;
physics_mesh_id_from_model_mesh_id . insert ( model . mesh , mesh_id ) ;
Some ( mesh_id )
} ,
Err ( e ) = > {
println! ( " Failed to build PhysicsMesh: {e} " ) ;
None
}
}
2024-01-29 22:37:05 -08:00
} ) {
Some ( mesh_id ) = > mesh_id ,
2024-08-09 14:16:10 -07:00
None = > continue ,
2024-01-29 22:37:05 -08:00
}
} ;
2024-08-09 14:16:10 -07:00
let transform = PhysicsMeshTransform ::new ( model . transform ) ;
match attr_id {
PhysicsAttributesId ::Contact ( attr_id ) = > {
contact_models . insert ( ContactModelId ::new ( model_id as u32 ) , ContactModel {
mesh_id ,
attr_id ,
transform ,
} ) ;
} ,
PhysicsAttributesId ::Intersect ( attr_id ) = > {
intersect_models . insert ( IntersectModelId ::new ( model_id as u32 ) , IntersectModel {
mesh_id ,
attr_id ,
transform ,
} ) ;
} ,
}
}
let meshes :HashMap < PhysicsMeshId , PhysicsMesh > = used_meshes . into_iter ( )
. enumerate ( )
. map ( | ( mesh_id , mesh ) |
( PhysicsMeshId ::new ( mesh_id as u32 ) , mesh )
) . collect ( ) ;
let convex_mesh_aabb_list =
//map the two lists into a single type so they can be processed with one closure
contact_models . iter ( ) . map ( | ( & model_id , model ) |
( PhysicsModelId ::Contact ( model_id ) , & model . mesh_id , & model . transform )
) . chain ( intersect_models . iter ( ) . map ( | ( & model_id , model ) |
( PhysicsModelId ::Intersect ( model_id ) , & model . mesh_id , & model . transform )
) )
. flat_map ( | ( model_id , mesh_id , transform ) | {
meshes [ mesh_id ] . submesh_views ( )
2024-01-29 22:37:05 -08:00
. enumerate ( ) . map ( move | ( submesh_id , view ) | {
let mut aabb = aabb ::Aabb ::default ( ) ;
2024-08-09 14:16:10 -07:00
let transformed_mesh = TransformedMesh ::new ( view , transform ) ;
2024-01-29 22:37:05 -08:00
for v in transformed_mesh . verts ( ) {
2024-09-11 15:58:13 -07:00
aabb . grow ( v . fix_1 ( ) ) ;
2024-01-29 22:37:05 -08:00
}
( ConvexMeshId {
model_id ,
submesh_id :PhysicsSubmeshId ::new ( submesh_id as u32 ) ,
} , aabb )
} )
} ) . collect ( ) ;
2024-08-09 14:16:10 -07:00
let bvh = bvh ::generate_bvh ( convex_mesh_aabb_list ) ;
let model_count = contact_models . len ( ) + intersect_models . len ( ) ;
let models = PhysicsModels {
meshes ,
contact_models ,
intersect_models ,
contact_attributes :used_contact_attributes . into_iter ( )
. enumerate ( )
. map ( | ( attr_id , attr ) |
( ContactAttributesId ::new ( attr_id as u32 ) , attr )
) . collect ( ) ,
intersect_attributes :used_intersect_attributes . into_iter ( )
. enumerate ( )
. map ( | ( attr_id , attr ) |
( IntersectAttributesId ::new ( attr_id as u32 ) , attr )
) . collect ( ) ,
} ;
self . data . bvh = bvh ;
self . data . models = models ;
self . data . modes = modes ;
//hitbox_mesh is unchanged
println! ( " Physics Objects: {} " , model_count ) ;
2024-01-29 22:37:05 -08:00
}
//tickless gaming
2024-08-01 11:47:20 -07:00
fn run_internal_exhaustive ( & mut self , time_limit :Time ) {
2024-01-29 22:37:05 -08:00
//prepare is ommitted - everything is done via instructions.
while let Some ( instruction ) = self . next_instruction ( time_limit ) { //collect
//process
2024-08-01 11:47:20 -07:00
self . process_instruction ( TimedInstruction {
time :instruction . time ,
instruction :PhysicsInstruction ::Internal ( instruction . instruction ) ,
} ) ;
2024-01-29 22:37:05 -08:00
//write hash lol
}
}
2025-01-07 22:54:58 -08:00
pub fn run_input_instruction ( & mut self , instruction :TimedInstruction < PhysicsInputInstruction , TimeInner > ) {
2024-08-01 11:47:20 -07:00
self . run_internal_exhaustive ( instruction . time ) ;
2024-02-16 00:19:44 -08:00
self . process_instruction ( TimedInstruction {
time :instruction . time ,
instruction :PhysicsInstruction ::Input ( instruction . instruction ) ,
} ) ;
2024-01-29 22:37:05 -08:00
}
2024-02-16 00:19:44 -08:00
}
2024-01-29 22:37:05 -08:00
2024-08-01 11:47:20 -07:00
//this is the one who asks
2025-01-07 22:54:58 -08:00
fn next_instruction_internal ( state :& PhysicsState , data :& PhysicsData , time_limit :Time ) ->Option < TimedInstruction < PhysicsInternalInstruction , TimeInner > > {
2023-09-08 11:33:20 -07:00
//JUST POLLING!!! NO MUTATION
2024-01-29 16:19:41 -08:00
let mut collector = instruction ::InstructionCollector ::new ( time_limit ) ;
2023-11-30 01:51:17 -08:00
2024-01-29 22:37:05 -08:00
collector . collect ( state . next_move_instruction ( ) ) ;
2023-11-30 01:51:17 -08:00
//check for collision ends
2024-01-29 22:37:05 -08:00
state . touching . predict_collision_end ( & mut collector , & data . models , & data . hitbox_mesh , & state . body , state . time ) ;
2023-11-30 01:51:17 -08:00
//check for collision starts
2024-01-29 16:19:41 -08:00
let mut aabb = aabb ::Aabb ::default ( ) ;
2024-01-29 22:37:05 -08:00
state . body . grow_aabb ( & mut aabb , state . time , collector . time ( ) ) ;
aabb . inflate ( data . hitbox_mesh . halfsize ) ;
2024-08-06 14:15:57 -07:00
//relative to moving platforms
//let relative_body=&VirtualBody::relative(&Body::default(),&state.body).body(state.time);
let relative_body = & state . body ;
2024-07-25 13:53:45 -07:00
data . bvh . the_tester ( & aabb , & mut | & convex_mesh_id | {
2023-11-30 01:51:17 -08:00
//no checks are needed because of the time limits.
2024-01-29 22:37:05 -08:00
let model_mesh = data . models . mesh ( convex_mesh_id ) ;
let minkowski = model_physics ::MinkowskiMesh ::minkowski_sum ( model_mesh , data . hitbox_mesh . transformed_mesh ( ) ) ;
2024-08-06 14:15:57 -07:00
collector . collect ( minkowski . predict_collision_in ( relative_body , collector . time ( ) )
2023-11-30 01:51:17 -08:00
//temp (?) code to avoid collision loops
2024-09-11 15:58:13 -07:00
. map_or ( None , | ( face , dt ) | {
let time = relative_body . time + dt . into ( ) ;
if time < = state . time { None } else { Some ( ( time , face , dt ) ) } } )
. map ( | ( time , face , dt ) |
2024-08-09 14:16:10 -07:00
TimedInstruction {
time ,
instruction :PhysicsInternalInstruction ::CollisionStart (
2024-09-11 15:58:13 -07:00
Collision ::new ( convex_mesh_id , face ) ,
dt
2024-08-09 14:16:10 -07:00
)
}
)
) ;
2023-10-05 23:52:04 -07:00
} ) ;
2023-09-08 20:12:58 -07:00
collector . instruction ( )
2023-09-08 11:33:20 -07:00
}
2023-09-08 17:15:49 -07:00
2023-11-30 01:51:17 -08:00
2024-01-29 22:37:05 -08:00
fn contact_normal ( models :& PhysicsModels , hitbox_mesh :& HitboxMesh , contact :& ContactCollision ) ->Planar64Vec3 {
2024-08-09 14:16:10 -07:00
let model_mesh = models . contact_mesh ( contact ) ;
2024-01-29 22:37:05 -08:00
let minkowski = model_physics ::MinkowskiMesh ::minkowski_sum ( model_mesh , hitbox_mesh . transformed_mesh ( ) ) ;
2024-09-11 15:58:13 -07:00
// TODO: normalize to i64::MAX>>1
minkowski . face_nd ( contact . face_id ) . 0. fix_1 ( )
2023-11-30 01:51:17 -08:00
}
2024-08-08 16:54:42 -07:00
fn recalculate_touching (
move_state :& mut MoveState ,
body :& mut Body ,
touching :& mut TouchingState ,
run :& mut run ::Run ,
mode_state :& mut ModeState ,
mode :Option < & gameplay_modes ::Mode > ,
models :& PhysicsModels ,
hitbox_mesh :& HitboxMesh ,
bvh :& bvh ::BvhNode < ConvexMeshId > ,
style :& StyleModifiers ,
camera :& PhysicsCamera ,
input_state :& InputState ,
time :Time ,
) {
//collision_end all existing contacts
//I would have preferred while let Some(contact)=contacts.pop()
//but there is no such method
while let Some ( & contact ) = touching . contacts . iter ( ) . next ( ) {
collision_end_contact ( move_state , body , touching , models , hitbox_mesh , style , camera , input_state , models . contact_attr ( contact . model_id ) , contact )
}
while let Some ( & intersect ) = touching . intersects . iter ( ) . next ( ) {
collision_end_intersect ( touching , mode , run , models . intersect_attr ( intersect . model_id ) , intersect , time ) ;
}
//find all models in the teleport region
let mut aabb = aabb ::Aabb ::default ( ) ;
aabb . grow ( body . position ) ;
aabb . inflate ( hitbox_mesh . halfsize ) ;
//relative to moving platforms
//let relative_body=&VirtualBody::relative(&Body::default(),&state.body).body(state.time);
bvh . the_tester ( & aabb , & mut | & convex_mesh_id | {
//no checks are needed because of the time limits.
let model_mesh = models . mesh ( convex_mesh_id ) ;
let minkowski = model_physics ::MinkowskiMesh ::minkowski_sum ( model_mesh , hitbox_mesh . transformed_mesh ( ) ) ;
if minkowski . is_point_in_mesh ( body . position ) {
match convex_mesh_id . model_id {
//being inside of contact objects is an invalid physics state
//but the physics isn't advanced enough to do anything about it yet
//TODO: PushSolve and search for the closest valid position
PhysicsModelId ::Contact ( _ ) = > ( ) ,
PhysicsModelId ::Intersect ( model_id ) = >
collision_start_intersect ( move_state , body , mode_state , touching , mode , run , models , hitbox_mesh , bvh , style , camera , input_state ,
models . intersect_attr ( model_id ) ,
IntersectCollision {
model_id ,
submesh_id :convex_mesh_id . submesh_id ,
} ,
time ,
) ,
}
}
} ) ;
}
fn set_position (
point :Planar64Vec3 ,
move_state :& mut MoveState ,
body :& mut Body ,
touching :& mut TouchingState ,
run :& mut run ::Run ,
mode_state :& mut ModeState ,
mode :Option < & gameplay_modes ::Mode > ,
models :& PhysicsModels ,
hitbox_mesh :& HitboxMesh ,
bvh :& bvh ::BvhNode < ConvexMeshId > ,
style :& StyleModifiers ,
camera :& PhysicsCamera ,
input_state :& InputState ,
time :Time ,
) ->Planar64Vec3 {
2023-11-30 01:51:17 -08:00
//test intersections at new position
//hovering above the surface 0 units is not intersecting. you will fall into it just fine
2023-10-18 16:18:17 -07:00
body . position = point ;
2024-08-08 16:54:42 -07:00
//calculate contacts and determine the actual state
recalculate_touching ( move_state , body , touching , run , mode_state , mode , models , hitbox_mesh , bvh , style , camera , input_state , time ) ;
2023-11-30 01:51:17 -08:00
point
}
2024-01-29 22:37:05 -08:00
fn set_velocity_cull ( body :& mut Body , touching :& mut TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , v :Planar64Vec3 ) ->bool {
2023-11-30 01:51:17 -08:00
//This is not correct but is better than what I have
let mut culled = false ;
touching . contacts . retain ( | contact | {
2024-01-29 22:37:05 -08:00
let n = contact_normal ( models , hitbox_mesh , contact ) ;
2024-09-11 15:58:13 -07:00
let r = n . dot ( v ) . is_positive ( ) ;
if r {
2023-11-30 01:51:17 -08:00
culled = true ;
println! ( " set_velocity_cull contact= {:?} " , contact ) ;
}
2024-09-11 15:58:13 -07:00
! r
2023-11-30 01:51:17 -08:00
} ) ;
2024-01-29 22:37:05 -08:00
set_velocity ( body , touching , models , hitbox_mesh , v ) ;
2023-11-30 01:51:17 -08:00
culled
}
2024-02-16 00:19:44 -08:00
fn set_velocity ( body :& mut Body , touching :& TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , mut v :Planar64Vec3 ) {
2024-01-29 22:37:05 -08:00
touching . constrain_velocity ( models , hitbox_mesh , & mut v ) ;
2023-11-30 01:51:17 -08:00
body . velocity = v ;
}
2024-01-29 22:37:05 -08:00
fn set_acceleration_cull ( body :& mut Body , touching :& mut TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , a :Planar64Vec3 ) ->bool {
2023-11-30 01:51:17 -08:00
//This is not correct but is better than what I have
let mut culled = false ;
touching . contacts . retain ( | contact | {
2024-01-29 22:37:05 -08:00
let n = contact_normal ( models , hitbox_mesh , contact ) ;
2024-09-11 15:58:13 -07:00
let r = n . dot ( a ) . is_positive ( ) ;
if r {
2023-11-30 01:51:17 -08:00
culled = true ;
println! ( " set_acceleration_cull contact= {:?} " , contact ) ;
}
2024-09-11 15:58:13 -07:00
! r
2023-11-30 01:51:17 -08:00
} ) ;
2024-01-29 22:37:05 -08:00
set_acceleration ( body , touching , models , hitbox_mesh , a ) ;
2023-11-30 01:51:17 -08:00
culled
}
2024-02-16 00:19:44 -08:00
fn set_acceleration ( body :& mut Body , touching :& TouchingState , models :& PhysicsModels , hitbox_mesh :& HitboxMesh , mut a :Planar64Vec3 ) {
2024-01-29 22:37:05 -08:00
touching . constrain_acceleration ( models , hitbox_mesh , & mut a ) ;
2023-11-30 01:51:17 -08:00
body . acceleration = a ;
}
2024-08-08 16:54:42 -07:00
fn teleport (
point :Planar64Vec3 ,
move_state :& mut MoveState ,
body :& mut Body ,
touching :& mut TouchingState ,
run :& mut run ::Run ,
mode_state :& mut ModeState ,
mode :Option < & gameplay_modes ::Mode > ,
models :& PhysicsModels ,
hitbox_mesh :& HitboxMesh ,
bvh :& bvh ::BvhNode < ConvexMeshId > ,
style :& StyleModifiers ,
camera :& PhysicsCamera ,
input_state :& InputState ,
time :Time ,
) {
set_position ( point , move_state , body , touching , run , mode_state , mode , models , hitbox_mesh , bvh , style , camera , input_state , time ) ;
2024-01-29 22:37:05 -08:00
set_acceleration ( body , touching , models , hitbox_mesh , style . gravity ) ;
2023-10-18 16:18:17 -07:00
}
2024-08-08 16:54:42 -07:00
enum TeleportToSpawnError {
NoModel ,
}
fn teleport_to_spawn (
2025-01-06 20:25:53 -08:00
spawn_model_id :ModelId ,
2024-08-08 16:54:42 -07:00
move_state :& mut MoveState ,
body :& mut Body ,
touching :& mut TouchingState ,
run :& mut run ::Run ,
mode_state :& mut ModeState ,
mode :& gameplay_modes ::Mode ,
models :& PhysicsModels ,
hitbox_mesh :& HitboxMesh ,
bvh :& bvh ::BvhNode < ConvexMeshId > ,
style :& StyleModifiers ,
camera :& PhysicsCamera ,
input_state :& InputState ,
time :Time ,
) ->Result < ( ) , TeleportToSpawnError > {
2025-01-06 21:50:43 -08:00
//jump count and checkpoints are always reset on teleport_to_spawn.
//Map makers are expected to use tools to prevent
//multi-boosting on JumpLimit boosters such as spawning into a SetVelocity
mode_state . clear ( ) ;
2024-09-11 15:58:13 -07:00
const EPSILON :Planar64 = Planar64 ::raw ( ( 1 < < 32 ) / 16 ) ;
2025-01-06 20:25:53 -08:00
let transform = models . get_model_transform ( spawn_model_id ) . ok_or ( TeleportToSpawnError ::NoModel ) ? ;
2024-09-11 15:58:13 -07:00
//TODO: transform.vertex.matrix3.col(1)+transform.vertex.translation
let point = transform . vertex . transform_point3 ( vec3 ::Y ) . fix_1 ( ) + Planar64Vec3 ::new ( [ Planar64 ::ZERO , style . hitbox . halfsize . y + EPSILON , Planar64 ::ZERO ] ) ;
2024-08-08 16:54:42 -07:00
teleport ( point , move_state , body , touching , run , mode_state , Some ( mode ) , models , hitbox_mesh , bvh , style , camera , input_state , time ) ;
Ok ( ( ) )
2023-10-28 16:38:16 -07:00
}
2023-10-18 16:18:17 -07:00
2025-01-06 20:25:53 -08:00
struct CheckpointCheckOutcome {
set_stage :Option < StageId > ,
teleport_to_model :Option < ModelId > ,
}
2025-01-07 05:59:31 -08:00
// stage_element.touch_result(mode,mode_state)
2025-01-06 20:25:53 -08:00
fn checkpoint_check (
mode_state :& ModeState ,
stage_element :& gameplay_modes ::StageElement ,
mode :& gameplay_modes ::Mode ,
) ->CheckpointCheckOutcome {
2025-01-07 06:00:48 -08:00
let current_stage_id = mode_state . get_stage_id ( ) ;
let target_stage_id = stage_element . stage_id ( ) ;
if current_stage_id < target_stage_id {
2025-01-06 20:25:53 -08:00
//checkpoint check
//check if current stage is complete
2025-01-07 06:00:48 -08:00
if let Some ( current_stage ) = mode . get_stage ( current_stage_id ) {
2025-01-06 20:25:53 -08:00
if ! current_stage . is_complete ( mode_state . ordered_checkpoint_count ( ) , mode_state . unordered_checkpoint_count ( ) ) {
return CheckpointCheckOutcome {
set_stage :None ,
teleport_to_model :Some ( current_stage . spawn ( ) ) ,
} ;
}
}
//check if all between stages have no checkpoints required to pass them
2025-01-07 06:00:48 -08:00
for stage_id in current_stage_id . get ( ) + 1 .. target_stage_id . get ( ) {
2025-01-06 20:25:53 -08:00
let stage_id = StageId ::new ( stage_id ) ;
//check if none of the between stages has checkpoints, if they do teleport back to that stage
match mode . get_stage ( stage_id ) {
Some ( stage ) = > if ! stage . is_empty ( ) {
return CheckpointCheckOutcome {
set_stage :Some ( stage_id ) ,
teleport_to_model :Some ( stage . spawn ( ) ) ,
} ;
} ,
//no such stage! set to last existing stage
None = > return CheckpointCheckOutcome {
set_stage :Some ( StageId ::new ( stage_id . get ( ) - 1 ) ) ,
teleport_to_model :None ,
} ,
}
} ;
//notably you do not get teleported for touching ordered checkpoints in the wrong order within the same stage.
return CheckpointCheckOutcome {
2025-01-07 06:00:48 -08:00
set_stage :Some ( target_stage_id ) ,
2025-01-06 20:25:53 -08:00
teleport_to_model :None ,
} ;
} else if stage_element . force ( ) {
//forced stage_element will set the stage_id even if the stage has already been passed
return CheckpointCheckOutcome {
2025-01-07 06:00:48 -08:00
set_stage :Some ( target_stage_id ) ,
2025-01-06 20:25:53 -08:00
teleport_to_model :None ,
} ;
}
CheckpointCheckOutcome {
set_stage :None ,
teleport_to_model :None ,
}
}
2024-08-09 14:16:10 -07:00
fn run_teleport_behaviour (
2024-08-08 16:54:42 -07:00
model_id :ModelId ,
wormhole :Option < & gameplay_attributes ::Wormhole > ,
mode :Option < & gameplay_modes ::Mode > ,
move_state :& mut MoveState ,
body :& mut Body ,
touching :& mut TouchingState ,
run :& mut run ::Run ,
mode_state :& mut ModeState ,
2024-08-09 14:16:10 -07:00
models :& PhysicsModels ,
hitbox_mesh :& HitboxMesh ,
2024-08-08 16:54:42 -07:00
bvh :& bvh ::BvhNode < ConvexMeshId > ,
style :& StyleModifiers ,
camera :& PhysicsCamera ,
input_state :& InputState ,
time :Time ,
) {
if let Some ( mode ) = mode {
2025-01-06 20:25:53 -08:00
if let Some ( stage_element ) = mode . get_element ( model_id ) {
if let Some ( stage ) = mode . get_stage ( stage_element . stage_id ( ) ) {
let CheckpointCheckOutcome { set_stage , teleport_to_model } = checkpoint_check ( mode_state , stage_element , mode ) ;
if let Some ( stage_id ) = set_stage {
mode_state . set_stage_id ( stage_id ) ;
2024-08-08 16:54:42 -07:00
}
2025-01-06 20:25:53 -08:00
if let Some ( model_id ) = teleport_to_model {
let _ = teleport_to_spawn ( model_id , move_state , body , touching , run , mode_state , mode , models , hitbox_mesh , bvh , style , camera , input_state , time ) ;
2024-08-08 16:54:42 -07:00
return ;
2025-01-06 20:25:53 -08:00
}
match stage_element . behaviour ( ) {
gameplay_modes ::StageElementBehaviour ::SpawnAt = > ( ) ,
gameplay_modes ::StageElementBehaviour ::Trigger
| gameplay_modes ::StageElementBehaviour ::Teleport = > if let Some ( mode_state_stage ) = mode . get_stage ( mode_state . get_stage_id ( ) ) {
//I guess this is correct behaviour when trying to teleport to a non-existent spawn but it's still weird
let _ = teleport_to_spawn ( mode_state_stage . spawn ( ) , move_state , body , touching , run , mode_state , mode , models , hitbox_mesh , bvh , style , camera , input_state , time ) ;
return ;
} ,
gameplay_modes ::StageElementBehaviour ::Platform = > ( ) ,
gameplay_modes ::StageElementBehaviour ::Check = > ( ) , //this is to run the checkpoint check behaviour without any other side effects
gameplay_modes ::StageElementBehaviour ::Checkpoint = > {
//each of these checks if the model is actually a valid respective checkpoint object
//accumulate sequential ordered checkpoints
mode_state . accumulate_ordered_checkpoint ( & stage , model_id ) ;
//insert model id in accumulated unordered checkpoints
mode_state . accumulate_unordered_checkpoint ( & stage , model_id ) ;
} ,
}
2023-10-18 16:18:17 -07:00
}
2024-01-29 22:37:05 -08:00
}
}
2024-08-08 16:54:42 -07:00
if let Some ( & gameplay_attributes ::Wormhole { destination_model } ) = wormhole {
if let ( Some ( origin ) , Some ( destination ) ) = ( models . get_model_transform ( model_id ) , models . get_model_transform ( destination_model ) ) {
let point = body . position - origin . vertex . translation + destination . vertex . translation ;
//TODO: camera angles
teleport ( point , move_state , body , touching , run , mode_state , mode , models , hitbox_mesh , bvh , style , camera , input_state , time ) ;
2023-10-18 16:18:17 -07:00
}
}
}
2025-01-06 21:05:25 -08:00
fn not_spawn_at (
mode :Option < & gameplay_modes ::Mode > ,
model_id :ModelId ,
) ->bool {
if let Some ( mode ) = mode {
if let Some ( stage_element ) = mode . get_element ( model_id ) {
return stage_element . behaviour ( ) ! = gameplay_modes ::StageElementBehaviour ::SpawnAt ;
}
}
true
}
2024-08-08 15:54:23 -07:00
fn collision_start_contact (
2024-08-08 16:54:42 -07:00
move_state :& mut MoveState ,
body :& mut Body ,
mode_state :& mut ModeState ,
touching :& mut TouchingState ,
run :& mut run ::Run ,
mode :Option < & gameplay_modes ::Mode > ,
models :& PhysicsModels ,
hitbox_mesh :& HitboxMesh ,
bvh :& bvh ::BvhNode < ConvexMeshId > ,
style :& StyleModifiers ,
camera :& PhysicsCamera ,
input_state :& InputState ,
2024-08-09 14:16:10 -07:00
attr :& gameplay_attributes ::ContactAttributes ,
2024-08-08 15:54:23 -07:00
contact :ContactCollision ,
2024-08-08 16:54:42 -07:00
time :Time ,
2024-08-08 15:54:23 -07:00
) {
2024-08-08 16:54:42 -07:00
let incident_velocity = body . velocity ;
//add to touching
touching . insert ( Collision ::Contact ( contact ) ) ;
//clip v
set_velocity ( body , touching , models , hitbox_mesh , incident_velocity ) ;
2025-01-06 21:45:42 -08:00
let mut allow_jump = true ;
2025-01-06 21:05:25 -08:00
let model_id = contact . model_id . into ( ) ;
let mut allow_run_teleport_behaviour = not_spawn_at ( mode , model_id ) ;
2024-08-08 16:54:42 -07:00
match & attr . contacting . contact_behaviour {
Some ( gameplay_attributes ::ContactingBehaviour ::Surf ) = > println! ( " I'm surfing! " ) ,
Some ( gameplay_attributes ::ContactingBehaviour ::Cling ) = > println! ( " Unimplemented! " ) ,
& Some ( gameplay_attributes ::ContactingBehaviour ::Elastic ( elasticity ) ) = > {
2024-09-11 15:58:13 -07:00
let reflected_velocity = body . velocity + ( ( body . velocity - incident_velocity ) * Planar64 ::raw ( elasticity as i64 + 1 ) ) . fix_1 ( ) ;
2024-08-08 16:54:42 -07:00
set_velocity ( body , touching , models , hitbox_mesh , reflected_velocity ) ;
} ,
Some ( gameplay_attributes ::ContactingBehaviour ::Ladder ( contacting_ladder ) ) = >
if let Some ( ladder_settings ) = & style . ladder {
if contacting_ladder . sticky {
//kill v
//actually you could do this with a booster attribute :thinking:
//it's a little bit different because maybe you want to chain ladders together
2024-09-11 15:58:13 -07:00
set_velocity ( body , touching , models , hitbox_mesh , vec3 ::ZERO ) ; //model.velocity
2024-08-08 16:54:42 -07:00
}
//ladder walkstate
let ( gravity , target_velocity ) = ladder_things ( ladder_settings , & contact , touching , models , hitbox_mesh , style , camera , input_state ) ;
let walk_state = ContactMoveState ::ladder ( ladder_settings , body , gravity , target_velocity , contact ) ;
move_state . set_move_state ( MoveState ::Ladder ( walk_state ) , body , touching , models , hitbox_mesh , style , camera , input_state ) ;
} ,
2025-01-06 21:45:42 -08:00
Some ( gameplay_attributes ::ContactingBehaviour ::NoJump ) = > allow_jump = false ,
2024-08-08 16:54:42 -07:00
None = > if let Some ( walk_settings ) = & style . walk {
2024-09-11 15:58:13 -07:00
if walk_settings . is_slope_walkable ( contact_normal ( models , hitbox_mesh , & contact ) , vec3 ::Y ) {
2025-01-06 21:05:25 -08:00
allow_run_teleport_behaviour = true ;
2024-08-08 16:54:42 -07:00
//ground
let ( gravity , target_velocity ) = ground_things ( walk_settings , & contact , touching , models , hitbox_mesh , style , camera , input_state ) ;
let walk_state = ContactMoveState ::ground ( walk_settings , body , gravity , target_velocity , contact ) ;
move_state . set_move_state ( MoveState ::Walk ( walk_state ) , body , touching , models , hitbox_mesh , style , camera , input_state ) ;
}
} ,
}
//I love making functions with 10 arguments to dodge the borrow checker
2025-01-06 21:05:25 -08:00
if allow_run_teleport_behaviour {
run_teleport_behaviour ( model_id , attr . general . wormhole . as_ref ( ) , mode , move_state , body , touching , run , mode_state , models , hitbox_mesh , bvh , style , camera , input_state , time ) ;
}
2025-01-06 21:45:42 -08:00
if allow_jump & & style . get_control ( Controls ::Jump , input_state . controls ) {
2024-08-08 16:54:42 -07:00
if let ( Some ( jump_settings ) , Some ( walk_state ) ) = ( & style . jump , move_state . get_walk_state ( ) ) {
2025-01-06 21:45:42 -08:00
let mut exceeded_jump_limit = false ;
if let Some ( mode ) = mode {
if let Some ( stage_element ) = mode . get_element ( model_id ) {
if ! mode_state . try_increment_jump_count ( model_id , stage_element . jump_limit ( ) ) . is_allowed ( ) {
exceeded_jump_limit = true ;
}
}
}
if exceeded_jump_limit {
if let Some ( mode ) = mode {
if let Some ( spawn_model_id ) = mode . get_spawn_model_id ( mode_state . get_stage_id ( ) ) {
let _ = teleport_to_spawn ( spawn_model_id , move_state , body , touching , run , mode_state , mode , models , hitbox_mesh , bvh , style , camera , input_state , time ) ;
}
}
} else {
let jump_dir = walk_state . jump_direction . direction ( models , hitbox_mesh , & walk_state . contact ) ;
let jumped_velocity = jump_settings . jumped_velocity ( style , jump_dir , body . velocity , attr . general . booster . as_ref ( ) ) ;
move_state . cull_velocity ( jumped_velocity , body , touching , models , hitbox_mesh , style , camera , input_state ) ;
}
2024-08-08 16:54:42 -07:00
}
}
match & attr . general . trajectory {
Some ( trajectory ) = > {
match trajectory {
gameplay_attributes ::SetTrajectory ::AirTime ( _ ) = > todo! ( ) ,
gameplay_attributes ::SetTrajectory ::Height ( _ ) = > todo! ( ) ,
gameplay_attributes ::SetTrajectory ::TargetPointTime { target_point : _ , time : _ } = > todo! ( ) ,
gameplay_attributes ::SetTrajectory ::TargetPointSpeed { target_point : _ , speed : _ , trajectory_choice : _ } = > todo! ( ) ,
& gameplay_attributes ::SetTrajectory ::Velocity ( velocity ) = > {
move_state . cull_velocity ( velocity , body , touching , models , hitbox_mesh , style , camera , input_state ) ;
} ,
gameplay_attributes ::SetTrajectory ::DotVelocity { direction : _ , dot : _ } = > todo! ( ) ,
}
} ,
None = > ( ) ,
}
//doing enum to set the acceleration when surfing
//doing input_and_body to refresh the walk state if you hit a wall while accelerating
move_state . apply_enum_and_input_and_body ( body , touching , models , hitbox_mesh , style , camera , input_state ) ;
2024-08-08 15:54:23 -07:00
}
fn collision_start_intersect (
2024-08-08 16:54:42 -07:00
move_state :& mut MoveState ,
body :& mut Body ,
mode_state :& mut ModeState ,
touching :& mut TouchingState ,
mode :Option < & gameplay_modes ::Mode > ,
run :& mut run ::Run ,
models :& PhysicsModels ,
hitbox_mesh :& HitboxMesh ,
bvh :& bvh ::BvhNode < ConvexMeshId > ,
style :& StyleModifiers ,
camera :& PhysicsCamera ,
input_state :& InputState ,
2024-08-09 14:16:10 -07:00
attr :& gameplay_attributes ::IntersectAttributes ,
2024-08-08 15:54:23 -07:00
intersect :IntersectCollision ,
2024-08-08 16:54:42 -07:00
time :Time ,
2024-08-08 15:54:23 -07:00
) {
2024-08-08 16:54:42 -07:00
//I think that setting the velocity to 0 was preventing surface contacts from entering an infinite loop
touching . insert ( Collision ::Intersect ( intersect ) ) ;
//insta booster!
if let Some ( booster ) = & attr . general . booster {
move_state . cull_velocity ( booster . boost ( body . velocity ) , body , touching , models , hitbox_mesh , style , camera , input_state ) ;
}
if let Some ( mode ) = mode {
let zone = mode . get_zone ( intersect . model_id . into ( ) ) ;
match zone {
Some ( gameplay_modes ::Zone ::Start ) = > {
println! ( " @@@@ Starting new run! " ) ;
* run = run ::Run ::new ( ) ;
} ,
Some ( gameplay_modes ::Zone ::Finish ) = > {
match run . finish ( time ) {
Ok ( ( ) ) = > println! ( " @@@@ Finished run time= {} " , run . time ( time ) ) ,
Err ( e ) = > println! ( " @@@@ Run Finish error: {e:?} " ) ,
}
} ,
Some ( gameplay_modes ::Zone ::Anticheat ) = > run . flag ( run ::FlagReason ::Anticheat ) ,
None = > ( ) ,
}
}
run_teleport_behaviour ( intersect . model_id . into ( ) , attr . general . wormhole . as_ref ( ) , mode , move_state , body , touching , run , mode_state , models , hitbox_mesh , bvh , style , camera , input_state , time ) ;
2024-08-08 15:54:23 -07:00
}
fn collision_end_contact (
2024-08-08 16:54:42 -07:00
move_state :& mut MoveState ,
body :& mut Body ,
touching :& mut TouchingState ,
models :& PhysicsModels ,
hitbox_mesh :& HitboxMesh ,
style :& StyleModifiers ,
camera :& PhysicsCamera ,
input_state :& InputState ,
2024-08-09 14:16:10 -07:00
_attr :& gameplay_attributes ::ContactAttributes ,
2024-08-08 15:54:23 -07:00
contact :ContactCollision ,
) {
2024-08-08 16:54:42 -07:00
touching . remove ( & Collision ::Contact ( contact ) ) ; //remove contact before calling contact_constrain_acceleration
//check ground
//TODO do better
//this is inner code from move_state.cull_velocity
match move_state . get_walk_state ( ) {
//did you stop touching the thing you were walking on?
Some ( walk_state ) = > if walk_state . contact = = contact {
move_state . set_move_state ( MoveState ::Air , body , touching , models , hitbox_mesh , style , camera , input_state ) ;
} ,
None = > move_state . apply_enum_and_body ( body , touching , models , hitbox_mesh , style , camera , input_state ) ,
}
2024-08-08 15:54:23 -07:00
}
fn collision_end_intersect (
2024-08-08 16:54:42 -07:00
touching :& mut TouchingState ,
mode :Option < & gameplay_modes ::Mode > ,
run :& mut run ::Run ,
2024-08-09 14:16:10 -07:00
_attr :& gameplay_attributes ::IntersectAttributes ,
2024-08-08 15:54:23 -07:00
intersect :IntersectCollision ,
2024-08-08 16:54:42 -07:00
time :Time ,
2024-08-08 15:54:23 -07:00
) {
2024-08-08 16:54:42 -07:00
touching . remove ( & Collision ::Intersect ( intersect ) ) ;
if let Some ( mode ) = mode {
let zone = mode . get_zone ( intersect . model_id . into ( ) ) ;
match zone {
Some ( gameplay_modes ::Zone ::Start ) = > {
match run . start ( time ) {
Ok ( ( ) ) = > println! ( " @@@@ Started run " ) ,
Err ( e ) = > println! ( " @@@@ Run Start error: {e:?} " ) ,
}
} ,
_ = > ( ) ,
}
}
2024-08-08 15:54:23 -07:00
}
2025-01-07 22:54:58 -08:00
fn atomic_internal_instruction ( state :& mut PhysicsState , data :& PhysicsData , ins :TimedInstruction < PhysicsInternalInstruction , TimeInner > ) {
2024-08-08 15:54:23 -07:00
state . time = ins . time ;
2024-09-11 15:58:13 -07:00
let ( should_advance_body , goober_time ) = match ins . instruction {
PhysicsInternalInstruction ::CollisionStart ( _ , dt )
| PhysicsInternalInstruction ::CollisionEnd ( _ , dt ) = > ( true , Some ( dt ) ) ,
PhysicsInternalInstruction ::StrafeTick
| PhysicsInternalInstruction ::ReachWalkTargetVelocity = > ( true , None ) ,
2024-08-08 15:54:23 -07:00
} ;
if should_advance_body {
2024-09-11 15:58:13 -07:00
match goober_time {
Some ( dt ) = > state . body . advance_time_ratio_dt ( dt ) ,
None = > state . body . advance_time ( state . time ) ,
}
2024-08-08 15:54:23 -07:00
}
match ins . instruction {
2024-09-11 15:58:13 -07:00
PhysicsInternalInstruction ::CollisionStart ( collision , _ ) = > {
2024-08-08 16:54:42 -07:00
let mode = data . modes . get_mode ( state . mode_state . get_mode_id ( ) ) ;
match collision {
Collision ::Contact ( contact ) = > collision_start_contact (
& mut state . move_state , & mut state . body , & mut state . mode_state , & mut state . touching , & mut state . run ,
mode ,
& data . models , & data . hitbox_mesh , & data . bvh , & state . style , & state . camera , & state . input_state ,
data . models . contact_attr ( contact . model_id ) ,
contact ,
state . time ,
) ,
Collision ::Intersect ( intersect ) = > collision_start_intersect (
& mut state . move_state , & mut state . body , & mut state . mode_state , & mut state . touching ,
mode ,
& mut state . run , & data . models , & data . hitbox_mesh , & data . bvh , & state . style , & state . camera , & state . input_state ,
data . models . intersect_attr ( intersect . model_id ) ,
intersect ,
state . time ,
) ,
}
2024-08-08 15:54:23 -07:00
} ,
2024-09-11 15:58:13 -07:00
PhysicsInternalInstruction ::CollisionEnd ( collision , _ ) = > match collision {
2024-08-08 16:54:42 -07:00
Collision ::Contact ( contact ) = > collision_end_contact (
& mut state . move_state , & mut state . body , & mut state . touching , & data . models , & data . hitbox_mesh , & state . style , & state . camera , & state . input_state ,
data . models . contact_attr ( contact . model_id ) ,
contact
) ,
Collision ::Intersect ( intersect ) = > collision_end_intersect (
& mut state . touching ,
data . modes . get_mode ( state . mode_state . get_mode_id ( ) ) ,
& mut state . run ,
data . models . intersect_attr ( intersect . model_id ) ,
intersect ,
state . time
) ,
2023-09-18 13:20:51 -07:00
} ,
2024-08-01 11:47:20 -07:00
PhysicsInternalInstruction ::StrafeTick = > {
2024-02-16 00:19:44 -08:00
//TODO make this less huge
if let Some ( strafe_settings ) = & state . style . strafe {
let controls = state . input_state . controls ;
if strafe_settings . activates ( controls ) {
let masked_controls = strafe_settings . mask ( controls ) ;
let control_dir = state . style . get_control_dir ( masked_controls ) ;
2024-09-11 15:58:13 -07:00
if control_dir ! = vec3 ::ZERO {
2024-02-16 00:19:44 -08:00
let camera_mat = state . camera . simulate_move_rotation_y ( state . input_state . lerp_delta ( state . time ) . x ) ;
2024-09-11 15:58:13 -07:00
if let Some ( ticked_velocity ) = strafe_settings . tick_velocity ( state . body . velocity , ( camera_mat * control_dir ) . with_length ( Planar64 ::ONE ) . divide ( ) . fix_1 ( ) ) {
2024-02-16 00:19:44 -08:00
//this is wrong but will work ig
//need to note which push planes activate in push solve and keep those
state . cull_velocity ( data , ticked_velocity ) ;
}
2023-11-30 01:51:17 -08:00
}
}
2023-09-08 20:14:18 -07:00
}
}
2024-08-01 11:47:20 -07:00
PhysicsInternalInstruction ::ReachWalkTargetVelocity = > {
2024-01-29 22:37:05 -08:00
match & mut state . move_state {
2024-02-16 00:19:44 -08:00
MoveState ::Air
| MoveState ::Water
| MoveState ::Fly
= > println! ( " ReachWalkTargetVelocity fired for non-walking MoveState " ) ,
2023-10-12 19:44:46 -07:00
MoveState ::Walk ( walk_state ) | MoveState ::Ladder ( walk_state ) = > {
2024-02-16 00:19:44 -08:00
match & walk_state . target {
//you are not supposed to reach a walk target which is already reached!
TransientAcceleration ::Reached = > unreachable! ( ) ,
TransientAcceleration ::Reachable { acceleration :_ , time :_ } = > {
//velocity is already handled by advance_time
//we know that the acceleration is precisely zero because the walk target is known to be reachable
//which means that gravity can be fully cancelled
//ignore moving platforms for now
2024-09-11 15:58:13 -07:00
set_acceleration ( & mut state . body , & state . touching , & data . models , & data . hitbox_mesh , vec3 ::ZERO ) ;
2024-02-16 00:19:44 -08:00
walk_state . target = TransientAcceleration ::Reached ;
2023-10-17 00:17:54 -07:00
} ,
2024-02-16 00:19:44 -08:00
//you are not supposed to reach an unreachable walk target!
TransientAcceleration ::Unreachable { acceleration :_ } = > unreachable! ( ) ,
2023-10-17 00:17:54 -07:00
}
2023-10-12 19:44:46 -07:00
}
}
2023-09-18 23:36:14 -07:00
} ,
2024-08-01 11:47:20 -07:00
}
}
2025-01-07 22:54:58 -08:00
fn atomic_input_instruction ( state :& mut PhysicsState , data :& PhysicsData , ins :TimedInstruction < PhysicsInputInstruction , TimeInner > ) {
2024-08-08 13:18:28 -07:00
state . time = ins . time ;
2024-08-01 11:47:20 -07:00
let should_advance_body = match ins . instruction {
//the body may as well be a quantum wave function
//as far as these instruction are concerned (they don't care where it is)
PhysicsInputInstruction ::SetSensitivity ( .. )
2024-08-06 11:10:43 -07:00
| PhysicsInputInstruction ::Reset
2024-08-01 12:57:58 -07:00
| PhysicsInputInstruction ::Restart
| PhysicsInputInstruction ::Spawn ( .. )
2024-08-01 11:47:20 -07:00
| PhysicsInputInstruction ::SetZoom ( .. )
| PhysicsInputInstruction ::Idle = > false ,
//these controls only update the body if you are on the ground
PhysicsInputInstruction ::SetNextMouse ( .. )
| PhysicsInputInstruction ::ReplaceMouse ( .. )
| PhysicsInputInstruction ::SetMoveForward ( .. )
| PhysicsInputInstruction ::SetMoveLeft ( .. )
| PhysicsInputInstruction ::SetMoveBack ( .. )
| PhysicsInputInstruction ::SetMoveRight ( .. )
| PhysicsInputInstruction ::SetMoveUp ( .. )
| PhysicsInputInstruction ::SetMoveDown ( .. )
| PhysicsInputInstruction ::SetJump ( .. ) = > {
match & state . move_state {
MoveState ::Fly
| MoveState ::Water
| MoveState ::Walk ( _ )
| MoveState ::Ladder ( _ ) = > true ,
MoveState ::Air = > false ,
}
} ,
//the body must be updated unconditionally
PhysicsInputInstruction ::PracticeFly = > true ,
} ;
if should_advance_body {
state . body . advance_time ( state . time ) ;
}
//TODO: UNTAB
2024-01-29 22:37:05 -08:00
let mut b_refresh_walk_target = true ;
2024-08-01 11:47:20 -07:00
match ins . instruction {
PhysicsInputInstruction ::SetSensitivity ( sensitivity ) = > state . camera . sensitivity = sensitivity ,
2024-01-29 22:37:05 -08:00
PhysicsInputInstruction ::SetNextMouse ( m ) = > {
2024-02-16 00:19:44 -08:00
state . camera . move_mouse ( state . input_state . mouse_delta ( ) ) ;
state . input_state . set_next_mouse ( m ) ;
2023-10-04 20:04:04 -07:00
} ,
2024-01-29 22:37:05 -08:00
PhysicsInputInstruction ::ReplaceMouse ( m0 , m1 ) = > {
2024-02-16 00:19:44 -08:00
state . camera . move_mouse ( m0 . pos - state . input_state . mouse . pos ) ;
state . input_state . replace_mouse ( m0 , m1 ) ;
2023-09-19 17:53:29 -07:00
} ,
2024-02-16 00:19:44 -08:00
PhysicsInputInstruction ::SetMoveForward ( s ) = > state . input_state . set_control ( Controls ::MoveForward , s ) ,
PhysicsInputInstruction ::SetMoveLeft ( s ) = > state . input_state . set_control ( Controls ::MoveLeft , s ) ,
PhysicsInputInstruction ::SetMoveBack ( s ) = > state . input_state . set_control ( Controls ::MoveBackward , s ) ,
PhysicsInputInstruction ::SetMoveRight ( s ) = > state . input_state . set_control ( Controls ::MoveRight , s ) ,
PhysicsInputInstruction ::SetMoveUp ( s ) = > state . input_state . set_control ( Controls ::MoveUp , s ) ,
PhysicsInputInstruction ::SetMoveDown ( s ) = > state . input_state . set_control ( Controls ::MoveDown , s ) ,
2024-01-29 22:37:05 -08:00
PhysicsInputInstruction ::SetJump ( s ) = > {
2024-02-16 00:19:44 -08:00
state . input_state . set_control ( Controls ::Jump , s ) ;
if let Some ( walk_state ) = state . move_state . get_walk_state ( ) {
if let Some ( jump_settings ) = & state . style . jump {
let jump_dir = walk_state . jump_direction . direction ( & data . models , & data . hitbox_mesh , & walk_state . contact ) ;
2024-08-09 14:16:10 -07:00
let booster_option = data . models . contact_attr ( walk_state . contact . model_id ) . general . booster . as_ref ( ) ;
2024-08-07 17:08:31 -07:00
let jumped_velocity = jump_settings . jumped_velocity ( & state . style , jump_dir , state . body . velocity , booster_option ) ;
2024-02-16 00:19:44 -08:00
state . cull_velocity ( & data , jumped_velocity ) ;
2023-11-30 01:51:17 -08:00
}
}
2024-08-01 11:47:20 -07:00
b_refresh_walk_target = false ;
2023-09-19 17:53:29 -07:00
} ,
2024-01-29 22:37:05 -08:00
PhysicsInputInstruction ::SetZoom ( s ) = > {
2024-02-16 00:19:44 -08:00
state . input_state . set_control ( Controls ::Zoom , s ) ;
2024-01-29 22:37:05 -08:00
b_refresh_walk_target = false ;
2023-09-19 17:53:29 -07:00
} ,
2024-08-06 11:10:43 -07:00
PhysicsInputInstruction ::Reset = > {
2024-08-01 12:57:58 -07:00
//totally reset physics state
state . reset_to_default ( ) ;
2024-08-06 11:10:43 -07:00
b_refresh_walk_target = false ;
} ,
PhysicsInputInstruction ::Restart = > {
//teleport to start zone
2024-08-08 16:54:42 -07:00
let mode = data . modes . get_mode ( state . mode_state . get_mode_id ( ) ) ;
let spawn_point = mode . and_then ( | mode |
2024-01-29 22:37:05 -08:00
//TODO: spawn at the bottom of the start zone plus the hitbox size
2024-08-01 12:57:58 -07:00
//TODO: set camera andles to face the same way as the start zone
2024-08-09 14:16:10 -07:00
data . models . get_model_transform ( mode . get_start ( ) . into ( ) ) . map ( | transform |
transform . vertex . translation
)
2024-09-11 15:58:13 -07:00
) . unwrap_or ( vec3 ::ZERO ) ;
2024-08-08 16:54:42 -07:00
set_position ( spawn_point , & mut state . move_state , & mut state . body , & mut state . touching , & mut state . run , & mut state . mode_state , mode , & data . models , & data . hitbox_mesh , & data . bvh , & state . style , & state . camera , & state . input_state , state . time ) ;
2024-09-11 15:58:13 -07:00
set_velocity ( & mut state . body , & state . touching , & data . models , & data . hitbox_mesh , vec3 ::ZERO ) ;
2024-02-16 00:19:44 -08:00
state . set_move_state ( data , MoveState ::Air ) ;
b_refresh_walk_target = false ;
2024-08-01 12:57:58 -07:00
}
PhysicsInputInstruction ::Spawn ( mode_id , stage_id ) = > {
//spawn at a particular stage
if let Some ( mode ) = data . modes . get_mode ( mode_id ) {
2024-08-08 16:54:42 -07:00
if let Some ( stage ) = mode . get_stage ( stage_id ) {
let _ = teleport_to_spawn (
2025-01-06 20:25:53 -08:00
stage . spawn ( ) ,
2024-08-08 16:54:42 -07:00
& mut state . move_state , & mut state . body , & mut state . touching , & mut state . run , & mut state . mode_state ,
mode ,
& data . models , & data . hitbox_mesh , & data . bvh , & state . style , & state . camera , & state . input_state , state . time
) ;
}
2024-08-01 12:57:58 -07:00
}
b_refresh_walk_target = false ;
2024-02-16 00:19:44 -08:00
} ,
PhysicsInputInstruction ::PracticeFly = > {
match & state . move_state {
MoveState ::Fly = > {
state . set_move_state ( data , MoveState ::Air ) ;
} ,
_ = > {
state . set_move_state ( data , MoveState ::Fly ) ;
} ,
}
2024-01-29 22:37:05 -08:00
b_refresh_walk_target = false ;
2023-09-19 17:53:29 -07:00
} ,
2024-08-01 11:47:20 -07:00
PhysicsInputInstruction ::Idle = > {
//literally idle!
b_refresh_walk_target = false ;
} ,
2023-09-19 17:53:29 -07:00
}
2024-01-29 22:37:05 -08:00
if b_refresh_walk_target {
2024-02-16 00:19:44 -08:00
state . apply_input_and_body ( data ) ;
state . cull_velocity ( data , state . body . velocity ) ;
2024-08-20 15:25:07 -07:00
//also check if accelerating away from surface
2023-09-18 21:10:07 -07:00
}
2024-08-01 11:47:20 -07:00
}
2025-01-07 22:54:58 -08:00
fn atomic_state_update ( state :& mut PhysicsState , data :& PhysicsData , ins :TimedInstruction < PhysicsInstruction , TimeInner > ) {
2024-08-01 11:47:20 -07:00
match & ins . instruction {
PhysicsInstruction ::Input ( PhysicsInputInstruction ::Idle )
| PhysicsInstruction ::Input ( PhysicsInputInstruction ::SetNextMouse ( _ ) )
| PhysicsInstruction ::Input ( PhysicsInputInstruction ::ReplaceMouse ( _ , _ ) )
2024-08-01 21:56:17 -07:00
| PhysicsInstruction ::Internal ( PhysicsInternalInstruction ::StrafeTick )
| PhysicsInstruction ::Internal ( PhysicsInternalInstruction ::ReachWalkTargetVelocity ) = > ( ) ,
2024-08-01 11:47:20 -07:00
_ = > println! ( " {} | {:?} " , ins . time , ins . instruction ) ,
}
2024-08-02 10:41:51 -07:00
if ins . time < state . time {
2024-09-11 15:58:13 -07:00
println! ( " @@@@ Time travel warning! state.time= {} ins.time= {} \n Instruction= {:?} " , state . time , ins . time , ins . instruction ) ;
2024-08-02 10:41:51 -07:00
}
2024-08-08 13:18:28 -07:00
//idle is special, it is specifically a no-op to get Internal events to catch up to real time
2024-08-01 11:47:20 -07:00
match ins . instruction {
2024-08-08 13:18:28 -07:00
PhysicsInstruction ::Input ( PhysicsInputInstruction ::Idle ) = > ( ) ,
2024-08-01 11:47:20 -07:00
PhysicsInstruction ::Internal ( instruction ) = > atomic_internal_instruction ( state , data , TimedInstruction { time :ins . time , instruction } ) ,
PhysicsInstruction ::Input ( instruction ) = > atomic_input_instruction ( state , data , TimedInstruction { time :ins . time , instruction } ) ,
2023-09-08 20:14:18 -07:00
}
2023-09-08 17:15:49 -07:00
}
2023-11-30 01:51:17 -08:00
2024-01-29 22:37:05 -08:00
#[ cfg(test) ]
mod test {
2024-09-11 15:58:13 -07:00
use strafesnet_common ::integer ::{ vec3 ::{ self , int as int3 } , mat3 } ;
2025-01-07 22:54:58 -08:00
use crate ::body ::VirtualBody ;
2024-01-29 22:37:05 -08:00
use super ::* ;
fn test_collision_axis_aligned ( relative_body :Body , expected_collision_time :Option < Time > ) {
2024-09-11 15:58:13 -07:00
let h0 = HitboxMesh ::new ( PhysicsMesh ::unit_cube ( ) , integer ::Planar64Affine3 ::new ( mat3 ::from_diagonal ( int3 ( 5 , 1 , 5 ) > > 1 ) , vec3 ::ZERO ) ) ;
2024-01-29 22:37:05 -08:00
let h1 = StyleModifiers ::roblox_bhop ( ) . calculate_mesh ( ) ;
let hitbox_mesh = h1 . transformed_mesh ( ) ;
let platform_mesh = h0 . transformed_mesh ( ) ;
let minkowski = model_physics ::MinkowskiMesh ::minkowski_sum ( platform_mesh , hitbox_mesh ) ;
2024-09-11 15:58:13 -07:00
let collision = minkowski . predict_collision_in ( & relative_body , Time ::from_secs ( 10 ) ) ;
assert_eq! ( collision . map ( | tup | relative_body . time + tup . 1. into ( ) ) , expected_collision_time , " Incorrect time of collision " ) ;
2024-01-29 22:37:05 -08:00
}
fn test_collision_rotated ( relative_body :Body , expected_collision_time :Option < Time > ) {
let h0 = HitboxMesh ::new ( PhysicsMesh ::unit_cube ( ) ,
integer ::Planar64Affine3 ::new (
2024-09-11 15:58:13 -07:00
integer ::Planar64Mat3 ::from_cols ( [
int3 ( 5 , 0 , 1 ) > > 1 ,
int3 ( 0 , 1 , 0 ) > > 1 ,
int3 ( - 1 , 0 , 5 ) > > 1 ,
] ) ,
vec3 ::ZERO
) ,
2024-01-29 22:37:05 -08:00
) ;
let h1 = StyleModifiers ::roblox_bhop ( ) . calculate_mesh ( ) ;
let hitbox_mesh = h1 . transformed_mesh ( ) ;
let platform_mesh = h0 . transformed_mesh ( ) ;
let minkowski = model_physics ::MinkowskiMesh ::minkowski_sum ( platform_mesh , hitbox_mesh ) ;
2024-09-11 15:58:13 -07:00
let collision = minkowski . predict_collision_in ( & relative_body , Time ::from_secs ( 10 ) ) ;
assert_eq! ( collision . map ( | tup | relative_body . time + tup . 1. into ( ) ) , expected_collision_time , " Incorrect time of collision " ) ;
2024-01-29 22:37:05 -08:00
}
fn test_collision ( relative_body :Body , expected_collision_time :Option < Time > ) {
test_collision_axis_aligned ( relative_body . clone ( ) , expected_collision_time ) ;
test_collision_rotated ( relative_body , expected_collision_time ) ;
}
#[ test ]
2024-09-11 15:58:13 -07:00
fn test_collision_degenerate_straight_down ( ) {
2024-01-29 22:37:05 -08:00
test_collision ( Body ::new (
2024-09-11 15:58:13 -07:00
int3 ( 0 , 5 , 0 ) ,
int3 ( 0 , - 1 , 0 ) ,
vec3 ::ZERO ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) , Some ( Time ::from_secs ( 2 ) ) ) ;
}
#[ test ]
fn test_collision_degenerate_east ( ) {
test_collision ( Body ::new (
2024-09-11 15:58:13 -07:00
int3 ( 3 , 5 , 0 ) ,
int3 ( 0 , - 1 , 0 ) ,
vec3 ::ZERO ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) , Some ( Time ::from_secs ( 2 ) ) ) ;
}
#[ test ]
fn test_collision_degenerate_south ( ) {
test_collision ( Body ::new (
2024-09-11 15:58:13 -07:00
int3 ( 0 , 5 , 3 ) ,
int3 ( 0 , - 1 , 0 ) ,
vec3 ::ZERO ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) , Some ( Time ::from_secs ( 2 ) ) ) ;
}
#[ test ]
fn test_collision_degenerate_west ( ) {
test_collision ( Body ::new (
2024-09-11 15:58:13 -07:00
int3 ( - 3 , 5 , 0 ) ,
int3 ( 0 , - 1 , 0 ) ,
vec3 ::ZERO ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) , Some ( Time ::from_secs ( 2 ) ) ) ;
}
#[ test ]
fn test_collision_degenerate_north ( ) {
test_collision ( Body ::new (
2024-09-11 15:58:13 -07:00
int3 ( 0 , 5 , - 3 ) ,
int3 ( 0 , - 1 , 0 ) ,
vec3 ::ZERO ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) , Some ( Time ::from_secs ( 2 ) ) ) ;
}
#[ test ]
fn test_collision_parabola_edge_east_from_west ( ) {
2024-09-11 15:58:13 -07:00
test_collision ( VirtualBody ::relative ( & Body ::ZERO , & Body ::new (
int3 ( 3 , 3 , 0 ) ,
int3 ( 100 , - 1 , 0 ) ,
int3 ( 0 , - 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) ) . body ( Time ::from_secs ( - 1 ) ) , Some ( Time ::from_secs ( 0 ) ) ) ;
}
#[ test ]
fn test_collision_parabola_edge_south_from_north ( ) {
2024-09-11 15:58:13 -07:00
test_collision ( VirtualBody ::relative ( & Body ::ZERO , & Body ::new (
int3 ( 0 , 3 , 3 ) ,
int3 ( 0 , - 1 , 100 ) ,
int3 ( 0 , - 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) ) . body ( Time ::from_secs ( - 1 ) ) , Some ( Time ::from_secs ( 0 ) ) ) ;
}
#[ test ]
fn test_collision_parabola_edge_west_from_east ( ) {
2024-09-11 15:58:13 -07:00
test_collision ( VirtualBody ::relative ( & Body ::ZERO , & Body ::new (
int3 ( - 3 , 3 , 0 ) ,
int3 ( - 100 , - 1 , 0 ) ,
int3 ( 0 , - 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) ) . body ( Time ::from_secs ( - 1 ) ) , Some ( Time ::from_secs ( 0 ) ) ) ;
}
#[ test ]
fn test_collision_parabola_edge_north_from_south ( ) {
2024-09-11 15:58:13 -07:00
test_collision ( VirtualBody ::relative ( & Body ::ZERO , & Body ::new (
int3 ( 0 , 3 , - 3 ) ,
int3 ( 0 , - 1 , - 100 ) ,
int3 ( 0 , - 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) ) . body ( Time ::from_secs ( - 1 ) ) , Some ( Time ::from_secs ( 0 ) ) ) ;
}
#[ test ]
fn test_collision_parabola_edge_north_from_ne ( ) {
2024-09-11 15:58:13 -07:00
test_collision ( VirtualBody ::relative ( & Body ::ZERO , & Body ::new (
int3 ( 0 , 6 , - 7 ) > > 1 ,
int3 ( - 10 , - 1 , 1 ) ,
int3 ( 0 , - 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) ) . body ( Time ::from_secs ( - 1 ) ) , Some ( Time ::from_secs ( 0 ) ) ) ;
}
#[ test ]
fn test_collision_parabola_edge_north_from_nw ( ) {
2024-09-11 15:58:13 -07:00
test_collision ( VirtualBody ::relative ( & Body ::ZERO , & Body ::new (
int3 ( 0 , 6 , - 7 ) > > 1 ,
int3 ( 10 , - 1 , 1 ) ,
int3 ( 0 , - 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) ) . body ( Time ::from_secs ( - 1 ) ) , Some ( Time ::from_secs ( 0 ) ) ) ;
}
#[ test ]
fn test_collision_parabola_edge_east_from_se ( ) {
2024-09-11 15:58:13 -07:00
test_collision ( VirtualBody ::relative ( & Body ::ZERO , & Body ::new (
int3 ( 7 , 6 , 0 ) > > 1 ,
int3 ( - 1 , - 1 , - 10 ) ,
int3 ( 0 , - 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) ) . body ( Time ::from_secs ( - 1 ) ) , Some ( Time ::from_secs ( 0 ) ) ) ;
}
#[ test ]
fn test_collision_parabola_edge_east_from_ne ( ) {
2024-09-11 15:58:13 -07:00
test_collision ( VirtualBody ::relative ( & Body ::ZERO , & Body ::new (
int3 ( 7 , 6 , 0 ) > > 1 ,
int3 ( - 1 , - 1 , 10 ) ,
int3 ( 0 , - 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) ) . body ( Time ::from_secs ( - 1 ) ) , Some ( Time ::from_secs ( 0 ) ) ) ;
}
#[ test ]
fn test_collision_parabola_edge_south_from_se ( ) {
2024-09-11 15:58:13 -07:00
test_collision ( VirtualBody ::relative ( & Body ::ZERO , & Body ::new (
int3 ( 0 , 6 , 7 ) > > 1 ,
int3 ( - 10 , - 1 , - 1 ) ,
int3 ( 0 , - 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) ) . body ( Time ::from_secs ( - 1 ) ) , Some ( Time ::from_secs ( 0 ) ) ) ;
}
#[ test ]
fn test_collision_parabola_edge_south_from_sw ( ) {
2024-09-11 15:58:13 -07:00
test_collision ( VirtualBody ::relative ( & Body ::ZERO , & Body ::new (
int3 ( 0 , 6 , 7 ) > > 1 ,
int3 ( 10 , - 1 , - 1 ) ,
int3 ( 0 , - 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) ) . body ( Time ::from_secs ( - 1 ) ) , Some ( Time ::from_secs ( 0 ) ) ) ;
}
#[ test ]
fn test_collision_parabola_edge_west_from_se ( ) {
2024-09-11 15:58:13 -07:00
test_collision ( VirtualBody ::relative ( & Body ::ZERO , & Body ::new (
int3 ( - 7 , 6 , 0 ) > > 1 ,
int3 ( 1 , - 1 , - 10 ) ,
int3 ( 0 , - 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) ) . body ( Time ::from_secs ( - 1 ) ) , Some ( Time ::from_secs ( 0 ) ) ) ;
}
#[ test ]
fn test_collision_parabola_edge_west_from_ne ( ) {
2024-09-11 15:58:13 -07:00
test_collision ( VirtualBody ::relative ( & Body ::ZERO , & Body ::new (
int3 ( - 7 , 6 , 0 ) > > 1 ,
int3 ( 1 , - 1 , 10 ) ,
int3 ( 0 , - 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) ) . body ( Time ::from_secs ( - 1 ) ) , Some ( Time ::from_secs ( 0 ) ) ) ;
}
#[ test ]
fn test_collision_oblique ( ) {
test_collision ( Body ::new (
2024-09-11 15:58:13 -07:00
int3 ( 0 , 5 , 0 ) ,
int3 ( 1 , - 64 , 2 ) > > 6 , // /64
vec3 ::ZERO ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) , Some ( Time ::from_secs ( 2 ) ) ) ;
}
#[ test ]
fn zoom_hit_nothing ( ) {
test_collision ( Body ::new (
2024-09-11 15:58:13 -07:00
int3 ( 0 , 10 , 0 ) ,
int3 ( 1 , 0 , 0 ) ,
int3 ( 0 , 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) , None ) ;
}
#[ test ]
fn already_inside_hit_nothing ( ) {
test_collision ( Body ::new (
2024-09-11 15:58:13 -07:00
vec3 ::ZERO ,
int3 ( 1 , 0 , 0 ) ,
int3 ( 0 , 1 , 0 ) ,
2024-01-29 22:37:05 -08:00
Time ::ZERO
) , None ) ;
}
2024-07-25 13:53:45 -07:00
}