Compare commits
5 Commits
master
...
game-mecha
Author | SHA1 | Date | |
---|---|---|---|
69caf0e6b3 | |||
6662410037 | |||
b91d760eb7 | |||
a78859b231 | |||
b04a6d870a |
122
src/body.rs
122
src/body.rs
@ -257,28 +257,70 @@ fn get_control_dir(controls: u32) -> glam::Vec3{
|
|||||||
return control_dir
|
return control_dir
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct PhysicsState {
|
pub struct GameMechanicsState{
|
||||||
pub body: Body,
|
pub spawn_id:u32,
|
||||||
pub hitbox_halfsize: glam::Vec3,
|
//jump_counts:HashMap<u32,u32>,
|
||||||
pub contacts: std::collections::HashSet::<RelativeCollision>,
|
}
|
||||||
|
impl std::default::Default for GameMechanicsState{
|
||||||
|
fn default() -> Self {
|
||||||
|
Self{
|
||||||
|
spawn_id:0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct WorldState{}
|
||||||
|
|
||||||
|
pub struct StyleModifiers{
|
||||||
|
pub constrols_mask:u32,//constrols which are unable to be activated
|
||||||
|
pub constrols_held:u32,//constrols which must be active to be able to strafe
|
||||||
|
pub mv:f32,
|
||||||
|
pub walkspeed:f32,
|
||||||
|
pub friction:f32,
|
||||||
|
pub walk_accel:f32,
|
||||||
|
pub gravity:glam::Vec3,
|
||||||
|
pub strafe_tick_num:TIME,
|
||||||
|
pub strafe_tick_den:TIME,
|
||||||
|
pub hitbox_halfsize:glam::Vec3,
|
||||||
|
}
|
||||||
|
impl std::default::Default for StyleModifiers{
|
||||||
|
fn default() -> Self {
|
||||||
|
Self{
|
||||||
|
constrols_mask: !0&!(CONTROL_MOVEUP|CONTROL_MOVEDOWN),
|
||||||
|
constrols_held: 0,
|
||||||
|
strafe_tick_num: 100,//100t
|
||||||
|
strafe_tick_den: 1_000_000_000,
|
||||||
|
gravity: glam::vec3(0.0,-100.0,0.0),
|
||||||
|
friction: 1.2,
|
||||||
|
walk_accel: 90.0,
|
||||||
|
mv: 2.7,
|
||||||
|
walkspeed: 18.0,
|
||||||
|
hitbox_halfsize: glam::vec3(1.0,2.5,1.0),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct PhysicsState{
|
||||||
|
pub time:TIME,
|
||||||
|
pub body:Body,
|
||||||
|
pub world:WorldState,//currently there is only one state the world can be in
|
||||||
|
pub game:GameMechanicsState,
|
||||||
|
pub style:StyleModifiers,
|
||||||
|
pub contacts:std::collections::HashSet::<RelativeCollision>,
|
||||||
//pub intersections: Vec<ModelId>,
|
//pub intersections: Vec<ModelId>,
|
||||||
pub models: Vec<ModelPhysics>,
|
|
||||||
//camera must exist in state because wormholes modify the camera, also camera punch
|
//camera must exist in state because wormholes modify the camera, also camera punch
|
||||||
pub camera: Camera,
|
pub camera:Camera,
|
||||||
pub mouse_interpolation: MouseInterpolationState,
|
pub mouse_interpolation:MouseInterpolationState,
|
||||||
pub controls: u32,
|
pub controls:u32,
|
||||||
pub time: TIME,
|
pub walk:WalkState,
|
||||||
pub strafe_tick_num: TIME,
|
pub grounded:bool,
|
||||||
pub strafe_tick_den: TIME,
|
//all models
|
||||||
pub tick: u32,
|
pub models:Vec<ModelPhysics>,
|
||||||
pub mv: f32,
|
|
||||||
pub walk: WalkState,
|
pub stages:Vec<crate::model::StageDescription>,
|
||||||
pub walkspeed: f32,
|
//the spawn point is where you spawn when you load into the map.
|
||||||
pub friction: f32,
|
//This is not the same as Reset which teleports you to Spawn0
|
||||||
pub walk_accel: f32,
|
pub spawn_point:glam::Vec3,
|
||||||
pub gravity: glam::Vec3,
|
|
||||||
pub grounded: bool,
|
|
||||||
pub spawn_point: glam::Vec3,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug,Clone,Copy,Hash,Eq,PartialEq)]
|
#[derive(Debug,Clone,Copy,Hash,Eq,PartialEq)]
|
||||||
@ -390,20 +432,40 @@ impl Aabb {
|
|||||||
type TreyMeshFace = AabbFace;
|
type TreyMeshFace = AabbFace;
|
||||||
type TreyMesh = Aabb;
|
type TreyMesh = Aabb;
|
||||||
|
|
||||||
|
enum PhysicsCollisionAttributes{
|
||||||
|
Contact{//track whether you are contacting the object
|
||||||
|
contacting:crate::model::ContactingAttributes,
|
||||||
|
general:crate::model::GameMechanicAttributes,
|
||||||
|
},
|
||||||
|
Intersect{//track whether you are intersecting the object
|
||||||
|
intersecting:crate::model::IntersectingAttributes,
|
||||||
|
general:crate::model::GameMechanicAttributes,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
pub struct ModelPhysics {
|
pub struct ModelPhysics {
|
||||||
//A model is a thing that has a hitbox. can be represented by a list of TreyMesh-es
|
//A model is a thing that has a hitbox. can be represented by a list of TreyMesh-es
|
||||||
//in this iteration, all it needs is extents.
|
//in this iteration, all it needs is extents.
|
||||||
mesh: TreyMesh,
|
mesh: TreyMesh,
|
||||||
|
attributes:PhysicsCollisionAttributes,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ModelPhysics {
|
impl ModelPhysics {
|
||||||
pub fn from_model(model:&crate::model::IndexedModel,model_transform:glam::Affine3A) -> Self {
|
fn from_model_transform_attributes(model:&crate::model::IndexedModel,transform:&glam::Affine3A,attributes:PhysicsCollisionAttributes)->Self{
|
||||||
let mut aabb=Aabb::new();
|
let mut aabb=Aabb::new();
|
||||||
for indexed_vertex in &model.unique_vertices {
|
for indexed_vertex in &model.unique_vertices {
|
||||||
aabb.grow(model_transform.transform_point3(glam::Vec3::from_array(model.unique_pos[indexed_vertex.pos as usize])));
|
aabb.grow(transform.transform_point3(glam::Vec3::from_array(model.unique_pos[indexed_vertex.pos as usize])));
|
||||||
}
|
}
|
||||||
Self{
|
Self{
|
||||||
mesh:aabb,
|
mesh:aabb,
|
||||||
|
attributes,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn from_model(model:&crate::model::IndexedModel,instance:&crate::model::ModelInstance) -> Option<Self> {
|
||||||
|
match &instance.attributes{
|
||||||
|
crate::model::CollisionAttributes::Decoration=>None,
|
||||||
|
crate::model::CollisionAttributes::Contact{contacting,general}=>Some(ModelPhysics::from_model_transform_attributes(model,&instance.transform,PhysicsCollisionAttributes::Contact{contacting:contacting.clone(),general:general.clone()})),
|
||||||
|
crate::model::CollisionAttributes::Intersect{intersecting,general}=>None,//Some(ModelPhysics::from_model_transform_attributes(model,&instance.transform,PhysicsCollisionAttributes::Intersecting{intersecting,general})),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pub fn unit_vertices(&self) -> [glam::Vec3;8] {
|
pub fn unit_vertices(&self) -> [glam::Vec3;8] {
|
||||||
@ -522,7 +584,7 @@ impl PhysicsState {
|
|||||||
}
|
}
|
||||||
fn next_strafe_instruction(&self) -> Option<TimedInstruction<PhysicsInstruction>> {
|
fn next_strafe_instruction(&self) -> Option<TimedInstruction<PhysicsInstruction>> {
|
||||||
return Some(TimedInstruction{
|
return Some(TimedInstruction{
|
||||||
time:(self.time*self.strafe_tick_num/self.strafe_tick_den+1)*self.strafe_tick_den/self.strafe_tick_num,
|
time:(self.time*self.style.strafe_tick_num/self.style.strafe_tick_den+1)*self.style.strafe_tick_den/self.style.strafe_tick_num,
|
||||||
//only poll the physics if there is a before and after mouse event
|
//only poll the physics if there is a before and after mouse event
|
||||||
instruction:PhysicsInstruction::StrafeTick
|
instruction:PhysicsInstruction::StrafeTick
|
||||||
});
|
});
|
||||||
@ -572,7 +634,7 @@ impl PhysicsState {
|
|||||||
self.body.acceleration=a;
|
self.body.acceleration=a;
|
||||||
self.walk.state=WalkEnum::Reached;
|
self.walk.state=WalkEnum::Reached;
|
||||||
}else{
|
}else{
|
||||||
let accel=self.walk_accel.min(self.gravity.length()*self.friction);
|
let accel=self.style.walk_accel.min(self.style.gravity.length()*self.style.friction);
|
||||||
let time_delta=target_diff.length()/accel;
|
let time_delta=target_diff.length()/accel;
|
||||||
let mut a=target_diff/time_delta;
|
let mut a=target_diff/time_delta;
|
||||||
self.contact_constrain_acceleration(&mut a);
|
self.contact_constrain_acceleration(&mut a);
|
||||||
@ -601,7 +663,7 @@ impl PhysicsState {
|
|||||||
fn mesh(&self) -> TreyMesh {
|
fn mesh(&self) -> TreyMesh {
|
||||||
let mut aabb=Aabb::new();
|
let mut aabb=Aabb::new();
|
||||||
for vertex in Aabb::unit_vertices(){
|
for vertex in Aabb::unit_vertices(){
|
||||||
aabb.grow(self.body.position+self.hitbox_halfsize*vertex);
|
aabb.grow(self.body.position+self.style.hitbox_halfsize*vertex);
|
||||||
}
|
}
|
||||||
aabb
|
aabb
|
||||||
}
|
}
|
||||||
@ -945,7 +1007,7 @@ impl crate::instruction::InstructionConsumer<PhysicsInstruction> for PhysicsStat
|
|||||||
},
|
},
|
||||||
PhysicsInstruction::CollisionEnd(c) => {
|
PhysicsInstruction::CollisionEnd(c) => {
|
||||||
self.contacts.remove(&c);//remove contact before calling contact_constrain_acceleration
|
self.contacts.remove(&c);//remove contact before calling contact_constrain_acceleration
|
||||||
let mut a=self.gravity;
|
let mut a=self.style.gravity;
|
||||||
self.contact_constrain_acceleration(&mut a);
|
self.contact_constrain_acceleration(&mut a);
|
||||||
self.body.acceleration=a;
|
self.body.acceleration=a;
|
||||||
//check ground
|
//check ground
|
||||||
@ -961,8 +1023,8 @@ impl crate::instruction::InstructionConsumer<PhysicsInstruction> for PhysicsStat
|
|||||||
let camera_mat=self.camera.simulate_move_rotation_y(self.mouse_interpolation.interpolated_position(self.time).x-self.mouse_interpolation.mouse0.x);
|
let camera_mat=self.camera.simulate_move_rotation_y(self.mouse_interpolation.interpolated_position(self.time).x-self.mouse_interpolation.mouse0.x);
|
||||||
let control_dir=camera_mat*get_control_dir(self.controls);
|
let control_dir=camera_mat*get_control_dir(self.controls);
|
||||||
let d=self.body.velocity.dot(control_dir);
|
let d=self.body.velocity.dot(control_dir);
|
||||||
if d<self.mv {
|
if d<self.style.mv {
|
||||||
let mut v=self.body.velocity+(self.mv-d)*control_dir;
|
let mut v=self.body.velocity+(self.style.mv-d)*control_dir;
|
||||||
self.contact_constrain_velocity(&mut v);
|
self.contact_constrain_velocity(&mut v);
|
||||||
self.body.velocity=v;
|
self.body.velocity=v;
|
||||||
}
|
}
|
||||||
@ -1008,7 +1070,7 @@ impl crate::instruction::InstructionConsumer<PhysicsInstruction> for PhysicsStat
|
|||||||
self.body.velocity=glam::Vec3::ZERO;
|
self.body.velocity=glam::Vec3::ZERO;
|
||||||
//manual clear //for c in self.contacts{process_instruction(CollisionEnd(c))}
|
//manual clear //for c in self.contacts{process_instruction(CollisionEnd(c))}
|
||||||
self.contacts.clear();
|
self.contacts.clear();
|
||||||
self.body.acceleration=self.gravity;
|
self.body.acceleration=self.style.gravity;
|
||||||
self.walk.state=WalkEnum::Reached;
|
self.walk.state=WalkEnum::Reached;
|
||||||
self.grounded=false;
|
self.grounded=false;
|
||||||
refresh_walk_target=false;
|
refresh_walk_target=false;
|
||||||
@ -1020,7 +1082,7 @@ impl crate::instruction::InstructionConsumer<PhysicsInstruction> for PhysicsStat
|
|||||||
if refresh_walk_target_velocity{
|
if refresh_walk_target_velocity{
|
||||||
let camera_mat=self.camera.simulate_move_rotation_y(self.mouse_interpolation.interpolated_position(self.time).x-self.mouse_interpolation.mouse0.x);
|
let camera_mat=self.camera.simulate_move_rotation_y(self.mouse_interpolation.interpolated_position(self.time).x-self.mouse_interpolation.mouse0.x);
|
||||||
let control_dir=camera_mat*get_control_dir(self.controls);
|
let control_dir=camera_mat*get_control_dir(self.controls);
|
||||||
self.walk.target_velocity=self.walkspeed*control_dir;
|
self.walk.target_velocity=self.style.walkspeed*control_dir;
|
||||||
}
|
}
|
||||||
self.refresh_walk_target();
|
self.refresh_walk_target();
|
||||||
}
|
}
|
||||||
|
@ -30,6 +30,64 @@ fn get_texture_refs(dom:&rbx_dom_weak::WeakDom) -> Vec<rbx_dom_weak::types::Ref>
|
|||||||
//next class
|
//next class
|
||||||
objects
|
objects
|
||||||
}
|
}
|
||||||
|
fn get_attributes(name:&str,can_collide:bool,velocity:glam::Vec3)->crate::model::CollisionAttributes{
|
||||||
|
let mut general=crate::model::GameMechanicAttributes::default();
|
||||||
|
let mut intersecting=crate::model::IntersectingAttributes::default();
|
||||||
|
let mut contacting=crate::model::ContactingAttributes::default();
|
||||||
|
match name{
|
||||||
|
// "Water"=>intersecting.water=Some(crate::model::IntersectingWater::default()),
|
||||||
|
// "Accelerator"=>(),
|
||||||
|
// "MapFinish"=>(),
|
||||||
|
// "MapAnticheat"=>(),
|
||||||
|
"Platform"=>general.stage_element=Some(crate::model::GameMechanicStageElement{
|
||||||
|
mode_id:0,
|
||||||
|
stage_id:0,
|
||||||
|
force:false,
|
||||||
|
behaviour:crate::model::StageElementBehaviour::Platform,
|
||||||
|
}),
|
||||||
|
other=>{
|
||||||
|
let regman=lazy_regex::regex!(r"^(Force)?(SpawnAt|Trigger|Teleport|Platform)(\d+)$");
|
||||||
|
if let Some(captures) = regman.captures(other) {
|
||||||
|
general.stage_element=Some(crate::model::GameMechanicStageElement{
|
||||||
|
mode_id:0,
|
||||||
|
stage_id:captures[3].parse::<u32>().unwrap(),
|
||||||
|
force:match captures.get(1){
|
||||||
|
Some(m)=>m.as_str()=="Force",
|
||||||
|
None=>false,
|
||||||
|
},
|
||||||
|
behaviour:match &captures[2]{
|
||||||
|
"SpawnAt"=>crate::model::StageElementBehaviour::SpawnAt,
|
||||||
|
"Trigger"=>crate::model::StageElementBehaviour::Trigger,
|
||||||
|
"Teleport"=>crate::model::StageElementBehaviour::Teleport,
|
||||||
|
"Platform"=>crate::model::StageElementBehaviour::Platform,
|
||||||
|
_=>panic!("regex[2] messed up bad"),
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//need some way to skip this
|
||||||
|
if velocity!=glam::Vec3::ZERO{
|
||||||
|
general.booster=Some(crate::model::GameMechanicBooster{velocity});
|
||||||
|
}
|
||||||
|
match can_collide{
|
||||||
|
true=>{
|
||||||
|
match name{
|
||||||
|
//"Bounce"=>(),
|
||||||
|
"Surf"=>contacting.surf=Some(crate::model::ContactingSurf{}),
|
||||||
|
"Ladder"=>contacting.ladder=Some(crate::model::ContactingLadder{sticky:true}),
|
||||||
|
other=>{
|
||||||
|
//REGEX!!!!
|
||||||
|
//Jump#
|
||||||
|
//WormholeIn#
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return crate::model::CollisionAttributes::Contact{contacting,general};
|
||||||
|
},
|
||||||
|
false=>return crate::model::CollisionAttributes::Decoration,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
struct RobloxAssetId(u64);
|
struct RobloxAssetId(u64);
|
||||||
struct RobloxAssetIdParseErr;
|
struct RobloxAssetIdParseErr;
|
||||||
@ -113,6 +171,8 @@ pub fn generate_indexed_models(dom:rbx_dom_weak::WeakDom) -> crate::model::Index
|
|||||||
//IndexedModelInstances includes textures
|
//IndexedModelInstances includes textures
|
||||||
let mut spawn_point=glam::Vec3::ZERO;
|
let mut spawn_point=glam::Vec3::ZERO;
|
||||||
|
|
||||||
|
let mut stages=Vec::new();
|
||||||
|
|
||||||
let mut indexed_models=Vec::new();
|
let mut indexed_models=Vec::new();
|
||||||
let mut model_id_from_description=std::collections::HashMap::<RobloxBasePartDescription,usize>::new();
|
let mut model_id_from_description=std::collections::HashMap::<RobloxBasePartDescription,usize>::new();
|
||||||
|
|
||||||
@ -127,13 +187,17 @@ pub fn generate_indexed_models(dom:rbx_dom_weak::WeakDom) -> crate::model::Index
|
|||||||
if let (
|
if let (
|
||||||
Some(rbx_dom_weak::types::Variant::CFrame(cf)),
|
Some(rbx_dom_weak::types::Variant::CFrame(cf)),
|
||||||
Some(rbx_dom_weak::types::Variant::Vector3(size)),
|
Some(rbx_dom_weak::types::Variant::Vector3(size)),
|
||||||
|
Some(rbx_dom_weak::types::Variant::Vector3(velocity)),
|
||||||
Some(rbx_dom_weak::types::Variant::Float32(transparency)),
|
Some(rbx_dom_weak::types::Variant::Float32(transparency)),
|
||||||
Some(rbx_dom_weak::types::Variant::Color3uint8(color3)),
|
Some(rbx_dom_weak::types::Variant::Color3uint8(color3)),
|
||||||
|
Some(rbx_dom_weak::types::Variant::Bool(can_collide)),
|
||||||
) = (
|
) = (
|
||||||
object.properties.get("CFrame"),
|
object.properties.get("CFrame"),
|
||||||
object.properties.get("Size"),
|
object.properties.get("Size"),
|
||||||
|
object.properties.get("Velocity"),
|
||||||
object.properties.get("Transparency"),
|
object.properties.get("Transparency"),
|
||||||
object.properties.get("Color"),
|
object.properties.get("Color"),
|
||||||
|
object.properties.get("CanCollide"),
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
let model_transform=glam::Affine3A::from_translation(
|
let model_transform=glam::Affine3A::from_translation(
|
||||||
@ -333,6 +397,7 @@ pub fn generate_indexed_models(dom:rbx_dom_weak::WeakDom) -> crate::model::Index
|
|||||||
indexed_models[model_id].instances.push(crate::model::ModelInstance {
|
indexed_models[model_id].instances.push(crate::model::ModelInstance {
|
||||||
transform:model_transform,
|
transform:model_transform,
|
||||||
color:glam::vec4(color3.r as f32/255f32, color3.g as f32/255f32, color3.b as f32/255f32, 1.0-*transparency),
|
color:glam::vec4(color3.r as f32/255f32, color3.g as f32/255f32, color3.b as f32/255f32, 1.0-*transparency),
|
||||||
|
attributes:get_attributes(&object.name,*can_collide,glam::vec3(velocity.x,velocity.y,velocity.z)),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -341,5 +406,6 @@ pub fn generate_indexed_models(dom:rbx_dom_weak::WeakDom) -> crate::model::Index
|
|||||||
textures:asset_id_from_texture_id.iter().map(|t|t.to_string()).collect(),
|
textures:asset_id_from_texture_id.iter().map(|t|t.to_string()).collect(),
|
||||||
models:indexed_models,
|
models:indexed_models,
|
||||||
spawn_point,
|
spawn_point,
|
||||||
|
stages,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
31
src/main.rs
31
src/main.rs
@ -88,7 +88,9 @@ impl GraphicsData {
|
|||||||
for model in &indexed_models.models{
|
for model in &indexed_models.models{
|
||||||
//make aabb and run vertices to get realistic bounds
|
//make aabb and run vertices to get realistic bounds
|
||||||
for model_instance in &model.instances{
|
for model_instance in &model.instances{
|
||||||
self.physics.models.push(body::ModelPhysics::from_model(&model,model_instance.transform));
|
if let Some(model_physics)=body::ModelPhysics::from_model(model,model_instance){
|
||||||
|
self.physics.models.push(model_physics);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
println!("Physics Objects: {}",self.physics.models.len());
|
println!("Physics Objects: {}",self.physics.models.len());
|
||||||
@ -375,33 +377,46 @@ impl framework::Example for GraphicsData {
|
|||||||
indexed_models[0].instances.push(ModelInstance{
|
indexed_models[0].instances.push(ModelInstance{
|
||||||
transform:glam::Affine3A::from_translation(glam::vec3(10.,0.,-10.)),
|
transform:glam::Affine3A::from_translation(glam::vec3(10.,0.,-10.)),
|
||||||
color:glam::Vec4::ONE,
|
color:glam::Vec4::ONE,
|
||||||
|
attributes:model::CollisionAttributes::contact(),
|
||||||
});
|
});
|
||||||
//quad monkeys
|
//quad monkeys
|
||||||
indexed_models[1].instances.push(ModelInstance{
|
indexed_models[1].instances.push(ModelInstance{
|
||||||
transform:glam::Affine3A::from_translation(glam::vec3(10.,5.,10.)),
|
transform:glam::Affine3A::from_translation(glam::vec3(10.,5.,10.)),
|
||||||
color:glam::Vec4::ONE,
|
color:glam::Vec4::ONE,
|
||||||
|
attributes:model::CollisionAttributes::contact(),
|
||||||
});
|
});
|
||||||
indexed_models[1].instances.push(ModelInstance{
|
indexed_models[1].instances.push(ModelInstance{
|
||||||
transform:glam::Affine3A::from_translation(glam::vec3(20.,5.,10.)),
|
transform:glam::Affine3A::from_translation(glam::vec3(20.,5.,10.)),
|
||||||
color:glam::vec4(1.0,0.0,0.0,1.0),
|
color:glam::vec4(1.0,0.0,0.0,1.0),
|
||||||
|
attributes:model::CollisionAttributes::contact(),
|
||||||
});
|
});
|
||||||
indexed_models[1].instances.push(ModelInstance{
|
indexed_models[1].instances.push(ModelInstance{
|
||||||
transform:glam::Affine3A::from_translation(glam::vec3(10.,5.,20.)),
|
transform:glam::Affine3A::from_translation(glam::vec3(10.,5.,20.)),
|
||||||
color:glam::vec4(0.0,1.0,0.0,1.0),
|
color:glam::vec4(0.0,1.0,0.0,1.0),
|
||||||
|
attributes:model::CollisionAttributes::contact(),
|
||||||
});
|
});
|
||||||
indexed_models[1].instances.push(ModelInstance{
|
indexed_models[1].instances.push(ModelInstance{
|
||||||
transform:glam::Affine3A::from_translation(glam::vec3(20.,5.,20.)),
|
transform:glam::Affine3A::from_translation(glam::vec3(20.,5.,20.)),
|
||||||
color:glam::vec4(0.0,0.0,1.0,1.0),
|
color:glam::vec4(0.0,0.0,1.0,1.0),
|
||||||
|
attributes:model::CollisionAttributes::contact(),
|
||||||
|
});
|
||||||
|
//decorative monkey
|
||||||
|
indexed_models[1].instances.push(ModelInstance{
|
||||||
|
transform:glam::Affine3A::from_translation(glam::vec3(15.,10.,15.)),
|
||||||
|
color:glam::vec4(0.5,0.5,0.5,0.5),
|
||||||
|
attributes:model::CollisionAttributes::Decoration,
|
||||||
});
|
});
|
||||||
//teapot
|
//teapot
|
||||||
indexed_models[2].instances.push(ModelInstance{
|
indexed_models[2].instances.push(ModelInstance{
|
||||||
transform:glam::Affine3A::from_scale_rotation_translation(glam::vec3(0.5, 1.0, 0.2),glam::quat(-0.22248298016985793,-0.839457167990537,-0.05603504040830783,-0.49261857546227916),glam::vec3(-10.,7.,10.)),
|
transform:glam::Affine3A::from_scale_rotation_translation(glam::vec3(0.5, 1.0, 0.2),glam::quat(-0.22248298016985793,-0.839457167990537,-0.05603504040830783,-0.49261857546227916),glam::vec3(-10.,7.,10.)),
|
||||||
color:glam::Vec4::ONE,
|
color:glam::Vec4::ONE,
|
||||||
|
attributes:model::CollisionAttributes::contact(),
|
||||||
});
|
});
|
||||||
//ground
|
//ground
|
||||||
indexed_models[3].instances.push(ModelInstance{
|
indexed_models[3].instances.push(ModelInstance{
|
||||||
transform:glam::Affine3A::from_translation(glam::vec3(0.,0.,0.))*glam::Affine3A::from_scale(glam::vec3(160.0, 1.0, 160.0)),
|
transform:glam::Affine3A::from_translation(glam::vec3(0.,0.,0.))*glam::Affine3A::from_scale(glam::vec3(160.0, 1.0, 160.0)),
|
||||||
color:glam::Vec4::ONE,
|
color:glam::Vec4::ONE,
|
||||||
|
attributes:model::CollisionAttributes::contact(),
|
||||||
});
|
});
|
||||||
|
|
||||||
let camera_bind_group_layout = device.create_bind_group_layout(&wgpu::BindGroupLayoutDescriptor {
|
let camera_bind_group_layout = device.create_bind_group_layout(&wgpu::BindGroupLayoutDescriptor {
|
||||||
@ -503,22 +518,17 @@ impl framework::Example for GraphicsData {
|
|||||||
spawn_point:glam::vec3(0.0,50.0,0.0),
|
spawn_point:glam::vec3(0.0,50.0,0.0),
|
||||||
body: body::Body::with_pva(glam::vec3(0.0,50.0,0.0),glam::vec3(0.0,0.0,0.0),glam::vec3(0.0,-100.0,0.0)),
|
body: body::Body::with_pva(glam::vec3(0.0,50.0,0.0),glam::vec3(0.0,0.0,0.0),glam::vec3(0.0,-100.0,0.0)),
|
||||||
time: 0,
|
time: 0,
|
||||||
tick: 0,
|
style:body::StyleModifiers::default(),
|
||||||
strafe_tick_num: 100,//100t
|
|
||||||
strafe_tick_den: 1_000_000_000,
|
|
||||||
gravity: glam::vec3(0.0,-100.0,0.0),
|
|
||||||
friction: 1.2,
|
|
||||||
walk_accel: 90.0,
|
|
||||||
mv: 2.7,
|
|
||||||
grounded: false,
|
grounded: false,
|
||||||
walkspeed: 18.0,
|
|
||||||
contacts: std::collections::HashSet::new(),
|
contacts: std::collections::HashSet::new(),
|
||||||
models: Vec::new(),
|
models: Vec::new(),
|
||||||
walk: body::WalkState::new(),
|
walk: body::WalkState::new(),
|
||||||
hitbox_halfsize: glam::vec3(1.0,2.5,1.0),
|
|
||||||
camera: body::Camera::from_offset(glam::vec3(0.0,4.5-2.5,0.0),(config.width as f32)/(config.height as f32)),
|
camera: body::Camera::from_offset(glam::vec3(0.0,4.5-2.5,0.0),(config.width as f32)/(config.height as f32)),
|
||||||
mouse_interpolation: body::MouseInterpolationState::new(),
|
mouse_interpolation: body::MouseInterpolationState::new(),
|
||||||
controls: 0,
|
controls: 0,
|
||||||
|
world:body::WorldState{},
|
||||||
|
game:body::GameMechanicsState::default(),
|
||||||
|
stages:Vec::new(),
|
||||||
};
|
};
|
||||||
|
|
||||||
//load textures
|
//load textures
|
||||||
@ -774,6 +784,7 @@ impl framework::Example for GraphicsData {
|
|||||||
textures:Vec::new(),
|
textures:Vec::new(),
|
||||||
models:indexed_models,
|
models:indexed_models,
|
||||||
spawn_point:glam::Vec3::Y*50.0,
|
spawn_point:glam::Vec3::Y*50.0,
|
||||||
|
stages:Vec::new(),
|
||||||
};
|
};
|
||||||
graphics.generate_model_physics(&indexed_model_instances);
|
graphics.generate_model_physics(&indexed_model_instances);
|
||||||
graphics.generate_model_graphics(&device,&queue,indexed_model_instances);
|
graphics.generate_model_graphics(&device,&queue,indexed_model_instances);
|
||||||
|
120
src/model.rs
120
src/model.rs
@ -56,15 +56,131 @@ pub struct ModelGraphicsInstance{
|
|||||||
pub color:glam::Vec4,
|
pub color:glam::Vec4,
|
||||||
}
|
}
|
||||||
pub struct ModelInstance{
|
pub struct ModelInstance{
|
||||||
|
//pub id:u64,//this does not actually help with map fixes resimulating bots, they must always be resimulated
|
||||||
pub transform:glam::Affine3A,
|
pub transform:glam::Affine3A,
|
||||||
pub color:glam::Vec4,
|
pub color:glam::Vec4,//transparency is in here
|
||||||
|
pub attributes:CollisionAttributes,
|
||||||
}
|
}
|
||||||
pub struct IndexedModelInstances{
|
pub struct IndexedModelInstances{
|
||||||
pub textures:Vec<String>,//RenderPattern
|
pub textures:Vec<String>,//RenderPattern
|
||||||
pub models:Vec<IndexedModel>,
|
pub models:Vec<IndexedModel>,
|
||||||
//object_index for spawns, triggers etc?
|
//may make this into an object later.
|
||||||
|
pub stages:Vec<StageDescription>,
|
||||||
pub spawn_point:glam::Vec3,
|
pub spawn_point:glam::Vec3,
|
||||||
}
|
}
|
||||||
|
//stage description referencing flattened ids is spooky, but the map loading is meant to be deterministic.
|
||||||
|
pub struct StageDescription{
|
||||||
|
pub start:u32,//start=model_id
|
||||||
|
pub spawns:Vec<u32>,//spawns[spawn_id]=model_id
|
||||||
|
pub ordered_checkpoints:Vec<u32>,//ordered_checkpoints[checkpoint_id]=model_id
|
||||||
|
pub unordered_checkpoints:Vec<u32>,//unordered_checkpoints[checkpoint_id]=model_id
|
||||||
|
}
|
||||||
|
|
||||||
|
//you have this effect while in contact
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct ContactingSurf{}
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct ContactingLadder{
|
||||||
|
pub sticky:bool
|
||||||
|
}
|
||||||
|
//you have this effect while intersecting
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct IntersectingWater{
|
||||||
|
pub viscosity:i64,
|
||||||
|
pub density:i64,
|
||||||
|
pub current:glam::Vec3,
|
||||||
|
}
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct IntersectingAccelerator{
|
||||||
|
pub acceleration:glam::Vec3
|
||||||
|
}
|
||||||
|
//All models can be given these attributes
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct GameMechanicJumpLimit{
|
||||||
|
pub count:u32,
|
||||||
|
}
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct GameMechanicBooster{
|
||||||
|
pub velocity:glam::Vec3,
|
||||||
|
}
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub enum ZoneBehaviour{
|
||||||
|
//Start is indexed
|
||||||
|
//Checkpoints are indexed
|
||||||
|
Finish,
|
||||||
|
Anitcheat,
|
||||||
|
}
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct GameMechanicZone{
|
||||||
|
pub mode_id:u32,
|
||||||
|
pub behaviour:ZoneBehaviour,
|
||||||
|
}
|
||||||
|
// enum TrapCondition{
|
||||||
|
// FasterThan(i64),
|
||||||
|
// SlowerThan(i64),
|
||||||
|
// InRange(i64,i64),
|
||||||
|
// OutsideRange(i64,i64),
|
||||||
|
// }
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub enum StageElementBehaviour{
|
||||||
|
//Spawn,//The behaviour of stepping on a spawn setting the spawnid
|
||||||
|
SpawnAt,
|
||||||
|
Trigger,
|
||||||
|
Teleport,
|
||||||
|
Platform,
|
||||||
|
//Speedtrap(TrapCondition),//Acts as a trigger with a speed condition
|
||||||
|
}
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct GameMechanicStageElement{
|
||||||
|
pub mode_id:u32,
|
||||||
|
pub stage_id:u32,//which spawn to send to
|
||||||
|
pub force:bool,//allow setting to lower spawn id i.e. 7->3
|
||||||
|
pub behaviour:StageElementBehaviour
|
||||||
|
}
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct GameMechanicWormhole{//(position,angles)*=origin.transform.inverse()*destination.transform
|
||||||
|
pub model_id:u32,
|
||||||
|
}
|
||||||
|
#[derive(Default,Clone)]
|
||||||
|
pub struct GameMechanicAttributes{
|
||||||
|
pub jump_limit:Option<GameMechanicJumpLimit>,
|
||||||
|
pub booster:Option<GameMechanicBooster>,
|
||||||
|
pub zone:Option<GameMechanicZone>,
|
||||||
|
pub stage_element:Option<GameMechanicStageElement>,
|
||||||
|
pub wormhole:Option<GameMechanicWormhole>,//stage_element and wormhole are in conflict
|
||||||
|
}
|
||||||
|
#[derive(Default,Clone)]
|
||||||
|
pub struct ContactingAttributes{
|
||||||
|
pub elasticity:Option<u32>,//[1/2^32,1] 0=None (elasticity+1)/2^32
|
||||||
|
//friction?
|
||||||
|
pub surf:Option<ContactingSurf>,
|
||||||
|
pub ladder:Option<ContactingLadder>,
|
||||||
|
}
|
||||||
|
#[derive(Default,Clone)]
|
||||||
|
pub struct IntersectingAttributes{
|
||||||
|
pub water:Option<IntersectingWater>,
|
||||||
|
pub accelerator:Option<IntersectingAccelerator>,
|
||||||
|
}
|
||||||
|
//Spawn(u32) NO! spawns are indexed in the map header instead of marked with attibutes
|
||||||
|
pub enum CollisionAttributes{
|
||||||
|
Decoration,//visual only
|
||||||
|
Contact{//track whether you are contacting the object
|
||||||
|
contacting:ContactingAttributes,
|
||||||
|
general:GameMechanicAttributes,
|
||||||
|
},
|
||||||
|
Intersect{//track whether you are intersecting the object
|
||||||
|
intersecting:IntersectingAttributes,
|
||||||
|
general:GameMechanicAttributes,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
impl CollisionAttributes{
|
||||||
|
pub fn contact() -> Self {
|
||||||
|
Self::Contact{
|
||||||
|
contacting:ContactingAttributes::default(),
|
||||||
|
general:GameMechanicAttributes::default()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn generate_indexed_model_list_from_obj(data:obj::ObjData,color:[f32;4]) -> Vec<IndexedModel>{
|
pub fn generate_indexed_model_list_from_obj(data:obj::ObjData,color:[f32;4]) -> Vec<IndexedModel>{
|
||||||
let mut unique_vertex_index = std::collections::HashMap::<obj::IndexTuple,u32>::new();
|
let mut unique_vertex_index = std::collections::HashMap::<obj::IndexTuple,u32>::new();
|
||||||
|
Loading…
Reference in New Issue
Block a user