Compare commits
91 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
493cf242c6
|
|||
|
9b3a9ce437
|
|||
|
73f0a85f81
|
|||
|
332e11a627
|
|||
|
2d7f5cdc7d
|
|||
|
822f4571d2
|
|||
|
ac4c84f562
|
|||
|
3a7aaa20f8
|
|||
|
2e77366a17
|
|||
|
8a1f434a2a
|
|||
|
80d3df4659
|
|||
|
47f94fe359
|
|||
|
c4a2811627
|
|||
|
b23d4d590d
|
|||
|
cb9307bdf2
|
|||
|
ae7582704b
|
|||
|
7f0f63570e
|
|||
|
dbe96a0451
|
|||
|
291cedff3f
|
|||
|
43a2c76906
|
|||
|
7d8dbf7e82
|
|||
|
7640ea824d
|
|||
|
f3b02bba92
|
|||
|
9d04df4894
|
|||
|
f90436f0cc
|
|||
|
5e14a85d84
|
|||
|
da718e4bd2
|
|||
|
5641e9a26f
|
|||
|
7e7839f4aa
|
|||
|
6448d7cc57
|
|||
|
b8be169092
|
|||
|
9c4c14c5dc
|
|||
|
ca40e65060
|
|||
|
492e72c1bc
|
|||
|
03bf2650fd
|
|||
|
0c8cf02287
|
|||
|
af1374906b
|
|||
|
570d33a030
|
|||
|
d0c38a6e66
|
|||
|
dd7a636fa9
|
|||
|
2483abe2ad
|
|||
|
6621e369f2
|
|||
|
1b833ef6b3
|
|||
|
8a1ab4e03c
|
|||
|
069db75d3a
|
|||
|
de54bcfc36
|
|||
|
e777b89c6d
|
|||
|
d17153d17d
|
|||
|
e514c27675
|
|||
|
d1c13757e0
|
|||
|
d93e558678
|
|||
|
0048306236
|
|||
|
ee50f8dc1e
|
|||
|
e4966b037f
|
|||
|
96cdd684d1
|
|||
|
838130fec4
|
|||
|
8a3badc270
|
|||
|
4470e88d7b
|
|||
|
df7bee6cd1
|
|||
|
4fdd254f2a
|
|||
|
8a9db203fa
|
|||
|
be05fd108a
|
|||
|
6160872469
|
|||
|
2c1fa5da22
|
|||
|
6fe45f4873
|
|||
|
e1dac67aa0
|
|||
|
f9ed33073e
|
|||
|
91636747d4
|
|||
|
03b72301a3
|
|||
|
214b23f780
|
|||
|
6d98407830
|
|||
|
978659e8c6
|
|||
|
d00871f87f
|
|||
|
d2ed97fcf2
|
|||
|
ab3c693f84
|
|||
|
f0c7677a77
|
|||
|
eed0abcc2c
|
|||
|
bc5cdc7313
|
|||
|
14a5a3f964
|
|||
|
18b7bba901
|
|||
|
1770ac7292
|
|||
|
7384886512
|
|||
|
1dbde609cc
|
|||
|
e024f37843
|
|||
|
4059cfa527
|
|||
|
e4f3418bc6
|
|||
|
6ca6d5e484
|
|||
|
0668ac2def
|
|||
|
73e3181d0c
|
|||
|
19ba8f2445
|
|||
|
0495d07e26
|
900
Cargo.lock
generated
900
Cargo.lock
generated
File diff suppressed because it is too large
Load Diff
@@ -11,7 +11,7 @@ id = { version = "0.1.0", registry = "strafesnet" }
|
|||||||
strafesnet_common = { path = "../../lib/common", registry = "strafesnet" }
|
strafesnet_common = { path = "../../lib/common", registry = "strafesnet" }
|
||||||
strafesnet_session = { path = "../session", registry = "strafesnet" }
|
strafesnet_session = { path = "../session", registry = "strafesnet" }
|
||||||
strafesnet_settings = { path = "../settings", registry = "strafesnet" }
|
strafesnet_settings = { path = "../settings", registry = "strafesnet" }
|
||||||
wgpu = "27.0.0"
|
wgpu = "28.0.0"
|
||||||
|
|
||||||
[lints]
|
[lints]
|
||||||
workspace = true
|
workspace = true
|
||||||
|
|||||||
@@ -616,7 +616,7 @@ impl GraphicsState{
|
|||||||
address_mode_w:wgpu::AddressMode::ClampToEdge,
|
address_mode_w:wgpu::AddressMode::ClampToEdge,
|
||||||
mag_filter:wgpu::FilterMode::Linear,
|
mag_filter:wgpu::FilterMode::Linear,
|
||||||
min_filter:wgpu::FilterMode::Linear,
|
min_filter:wgpu::FilterMode::Linear,
|
||||||
mipmap_filter:wgpu::FilterMode::Linear,
|
mipmap_filter:wgpu::MipmapFilterMode::Linear,
|
||||||
..Default::default()
|
..Default::default()
|
||||||
});
|
});
|
||||||
let repeat_sampler=device.create_sampler(&wgpu::SamplerDescriptor{
|
let repeat_sampler=device.create_sampler(&wgpu::SamplerDescriptor{
|
||||||
@@ -626,7 +626,7 @@ impl GraphicsState{
|
|||||||
address_mode_w:wgpu::AddressMode::Repeat,
|
address_mode_w:wgpu::AddressMode::Repeat,
|
||||||
mag_filter:wgpu::FilterMode::Linear,
|
mag_filter:wgpu::FilterMode::Linear,
|
||||||
min_filter:wgpu::FilterMode::Linear,
|
min_filter:wgpu::FilterMode::Linear,
|
||||||
mipmap_filter:wgpu::FilterMode::Linear,
|
mipmap_filter:wgpu::MipmapFilterMode::Linear,
|
||||||
anisotropy_clamp:16,
|
anisotropy_clamp:16,
|
||||||
..Default::default()
|
..Default::default()
|
||||||
});
|
});
|
||||||
@@ -754,7 +754,7 @@ impl GraphicsState{
|
|||||||
&skybox_texture_bind_group_layout,
|
&skybox_texture_bind_group_layout,
|
||||||
&model_bind_group_layout,
|
&model_bind_group_layout,
|
||||||
],
|
],
|
||||||
push_constant_ranges:&[],
|
immediate_size:0,
|
||||||
});
|
});
|
||||||
let sky_pipeline_layout=device.create_pipeline_layout(&wgpu::PipelineLayoutDescriptor{
|
let sky_pipeline_layout=device.create_pipeline_layout(&wgpu::PipelineLayoutDescriptor{
|
||||||
label:None,
|
label:None,
|
||||||
@@ -762,7 +762,7 @@ impl GraphicsState{
|
|||||||
&camera_bind_group_layout,
|
&camera_bind_group_layout,
|
||||||
&skybox_texture_bind_group_layout,
|
&skybox_texture_bind_group_layout,
|
||||||
],
|
],
|
||||||
push_constant_ranges:&[],
|
immediate_size:0,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Create the render pipelines
|
// Create the render pipelines
|
||||||
@@ -793,7 +793,7 @@ impl GraphicsState{
|
|||||||
bias:wgpu::DepthBiasState::default(),
|
bias:wgpu::DepthBiasState::default(),
|
||||||
}),
|
}),
|
||||||
multisample:wgpu::MultisampleState::default(),
|
multisample:wgpu::MultisampleState::default(),
|
||||||
multiview:None,
|
multiview_mask:None,
|
||||||
cache:None,
|
cache:None,
|
||||||
});
|
});
|
||||||
let model_pipeline=device.create_render_pipeline(&wgpu::RenderPipelineDescriptor{
|
let model_pipeline=device.create_render_pipeline(&wgpu::RenderPipelineDescriptor{
|
||||||
@@ -828,7 +828,7 @@ impl GraphicsState{
|
|||||||
bias:wgpu::DepthBiasState::default(),
|
bias:wgpu::DepthBiasState::default(),
|
||||||
}),
|
}),
|
||||||
multisample:wgpu::MultisampleState::default(),
|
multisample:wgpu::MultisampleState::default(),
|
||||||
multiview:None,
|
multiview_mask:None,
|
||||||
cache:None,
|
cache:None,
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -880,7 +880,7 @@ impl GraphicsState{
|
|||||||
camera_buf,
|
camera_buf,
|
||||||
models:Vec::new(),
|
models:Vec::new(),
|
||||||
depth_view,
|
depth_view,
|
||||||
staging_belt:wgpu::util::StagingBelt::new(0x100),
|
staging_belt:wgpu::util::StagingBelt::new(device.clone(),0x100),
|
||||||
bind_group_layouts:GraphicsBindGroupLayouts{model:model_bind_group_layout},
|
bind_group_layouts:GraphicsBindGroupLayouts{model:model_bind_group_layout},
|
||||||
samplers:GraphicsSamplers{repeat:repeat_sampler},
|
samplers:GraphicsSamplers{repeat:repeat_sampler},
|
||||||
temp_squid_texture_view:squid_texture_view,
|
temp_squid_texture_view:squid_texture_view,
|
||||||
@@ -918,7 +918,6 @@ impl GraphicsState{
|
|||||||
&self.camera_buf,
|
&self.camera_buf,
|
||||||
0,
|
0,
|
||||||
wgpu::BufferSize::new((camera_uniforms.len() * 4) as wgpu::BufferAddress).unwrap(),
|
wgpu::BufferSize::new((camera_uniforms.len() * 4) as wgpu::BufferAddress).unwrap(),
|
||||||
device,
|
|
||||||
)
|
)
|
||||||
.copy_from_slice(bytemuck::cast_slice(&camera_uniforms));
|
.copy_from_slice(bytemuck::cast_slice(&camera_uniforms));
|
||||||
//This code only needs to run when the uniforms change
|
//This code only needs to run when the uniforms change
|
||||||
@@ -965,6 +964,7 @@ impl GraphicsState{
|
|||||||
}),
|
}),
|
||||||
timestamp_writes:Default::default(),
|
timestamp_writes:Default::default(),
|
||||||
occlusion_query_set:Default::default(),
|
occlusion_query_set:Default::default(),
|
||||||
|
multiview_mask:None,
|
||||||
});
|
});
|
||||||
|
|
||||||
rpass.set_bind_group(0,&self.bind_groups.camera,&[]);
|
rpass.set_bind_group(0,&self.bind_groups.camera,&[]);
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ edition = "2024"
|
|||||||
arrayvec = "0.7.6"
|
arrayvec = "0.7.6"
|
||||||
glam = "0.30.0"
|
glam = "0.30.0"
|
||||||
id = { version = "0.1.0", registry = "strafesnet" }
|
id = { version = "0.1.0", registry = "strafesnet" }
|
||||||
mlua = { version = "0.11.5", features = ["luau"] }
|
|
||||||
strafesnet_common = { path = "../../lib/common", registry = "strafesnet" }
|
strafesnet_common = { path = "../../lib/common", registry = "strafesnet" }
|
||||||
|
|
||||||
[lints]
|
[lints]
|
||||||
|
|||||||
@@ -21,12 +21,6 @@ impl<M:MeshQuery> CrawlResult<M>{
|
|||||||
CrawlResult::Hit(face,time)=>Some((face,time)),
|
CrawlResult::Hit(face,time)=>Some((face,time)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pub fn miss(self)->Option<FEV<M>>{
|
|
||||||
match self{
|
|
||||||
CrawlResult::Miss(fev)=>Some(fev),
|
|
||||||
CrawlResult::Hit(_,_)=>None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: move predict_collision_face_out algorithm in here or something
|
// TODO: move predict_collision_face_out algorithm in here or something
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ mod face_crawler;
|
|||||||
mod model;
|
mod model;
|
||||||
mod push_solve;
|
mod push_solve;
|
||||||
mod minimum_difference;
|
mod minimum_difference;
|
||||||
mod minimum_difference_lua;
|
|
||||||
|
|
||||||
pub mod physics;
|
pub mod physics;
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,23 @@
|
|||||||
use strafesnet_common::integer::vec3;
|
use strafesnet_common::integer::vec3;
|
||||||
use strafesnet_common::integer::vec3::Vector3;
|
use strafesnet_common::integer::vec3::Vector3;
|
||||||
use strafesnet_common::integer::{Fixed,Planar64Vec3};
|
use strafesnet_common::integer::{Fixed,Planar64,Planar64Vec3};
|
||||||
|
|
||||||
use crate::model::{MeshQuery,MinkowskiMesh,MinkowskiVert};
|
use crate::model::{DirectedEdge,FEV,MeshQuery};
|
||||||
|
// TODO: remove mesh invert
|
||||||
|
use crate::model::{MinkowskiMesh,MinkowskiVert};
|
||||||
|
|
||||||
// This algorithm is based on Lua code
|
// This algorithm is based on Lua code
|
||||||
// written by Trey Reynolds in 2021
|
// written by Trey Reynolds in 2021
|
||||||
|
|
||||||
type Simplex<const N:usize>=[MinkowskiVert;N];
|
type Simplex<const N:usize,Vert>=[Vert;N];
|
||||||
enum Simplex1_3{
|
#[derive(Clone,Copy)]
|
||||||
Simplex1(Simplex<1>),
|
enum Simplex1_3<Vert>{
|
||||||
Simplex2(Simplex<2>),
|
Simplex1(Simplex<1,Vert>),
|
||||||
Simplex3(Simplex<3>),
|
Simplex2(Simplex<2,Vert>),
|
||||||
|
Simplex3(Simplex<3,Vert>),
|
||||||
}
|
}
|
||||||
impl Simplex1_3{
|
impl<Vert> Simplex1_3<Vert>{
|
||||||
fn push_front(self,v:MinkowskiVert)->Simplex2_4{
|
fn push_front(self,v:Vert)->Simplex2_4<Vert>{
|
||||||
match self{
|
match self{
|
||||||
Simplex1_3::Simplex1([v0])=>Simplex2_4::Simplex2([v,v0]),
|
Simplex1_3::Simplex1([v0])=>Simplex2_4::Simplex2([v,v0]),
|
||||||
Simplex1_3::Simplex2([v0,v1])=>Simplex2_4::Simplex3([v,v0,v1]),
|
Simplex1_3::Simplex2([v0,v1])=>Simplex2_4::Simplex3([v,v0,v1]),
|
||||||
@@ -22,10 +25,11 @@ impl Simplex1_3{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
enum Simplex2_4{
|
#[derive(Clone,Copy)]
|
||||||
Simplex2(Simplex<2>),
|
enum Simplex2_4<Vert>{
|
||||||
Simplex3(Simplex<3>),
|
Simplex2(Simplex<2,Vert>),
|
||||||
Simplex4(Simplex<4>),
|
Simplex3(Simplex<3,Vert>),
|
||||||
|
Simplex4(Simplex<4,Vert>),
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -41,23 +45,23 @@ local function absDet(r, u, v, w)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
*/
|
*/
|
||||||
impl Simplex2_4{
|
impl<Vert> Simplex2_4<Vert>{
|
||||||
fn det_is_zero(&self,mesh:&MinkowskiMesh)->bool{
|
fn det_is_zero<M:MeshQuery<Vert=Vert>>(self,mesh:&M)->bool{
|
||||||
match self{
|
match self{
|
||||||
&Self::Simplex4([p0,p1,p2,p3])=>{
|
Self::Simplex4([p0,p1,p2,p3])=>{
|
||||||
let p0=mesh.vert(p0);
|
let p0=mesh.vert(p0);
|
||||||
let p1=mesh.vert(p1);
|
let p1=mesh.vert(p1);
|
||||||
let p2=mesh.vert(p2);
|
let p2=mesh.vert(p2);
|
||||||
let p3=mesh.vert(p3);
|
let p3=mesh.vert(p3);
|
||||||
(p1-p0).cross(p2-p0).dot(p3-p0)==Fixed::ZERO
|
(p1-p0).cross(p2-p0).dot(p3-p0)==Fixed::ZERO
|
||||||
},
|
},
|
||||||
&Self::Simplex3([p0,p1,p2])=>{
|
Self::Simplex3([p0,p1,p2])=>{
|
||||||
let p0=mesh.vert(p0);
|
let p0=mesh.vert(p0);
|
||||||
let p1=mesh.vert(p1);
|
let p1=mesh.vert(p1);
|
||||||
let p2=mesh.vert(p2);
|
let p2=mesh.vert(p2);
|
||||||
(p1-p0).cross(p2-p0)==vec3::zero()
|
(p1-p0).cross(p2-p0)==vec3::zero()
|
||||||
},
|
},
|
||||||
&Self::Simplex2([p0,p1])=>{
|
Self::Simplex2([p0,p1])=>{
|
||||||
let p0=mesh.vert(p0);
|
let p0=mesh.vert(p0);
|
||||||
let p1=mesh.vert(p1);
|
let p1=mesh.vert(p1);
|
||||||
p1-p0==vec3::zero()
|
p1-p0==vec3::zero()
|
||||||
@@ -96,14 +100,45 @@ const fn choose_any_direction()->Planar64Vec3{
|
|||||||
vec3::X
|
vec3::X
|
||||||
}
|
}
|
||||||
|
|
||||||
fn reduce1(
|
fn narrow_dir2(dir:Vector3<Fixed<2,64>>)->Planar64Vec3{
|
||||||
[v0]:Simplex<1>,
|
if dir==vec3::zero(){
|
||||||
mesh:&MinkowskiMesh,
|
return dir.narrow_1().unwrap();
|
||||||
|
}
|
||||||
|
let x=dir.x.as_bits().unsigned_abs().bits();
|
||||||
|
let y=dir.y.as_bits().unsigned_abs().bits();
|
||||||
|
let z=dir.z.as_bits().unsigned_abs().bits();
|
||||||
|
let big=x.max(y).max(z);
|
||||||
|
const MAX_BITS:u32=64+31;
|
||||||
|
if MAX_BITS<big{
|
||||||
|
dir>>(big-MAX_BITS)
|
||||||
|
}else{
|
||||||
|
dir
|
||||||
|
}.narrow_1().unwrap()
|
||||||
|
}
|
||||||
|
fn narrow_dir3(dir:Vector3<Fixed<3,96>>)->Planar64Vec3{
|
||||||
|
if dir==vec3::zero(){
|
||||||
|
return dir.narrow_1().unwrap();
|
||||||
|
}
|
||||||
|
let x=dir.x.as_bits().unsigned_abs().bits();
|
||||||
|
let y=dir.y.as_bits().unsigned_abs().bits();
|
||||||
|
let z=dir.z.as_bits().unsigned_abs().bits();
|
||||||
|
let big=x.max(y).max(z);
|
||||||
|
const MAX_BITS:u32=96+31;
|
||||||
|
if MAX_BITS<big{
|
||||||
|
dir>>(big-MAX_BITS)
|
||||||
|
}else{
|
||||||
|
dir
|
||||||
|
}.narrow_1().unwrap()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn reduce1<M:MeshQuery>(
|
||||||
|
[v0]:Simplex<1,M::Vert>,
|
||||||
|
mesh:&M,
|
||||||
point:Planar64Vec3,
|
point:Planar64Vec3,
|
||||||
)->Reduced{
|
)->Reduced<M::Vert>{
|
||||||
// --debug.profilebegin("reduceSimplex0")
|
// --debug.profilebegin("reduceSimplex0")
|
||||||
// local a = a1 - a0
|
// local a = a1 - a0
|
||||||
let p0=-mesh.vert(v0);
|
let p0=mesh.vert(v0);
|
||||||
|
|
||||||
// local p = -a
|
// local p = -a
|
||||||
let p=-(p0+point);
|
let p=-(p0+point);
|
||||||
@@ -125,16 +160,16 @@ fn reduce1(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// local function reduceSimplex1(a0, a1, b0, b1)
|
// local function reduceSimplex1(a0, a1, b0, b1)
|
||||||
fn reduce2(
|
fn reduce2<M:MeshQuery>(
|
||||||
[v0,v1]:Simplex<2>,
|
[v0,v1]:Simplex<2,M::Vert>,
|
||||||
mesh:&MinkowskiMesh,
|
mesh:&M,
|
||||||
point:Planar64Vec3,
|
point:Planar64Vec3,
|
||||||
)->Reduced{
|
)->Reduced<M::Vert>{
|
||||||
// --debug.profilebegin("reduceSimplex1")
|
// --debug.profilebegin("reduceSimplex1")
|
||||||
// local a = a1 - a0
|
// local a = a1 - a0
|
||||||
// local b = b1 - b0
|
// local b = b1 - b0
|
||||||
let p0=-mesh.vert(v0);
|
let p0=mesh.vert(v0);
|
||||||
let p1=-mesh.vert(v1);
|
let p1=mesh.vert(v1);
|
||||||
|
|
||||||
// local p = -a
|
// local p = -a
|
||||||
// local u = b - a
|
// local u = b - a
|
||||||
@@ -161,7 +196,7 @@ fn reduce2(
|
|||||||
// -- modify the direction to take into account a0R and b0R
|
// -- modify the direction to take into account a0R and b0R
|
||||||
// return direction, a0, a1, b0, b1
|
// return direction, a0, a1, b0, b1
|
||||||
return Reduced{
|
return Reduced{
|
||||||
dir:direction.narrow_1().unwrap(),
|
dir:narrow_dir3(direction),
|
||||||
simplex:Simplex1_3::Simplex2([v0,v1]),
|
simplex:Simplex1_3::Simplex2([v0,v1]),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -182,18 +217,18 @@ fn reduce2(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// local function reduceSimplex2(a0, a1, b0, b1, c0, c1)
|
// local function reduceSimplex2(a0, a1, b0, b1, c0, c1)
|
||||||
fn reduce3(
|
fn reduce3<M:MeshQuery>(
|
||||||
[v0,mut v1,v2]:Simplex<3>,
|
[v0,mut v1,v2]:Simplex<3,M::Vert>,
|
||||||
mesh:&MinkowskiMesh,
|
mesh:&M,
|
||||||
point:Planar64Vec3,
|
point:Planar64Vec3,
|
||||||
)->Reduced{
|
)->Reduced<M::Vert>{
|
||||||
// --debug.profilebegin("reduceSimplex2")
|
// --debug.profilebegin("reduceSimplex2")
|
||||||
// local a = a1 - a0
|
// local a = a1 - a0
|
||||||
// local b = b1 - b0
|
// local b = b1 - b0
|
||||||
// local c = c1 - c0
|
// local c = c1 - c0
|
||||||
let p0=-mesh.vert(v0);
|
let p0=mesh.vert(v0);
|
||||||
let p1=-mesh.vert(v1);
|
let p1=mesh.vert(v1);
|
||||||
let p2=-mesh.vert(v2);
|
let p2=mesh.vert(v2);
|
||||||
|
|
||||||
// local p = -a
|
// local p = -a
|
||||||
// local u = b - a
|
// local u = b - a
|
||||||
@@ -227,7 +262,7 @@ fn reduce3(
|
|||||||
|
|
||||||
// return direction, a0, a1, b0, b1, c0, c1
|
// return direction, a0, a1, b0, b1, c0, c1
|
||||||
return Reduced{
|
return Reduced{
|
||||||
dir:direction.narrow_1().unwrap(),
|
dir:narrow_dir2(direction),
|
||||||
simplex:Simplex1_3::Simplex3([v0,v1,v2]),
|
simplex:Simplex1_3::Simplex3([v0,v1,v2]),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -261,14 +296,14 @@ fn reduce3(
|
|||||||
if direction==vec3::zero(){
|
if direction==vec3::zero(){
|
||||||
// direction = uv
|
// direction = uv
|
||||||
return Reduced{
|
return Reduced{
|
||||||
dir:uv.narrow_1().unwrap(),
|
dir:narrow_dir2(uv),
|
||||||
simplex:Simplex1_3::Simplex2([v0,v1]),
|
simplex:Simplex1_3::Simplex2([v0,v1]),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// return direction, a0, a1, b0, b1
|
// return direction, a0, a1, b0, b1
|
||||||
return Reduced{
|
return Reduced{
|
||||||
dir:direction.narrow_1().unwrap(),
|
dir:narrow_dir3(direction),
|
||||||
simplex:Simplex1_3::Simplex2([v0,v1]),
|
simplex:Simplex1_3::Simplex2([v0,v1]),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -279,7 +314,7 @@ fn reduce3(
|
|||||||
if dir==vec3::zero(){
|
if dir==vec3::zero(){
|
||||||
// direction = uv
|
// direction = uv
|
||||||
return Reduced{
|
return Reduced{
|
||||||
dir:uv.narrow_1().unwrap(),
|
dir:narrow_dir2(uv),
|
||||||
simplex:Simplex1_3::Simplex1([v0]),
|
simplex:Simplex1_3::Simplex1([v0]),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -291,20 +326,20 @@ fn reduce3(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// local function reduceSimplex3(a0, a1, b0, b1, c0, c1, d0, d1)
|
// local function reduceSimplex3(a0, a1, b0, b1, c0, c1, d0, d1)
|
||||||
fn reduce4(
|
fn reduce4<M:MeshQuery>(
|
||||||
[v0,mut v1,mut v2,v3]:Simplex<4>,
|
[v0,mut v1,mut v2,v3]:Simplex<4,M::Vert>,
|
||||||
mesh:&MinkowskiMesh,
|
mesh:&M,
|
||||||
point:Planar64Vec3,
|
point:Planar64Vec3,
|
||||||
)->Reduce{
|
)->Reduce<M::Vert>{
|
||||||
// --debug.profilebegin("reduceSimplex3")
|
// --debug.profilebegin("reduceSimplex3")
|
||||||
// local a = a1 - a0
|
// local a = a1 - a0
|
||||||
// local b = b1 - b0
|
// local b = b1 - b0
|
||||||
// local c = c1 - c0
|
// local c = c1 - c0
|
||||||
// local d = d1 - d0
|
// local d = d1 - d0
|
||||||
let p0=-mesh.vert(v0);
|
let p0=mesh.vert(v0);
|
||||||
let p1=-mesh.vert(v1);
|
let p1=mesh.vert(v1);
|
||||||
let p2=-mesh.vert(v2);
|
let p2=mesh.vert(v2);
|
||||||
let p3=-mesh.vert(v3);
|
let p3=mesh.vert(v3);
|
||||||
|
|
||||||
// local p = -a
|
// local p = -a
|
||||||
// local u = b - a
|
// local u = b - a
|
||||||
@@ -395,12 +430,12 @@ fn reduce4(
|
|||||||
// return direction, a0, a1, b0, b1, c0, c1
|
// return direction, a0, a1, b0, b1, c0, c1
|
||||||
if uv_w.is_negative(){
|
if uv_w.is_negative(){
|
||||||
return Reduce::Reduced(Reduced{
|
return Reduce::Reduced(Reduced{
|
||||||
dir:uv.narrow_1().unwrap(),
|
dir:narrow_dir2(uv),
|
||||||
simplex:Simplex1_3::Simplex3([v0,v1,v2]),
|
simplex:Simplex1_3::Simplex3([v0,v1,v2]),
|
||||||
});
|
});
|
||||||
}else{
|
}else{
|
||||||
return Reduce::Reduced(Reduced{
|
return Reduce::Reduced(Reduced{
|
||||||
dir:-uv.narrow_1().unwrap(),
|
dir:narrow_dir2(-uv),
|
||||||
simplex:Simplex1_3::Simplex3([v0,v1,v2]),
|
simplex:Simplex1_3::Simplex3([v0,v1,v2]),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -437,12 +472,12 @@ fn reduce4(
|
|||||||
// return direction, a0, a1, b0, b1
|
// return direction, a0, a1, b0, b1
|
||||||
if uv_w.is_negative(){
|
if uv_w.is_negative(){
|
||||||
return Reduce::Reduced(Reduced{
|
return Reduce::Reduced(Reduced{
|
||||||
dir:uv.narrow_1().unwrap(),
|
dir:narrow_dir2(uv),
|
||||||
simplex:Simplex1_3::Simplex2([v0,v1]),
|
simplex:Simplex1_3::Simplex2([v0,v1]),
|
||||||
});
|
});
|
||||||
}else{
|
}else{
|
||||||
return Reduce::Reduced(Reduced{
|
return Reduce::Reduced(Reduced{
|
||||||
dir:-uv.narrow_1().unwrap(),
|
dir:narrow_dir2(-uv),
|
||||||
simplex:Simplex1_3::Simplex2([v0,v1]),
|
simplex:Simplex1_3::Simplex2([v0,v1]),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -450,7 +485,7 @@ fn reduce4(
|
|||||||
|
|
||||||
// return direction, a0, a1, b0, b1
|
// return direction, a0, a1, b0, b1
|
||||||
return Reduce::Reduced(Reduced{
|
return Reduce::Reduced(Reduced{
|
||||||
dir:direction.narrow_1().unwrap(),
|
dir:narrow_dir3(direction),
|
||||||
simplex:Simplex1_3::Simplex2([v0,v1]),
|
simplex:Simplex1_3::Simplex2([v0,v1]),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -462,12 +497,12 @@ fn reduce4(
|
|||||||
// direction = uvw < 0 and uv or -uv
|
// direction = uvw < 0 and uv or -uv
|
||||||
if uv_w.is_negative(){
|
if uv_w.is_negative(){
|
||||||
return Reduce::Reduced(Reduced{
|
return Reduce::Reduced(Reduced{
|
||||||
dir:uv.narrow_1().unwrap(),
|
dir:narrow_dir2(uv),
|
||||||
simplex:Simplex1_3::Simplex1([v0]),
|
simplex:Simplex1_3::Simplex1([v0]),
|
||||||
});
|
});
|
||||||
}else{
|
}else{
|
||||||
return Reduce::Reduced(Reduced{
|
return Reduce::Reduced(Reduced{
|
||||||
dir:-uv.narrow_1().unwrap(),
|
dir:narrow_dir2(-uv),
|
||||||
simplex:Simplex1_3::Simplex1([v0]),
|
simplex:Simplex1_3::Simplex1([v0]),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -480,18 +515,18 @@ fn reduce4(
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
struct Reduced{
|
struct Reduced<Vert>{
|
||||||
dir:Planar64Vec3,
|
dir:Planar64Vec3,
|
||||||
simplex:Simplex1_3,
|
simplex:Simplex1_3<Vert>,
|
||||||
}
|
}
|
||||||
|
|
||||||
enum Reduce{
|
enum Reduce<Vert>{
|
||||||
Escape(Simplex<4>),
|
Escape(Simplex<4,Vert>),
|
||||||
Reduced(Reduced),
|
Reduced(Reduced<Vert>),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Simplex2_4{
|
impl<Vert> Simplex2_4<Vert>{
|
||||||
fn reduce(self,mesh:&MinkowskiMesh,point:Planar64Vec3)->Reduce{
|
fn reduce<M:MeshQuery<Vert=Vert>>(self,mesh:&M,point:Planar64Vec3)->Reduce<Vert>{
|
||||||
match self{
|
match self{
|
||||||
Self::Simplex2(simplex)=>Reduce::Reduced(reduce2(simplex,mesh,point)),
|
Self::Simplex2(simplex)=>Reduce::Reduced(reduce2(simplex,mesh,point)),
|
||||||
Self::Simplex3(simplex)=>Reduce::Reduced(reduce3(simplex,mesh,point)),
|
Self::Simplex3(simplex)=>Reduce::Reduced(reduce3(simplex,mesh,point)),
|
||||||
@@ -500,57 +535,13 @@ impl Simplex2_4{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// local function expand(
|
pub fn contains_point(mesh:&MinkowskiMesh<'_>,point:Planar64Vec3)->bool{
|
||||||
// queryP, queryQ,
|
|
||||||
// vertA0, vertA1,
|
|
||||||
// vertB0, vertB1,
|
|
||||||
// vertC0, vertC1,
|
|
||||||
// vertD0, vertD1,
|
|
||||||
// accuracy
|
|
||||||
// )
|
|
||||||
fn refine_to_exact(mesh:&MinkowskiMesh,simplex:Simplex<4>)->Simplex2_4{
|
|
||||||
unimplemented!()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Intermediate data structure containing a partially complete calculation.
|
|
||||||
/// Sometimes you only care about the topology, and not about the
|
|
||||||
/// exact point of intersection details.
|
|
||||||
pub struct Topology{
|
|
||||||
simplex:Simplex2_4,
|
|
||||||
}
|
|
||||||
impl Topology{
|
|
||||||
/// Returns None if the point is intersecting the mesh.
|
|
||||||
pub fn closest_point_details(self,mesh:&MinkowskiMesh)->Option<Details>{
|
|
||||||
// NOTE: if hits is true, this if statement necessarily evaluates to true.
|
|
||||||
// i.e. hits implies this statement
|
|
||||||
// if -dist <= exitRadius + radiusP + radiusQ then
|
|
||||||
// local posP, posQ = decompose(-point, a0, a1, b0, b1, c0, c1)
|
|
||||||
// return hits, -dist - radiusP - radiusQ,
|
|
||||||
// posP - radiusP*norm, -norm,
|
|
||||||
// posQ + radiusQ*norm, norm
|
|
||||||
// end
|
|
||||||
// return false
|
|
||||||
unimplemented!()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pub struct Details{
|
|
||||||
// distance:Planar64,
|
|
||||||
// p_pos:Planar64Vec3,
|
|
||||||
// p_norm:Planar64Vec3,
|
|
||||||
// q_pos:Planar64Vec3,
|
|
||||||
// q_norm:Planar64Vec3,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn contains_point(mesh:&MinkowskiMesh,point:Planar64Vec3)->bool{
|
|
||||||
const ENABLE_FAST_FAIL:bool=true;
|
const ENABLE_FAST_FAIL:bool=true;
|
||||||
minimum_difference::<ENABLE_FAST_FAIL,_>(mesh,point,
|
// TODO: remove mesh negation
|
||||||
|
minimum_difference::<ENABLE_FAST_FAIL,_,_>(&-mesh,point,
|
||||||
// on_exact
|
// on_exact
|
||||||
|last_pos,direction|{
|
|is_intersecting,_simplex|{
|
||||||
// local norm = direction.unit
|
is_intersecting
|
||||||
// local dist = a:Dot(norm)
|
|
||||||
// local hits = -dist < radiusP + radiusQ
|
|
||||||
// return hits
|
|
||||||
(last_pos+point).dot(direction).is_positive()
|
|
||||||
},
|
},
|
||||||
// on_escape
|
// on_escape
|
||||||
|_simplex|{
|
|_simplex|{
|
||||||
@@ -561,16 +552,248 @@ pub fn contains_point(mesh:&MinkowskiMesh,point:Planar64Vec3)->bool{
|
|||||||
||false
|
||false
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
pub fn closest_fev(mesh:&MinkowskiMesh,point:Planar64Vec3)->Topology{
|
|
||||||
|
//infinity fev algorithm state transition
|
||||||
|
#[derive(Debug)]
|
||||||
|
enum Transition<Vert>{
|
||||||
|
Done,//found closest vert, no edges are better
|
||||||
|
Vert(Vert),//transition to vert
|
||||||
|
}
|
||||||
|
enum EV<M:MeshQuery>{
|
||||||
|
Vert(M::Vert),
|
||||||
|
Edge(<M::Edge as DirectedEdge>::UndirectedEdge),
|
||||||
|
}
|
||||||
|
impl<M:MeshQuery> From<EV<M>> for FEV<M>{
|
||||||
|
fn from(value:EV<M>)->Self{
|
||||||
|
match value{
|
||||||
|
EV::Vert(minkowski_vert)=>FEV::Vert(minkowski_vert),
|
||||||
|
EV::Edge(minkowski_edge)=>FEV::Edge(minkowski_edge),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
trait Contains{
|
||||||
|
fn contains(&self,point:Planar64Vec3)->bool;
|
||||||
|
}
|
||||||
|
|
||||||
|
// convenience type to check if a point is within some threshold of a plane.
|
||||||
|
struct ThickPlane{
|
||||||
|
point:Planar64Vec3,
|
||||||
|
normal:Vector3<Fixed<2,64>>,
|
||||||
|
epsilon:Fixed<3,96>,
|
||||||
|
}
|
||||||
|
impl ThickPlane{
|
||||||
|
fn new<M:MeshQuery>(mesh:&M,[v0,v1,v2]:Simplex<3,M::Vert>)->Self{
|
||||||
|
let p0=mesh.vert(v0);
|
||||||
|
let p1=mesh.vert(v1);
|
||||||
|
let p2=mesh.vert(v2);
|
||||||
|
let point=p0;
|
||||||
|
let normal=(p1-p0).cross(p2-p0);
|
||||||
|
// Allow ~ 2*sqrt(3) units of thickness on the plane
|
||||||
|
// This is to account for the variance of two voxels across the longest diagonal
|
||||||
|
let epsilon=(normal.length()*(Planar64::EPSILON*3)).wrap_3();
|
||||||
|
Self{point,normal,epsilon}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
impl Contains for ThickPlane{
|
||||||
|
fn contains(&self,point:Planar64Vec3)->bool{
|
||||||
|
(point-self.point).dot(self.normal).abs()<=self.epsilon
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ThickLine{
|
||||||
|
point:Planar64Vec3,
|
||||||
|
dir:Planar64Vec3,
|
||||||
|
epsilon:Fixed<4,128>,
|
||||||
|
}
|
||||||
|
impl ThickLine{
|
||||||
|
fn new<M:MeshQuery>(mesh:&M,[v0,v1]:Simplex<2,M::Vert>)->Self{
|
||||||
|
let p0=mesh.vert(v0);
|
||||||
|
let p1=mesh.vert(v1);
|
||||||
|
let point=p0;
|
||||||
|
let dir=p1-p0;
|
||||||
|
// Allow ~ 2*sqrt(3) units of thickness on the plane
|
||||||
|
// This is to account for the variance of two voxels across the longest diagonal
|
||||||
|
let epsilon=(dir.length_squared()*(Planar64::EPSILON*3)).widen_4();
|
||||||
|
Self{point,dir,epsilon}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
impl Contains for ThickLine{
|
||||||
|
fn contains(&self,point:Planar64Vec3)->bool{
|
||||||
|
(point-self.point).cross(self.dir).length_squared()<=self.epsilon
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct EVFinder<'a,M,C>{
|
||||||
|
mesh:&'a M,
|
||||||
|
constraint:C,
|
||||||
|
best_distance_squared:Fixed<2,64>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<M:MeshQuery,C:Contains> EVFinder<'_,M,C>{
|
||||||
|
fn next_transition_vert(&mut self,vert_id:M::Vert,point:Planar64Vec3)->Transition<M::Vert>{
|
||||||
|
let mut best_transition=Transition::Done;
|
||||||
|
for &directed_edge_id in self.mesh.vert_edges(vert_id).as_ref(){
|
||||||
|
//test if this edge's opposite vertex closer
|
||||||
|
let edge_verts=self.mesh.edge_verts(directed_edge_id.as_undirected());
|
||||||
|
//select opposite vertex
|
||||||
|
let test_vert_id=edge_verts.as_ref()[directed_edge_id.parity() as usize];
|
||||||
|
let test_pos=self.mesh.vert(test_vert_id);
|
||||||
|
let diff=point-test_pos;
|
||||||
|
let distance_squared=diff.dot(diff);
|
||||||
|
// ensure test_vert_id is coplanar to simplex
|
||||||
|
if distance_squared<self.best_distance_squared&&self.constraint.contains(test_pos){
|
||||||
|
best_transition=Transition::Vert(test_vert_id);
|
||||||
|
self.best_distance_squared=distance_squared;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
best_transition
|
||||||
|
}
|
||||||
|
fn final_ev(&mut self,vert_id:M::Vert,point:Planar64Vec3)->EV<M>{
|
||||||
|
let mut best_transition=EV::Vert(vert_id);
|
||||||
|
let vert_pos=self.mesh.vert(vert_id);
|
||||||
|
let diff=point-vert_pos;
|
||||||
|
for &directed_edge_id in self.mesh.vert_edges(vert_id).as_ref(){
|
||||||
|
//test if this edge is closer
|
||||||
|
let edge_verts=self.mesh.edge_verts(directed_edge_id.as_undirected());
|
||||||
|
let test_vert_id=edge_verts.as_ref()[directed_edge_id.parity() as usize];
|
||||||
|
let test_pos=self.mesh.vert(test_vert_id);
|
||||||
|
let edge_n=test_pos-vert_pos;
|
||||||
|
let d=edge_n.dot(diff);
|
||||||
|
//test the edge
|
||||||
|
let edge_nn=edge_n.dot(edge_n);
|
||||||
|
// ensure edge contains closest point and directed_edge_id is coplanar to simplex
|
||||||
|
if !d.is_negative()&&d<=edge_nn&&self.constraint.contains(test_pos){
|
||||||
|
let distance_squared={
|
||||||
|
let c=diff.cross(edge_n);
|
||||||
|
//wrap for speed
|
||||||
|
(c.dot(c)/edge_nn).divide().wrap_2()
|
||||||
|
};
|
||||||
|
if distance_squared<=self.best_distance_squared{
|
||||||
|
best_transition=EV::Edge(directed_edge_id.as_undirected());
|
||||||
|
self.best_distance_squared=distance_squared;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
best_transition
|
||||||
|
}
|
||||||
|
fn crawl_boundaries(&mut self,mut vert_id:M::Vert,point:Planar64Vec3)->EV<M>{
|
||||||
|
loop{
|
||||||
|
match self.next_transition_vert(vert_id,point){
|
||||||
|
Transition::Done=>return self.final_ev(vert_id,point),
|
||||||
|
Transition::Vert(new_vert_id)=>vert_id=new_vert_id,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/// This function drops a vertex down to an edge or a face if the path from infinity did not cross any vertex-edge boundaries but the point is supposed to have already crossed a boundary down from a vertex
|
||||||
|
fn crawl_to_closest_ev<M:MeshQuery>(mesh:&M,simplex:Simplex<2,M::Vert>,point:Planar64Vec3)->EV<M>{
|
||||||
|
// naively start at the closest vertex
|
||||||
|
// the closest vertex is not necessarily the one with the fewest boundary hops
|
||||||
|
// but it doesn't matter, we will get there regardless.
|
||||||
|
let (vert_id,best_distance_squared)=simplex.into_iter().map(|vert_id|{
|
||||||
|
let diff=point-mesh.vert(vert_id);
|
||||||
|
(vert_id,diff.dot(diff))
|
||||||
|
}).min_by_key(|&(_,d)|d).unwrap();
|
||||||
|
|
||||||
|
let constraint=ThickLine::new(mesh,simplex);
|
||||||
|
let mut finder=EVFinder{constraint,mesh,best_distance_squared};
|
||||||
|
//start on any vertex
|
||||||
|
//cross uncrossable vertex-edge boundaries until you find the closest vertex or edge
|
||||||
|
//cross edge-face boundary if it's uncrossable
|
||||||
|
finder.crawl_boundaries(vert_id,point)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// This function drops a vertex down to an edge or a face if the path from infinity did not cross any vertex-edge boundaries but the point is supposed to have already crossed a boundary down from a vertex
|
||||||
|
fn crawl_to_closest_fev<'a>(mesh:&MinkowskiMesh<'a>,simplex:Simplex<3,MinkowskiVert>,point:Planar64Vec3)->FEV::<MinkowskiMesh<'a>>{
|
||||||
|
// naively start at the closest vertex
|
||||||
|
// the closest vertex is not necessarily the one with the fewest boundary hops
|
||||||
|
// but it doesn't matter, we will get there regardless.
|
||||||
|
let (vert_id,best_distance_squared)=simplex.into_iter().map(|vert_id|{
|
||||||
|
let diff=point-mesh.vert(vert_id);
|
||||||
|
(vert_id,diff.dot(diff))
|
||||||
|
}).min_by_key(|&(_,d)|d).unwrap();
|
||||||
|
|
||||||
|
let constraint=ThickPlane::new(mesh,simplex);
|
||||||
|
let mut finder=EVFinder{constraint,mesh,best_distance_squared};
|
||||||
|
//start on any vertex
|
||||||
|
//cross uncrossable vertex-edge boundaries until you find the closest vertex or edge
|
||||||
|
//cross edge-face boundary if it's uncrossable
|
||||||
|
match finder.crawl_boundaries(vert_id,point){
|
||||||
|
//if a vert is returned, it is the closest point to the infinity point
|
||||||
|
EV::Vert(vert_id)=>FEV::Vert(vert_id),
|
||||||
|
EV::Edge(edge_id)=>{
|
||||||
|
//cross to face if we are on the wrong side
|
||||||
|
let edge_n=mesh.edge_n(edge_id);
|
||||||
|
// point is multiplied by two because vert_sum sums two vertices.
|
||||||
|
let delta_pos=point*2-{
|
||||||
|
let &[v0,v1]=mesh.edge_verts(edge_id).as_ref();
|
||||||
|
mesh.vert(v0)+mesh.vert(v1)
|
||||||
|
};
|
||||||
|
for (i,&face_id) in mesh.edge_faces(edge_id).as_ref().iter().enumerate(){
|
||||||
|
//test if this face is closer
|
||||||
|
let (face_n,d)=mesh.face_nd(face_id);
|
||||||
|
//if test point is behind face, the face is invalid
|
||||||
|
// TODO: find out why I thought of this backwards
|
||||||
|
if !(face_n.dot(point)-d).is_positive(){
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
//edge-face boundary nd, n facing out of the face towards the edge
|
||||||
|
let boundary_n=face_n.cross(edge_n)*(i as i64*2-1);
|
||||||
|
let boundary_d=boundary_n.dot(delta_pos);
|
||||||
|
//is test point behind edge, i.e. contained in the face
|
||||||
|
if !boundary_d.is_positive(){
|
||||||
|
//both faces cannot pass this condition, return early if one does.
|
||||||
|
return FEV::Face(face_id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
FEV::Edge(edge_id)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn closest_fev_not_inside<'a>(mesh:&MinkowskiMesh<'a>,point:Planar64Vec3)->Option<FEV<MinkowskiMesh<'a>>>{
|
||||||
const ENABLE_FAST_FAIL:bool=false;
|
const ENABLE_FAST_FAIL:bool=false;
|
||||||
minimum_difference::<ENABLE_FAST_FAIL,_>(mesh,point,
|
// TODO: remove mesh negation
|
||||||
|
minimum_difference::<ENABLE_FAST_FAIL,_,_>(&-mesh,point,
|
||||||
// on_exact
|
// on_exact
|
||||||
|_last_pos,_direction|unimplemented!(),
|
|is_intersecting,simplex|{
|
||||||
|
if is_intersecting{
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
// Convert simplex to FEV
|
||||||
|
// Vertices must be inverted since the mesh is inverted
|
||||||
|
Some(match simplex{
|
||||||
|
Simplex1_3::Simplex1([v0])=>FEV::Vert(-v0),
|
||||||
|
Simplex1_3::Simplex2([v0,v1])=>{
|
||||||
|
// invert
|
||||||
|
let (v0,v1)=(-v0,-v1);
|
||||||
|
let ev=crawl_to_closest_ev(mesh,[v0,v1],point);
|
||||||
|
if !matches!(ev,EV::Edge(_)){
|
||||||
|
println!("I can't believe it's not an edge!");
|
||||||
|
}
|
||||||
|
ev.into()
|
||||||
|
},
|
||||||
|
Simplex1_3::Simplex3([v0,v1,v2])=>{
|
||||||
|
// invert
|
||||||
|
let (v0,v1,v2)=(-v0,-v1,-v2);
|
||||||
|
// Shimmy to the side until you find a face that contains the closest point
|
||||||
|
// it's ALWAYS representable as a face, but this algorithm may
|
||||||
|
// return E or V in edge cases but I don't think that will break the face crawler
|
||||||
|
let fev=crawl_to_closest_fev(mesh,[v0,v1,v2],point);
|
||||||
|
if !matches!(fev,FEV::Face(_)){
|
||||||
|
println!("I can't believe it's not a face!");
|
||||||
|
}
|
||||||
|
fev
|
||||||
|
},
|
||||||
|
})
|
||||||
|
},
|
||||||
// on_escape
|
// on_escape
|
||||||
|simplex|{
|
|_simplex|{
|
||||||
|
// intersection is guaranteed at this point
|
||||||
// local norm, dist, u0, u1, v0, v1, w0, w1 = expand(queryP, queryQ, a0, a1, b0, b1, c0, c1, d0, d1, 1e-5)
|
// local norm, dist, u0, u1, v0, v1, w0, w1 = expand(queryP, queryQ, a0, a1, b0, b1, c0, c1, d0, d1, 1e-5)
|
||||||
let simplex=refine_to_exact(mesh,simplex);
|
// let simplex=refine_to_exact(mesh,simplex);
|
||||||
Topology{simplex}
|
None
|
||||||
},
|
},
|
||||||
// fast_fail value is irrelevant and will never be returned!
|
// fast_fail value is irrelevant and will never be returned!
|
||||||
||unreachable!()
|
||unreachable!()
|
||||||
@@ -582,25 +805,25 @@ pub fn closest_fev(mesh:&MinkowskiMesh,point:Planar64Vec3)->Topology{
|
|||||||
// queryQ, radiusQ,
|
// queryQ, radiusQ,
|
||||||
// exitRadius, testIntersection
|
// exitRadius, testIntersection
|
||||||
// )
|
// )
|
||||||
fn minimum_difference<const ENABLE_FAST_FAIL:bool,T>(
|
fn minimum_difference<const ENABLE_FAST_FAIL:bool,T,M:MeshQuery>(
|
||||||
mesh:&MinkowskiMesh,
|
mesh:&M,
|
||||||
point:Planar64Vec3,
|
point:Planar64Vec3,
|
||||||
on_exact:impl FnOnce(Planar64Vec3,Planar64Vec3)->T,
|
on_exact:impl FnOnce(bool,Simplex1_3<M::Vert>)->T,
|
||||||
on_escape:impl FnOnce(Simplex<4>)->T,
|
on_escape:impl FnOnce(Simplex<4,M::Vert>)->T,
|
||||||
on_fast_fail:impl FnOnce()->T,
|
on_fast_fail:impl FnOnce()->T,
|
||||||
)->T{
|
)->T{
|
||||||
// local initialAxis = queryQ() - queryP()
|
// local initialAxis = queryQ() - queryP()
|
||||||
// local new_point_p = queryP(initialAxis)
|
// local new_point_p = queryP(initialAxis)
|
||||||
// local new_point_q = queryQ(-initialAxis)
|
// local new_point_q = queryQ(-initialAxis)
|
||||||
// local direction, a0, a1, b0, b1, c0, c1, d0, d1
|
// local direction, a0, a1, b0, b1, c0, c1, d0, d1
|
||||||
let mut initial_axis=-mesh.hint_point()+point;
|
let mut initial_axis=mesh.hint_point()+point;
|
||||||
// degenerate case
|
// degenerate case
|
||||||
if initial_axis==vec3::zero(){
|
if initial_axis==vec3::zero(){
|
||||||
initial_axis=choose_any_direction();
|
initial_axis=choose_any_direction();
|
||||||
}
|
}
|
||||||
let last_point=mesh.farthest_vert(direction);
|
let last_point=mesh.farthest_vert(-initial_axis);
|
||||||
// this represents the 'a' value in the commented code
|
// this represents the 'a' value in the commented code
|
||||||
let mut last_pos=-mesh.vert(last_point);
|
let mut last_pos=mesh.vert(last_point);
|
||||||
let Reduced{dir:mut direction,simplex:mut simplex_small}=reduce1([last_point],mesh,point);
|
let Reduced{dir:mut direction,simplex:mut simplex_small}=reduce1([last_point],mesh,point);
|
||||||
|
|
||||||
// exitRadius = testIntersection and 0 or exitRadius or 1/0
|
// exitRadius = testIntersection and 0 or exitRadius or 1/0
|
||||||
@@ -609,8 +832,8 @@ fn minimum_difference<const ENABLE_FAST_FAIL:bool,T>(
|
|||||||
// new_point_p = queryP(-direction)
|
// new_point_p = queryP(-direction)
|
||||||
// new_point_q = queryQ(direction)
|
// new_point_q = queryQ(direction)
|
||||||
// local next_point = new_point_q - new_point_p
|
// local next_point = new_point_q - new_point_p
|
||||||
let next_point=mesh.farthest_vert(-direction);
|
let next_point=mesh.farthest_vert(direction);
|
||||||
let next_pos=-mesh.vert(next_point);
|
let next_pos=mesh.vert(next_point);
|
||||||
|
|
||||||
// if -direction:Dot(next_point) > (exitRadius + radiusP + radiusQ)*direction.magnitude then
|
// if -direction:Dot(next_point) > (exitRadius + radiusP + radiusQ)*direction.magnitude then
|
||||||
if ENABLE_FAST_FAIL&&direction.dot(next_pos+point).is_negative(){
|
if ENABLE_FAST_FAIL&&direction.dot(next_pos+point).is_negative(){
|
||||||
@@ -625,7 +848,11 @@ fn minimum_difference<const ENABLE_FAST_FAIL:bool,T>(
|
|||||||
if !direction.dot(next_pos-last_pos).is_positive()
|
if !direction.dot(next_pos-last_pos).is_positive()
|
||||||
||simplex_big.det_is_zero(mesh){
|
||simplex_big.det_is_zero(mesh){
|
||||||
// Found enough information to compute the exact closest point.
|
// Found enough information to compute the exact closest point.
|
||||||
return on_exact(last_pos,direction);
|
// local norm = direction.unit
|
||||||
|
// local dist = a:Dot(norm)
|
||||||
|
// local hits = -dist < radiusP + radiusQ
|
||||||
|
let is_intersecting=(last_pos+point).dot(direction).is_positive();
|
||||||
|
return on_exact(is_intersecting,simplex_small);
|
||||||
}
|
}
|
||||||
|
|
||||||
// direction, a0, a1, b0, b1, c0, c1, d0, d1 = reduceSimplex(new_point_p, new_point_q, a0, a1, b0, b1, c0, c1)
|
// direction, a0, a1, b0, b1, c0, c1, d0, d1 = reduceSimplex(new_point_p, new_point_q, a0, a1, b0, b1, c0, c1)
|
||||||
@@ -646,3 +873,41 @@ fn minimum_difference<const ENABLE_FAST_FAIL:bool,T>(
|
|||||||
last_pos=next_pos;
|
last_pos=next_pos;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test{
|
||||||
|
use super::*;
|
||||||
|
use crate::model::{PhysicsMesh,PhysicsMeshView};
|
||||||
|
|
||||||
|
fn mesh_contains_point(mesh:PhysicsMeshView<'_>,point:Planar64Vec3)->bool{
|
||||||
|
const ENABLE_FAST_FAIL:bool=true;
|
||||||
|
// TODO: remove mesh negation
|
||||||
|
minimum_difference::<ENABLE_FAST_FAIL,_,_>(&mesh,point,
|
||||||
|
// on_exact
|
||||||
|
|is_intersecting,_simplex|{
|
||||||
|
is_intersecting
|
||||||
|
},
|
||||||
|
// on_escape
|
||||||
|
|_simplex|{
|
||||||
|
// intersection is guaranteed at this point
|
||||||
|
true
|
||||||
|
},
|
||||||
|
// fast_fail value
|
||||||
|
||false
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_cube_points(){
|
||||||
|
let mesh=PhysicsMesh::unit_cube();
|
||||||
|
let mesh_view=mesh.complete_mesh_view();
|
||||||
|
for x in -2..=2{
|
||||||
|
for y in -2..=2{
|
||||||
|
for z in -2..=2{
|
||||||
|
let point=vec3::int(x,y,z)>>1;
|
||||||
|
assert!(mesh_contains_point(mesh_view,point),"Mesh did not contain point {point}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,159 +0,0 @@
|
|||||||
use mlua::{Lua,FromLuaMulti,IntoLuaMulti,Function,Result as LuaResult,Vector};
|
|
||||||
use strafesnet_common::integer::{Planar64,Planar64Vec3,FixedFromFloatError};
|
|
||||||
|
|
||||||
use crate::model::{MeshQuery,MinkowskiMesh};
|
|
||||||
|
|
||||||
pub fn contains_point(
|
|
||||||
mesh:&MinkowskiMesh,
|
|
||||||
point:Planar64Vec3,
|
|
||||||
)->LuaResult<bool>{
|
|
||||||
Ok(minimum_difference(mesh,point,true)?.hits)
|
|
||||||
}
|
|
||||||
pub fn minimum_difference_details(
|
|
||||||
mesh:&MinkowskiMesh,
|
|
||||||
point:Planar64Vec3,
|
|
||||||
)->LuaResult<(bool,Details)>{
|
|
||||||
let md=minimum_difference(mesh,point,false)?;
|
|
||||||
Ok((md.hits,md.details.unwrap()))
|
|
||||||
}
|
|
||||||
fn p64v3(v:Vector)->Result<Planar64Vec3,FixedFromFloatError>{
|
|
||||||
Ok(Planar64Vec3::new([
|
|
||||||
v.x().try_into()?,
|
|
||||||
v.y().try_into()?,
|
|
||||||
v.z().try_into()?,
|
|
||||||
]))
|
|
||||||
}
|
|
||||||
fn vec(v:Planar64Vec3)->Vector{
|
|
||||||
Vector::new(v.x.into(),v.y.into(),v.z.into())
|
|
||||||
}
|
|
||||||
struct MinimumDifference{
|
|
||||||
hits:bool,
|
|
||||||
details:Option<Details>
|
|
||||||
}
|
|
||||||
pub struct Details{
|
|
||||||
pub distance:Planar64,
|
|
||||||
pub p_pos:Planar64Vec3,
|
|
||||||
pub p_norm:Planar64Vec3,
|
|
||||||
pub q_pos:Planar64Vec3,
|
|
||||||
pub q_norm:Planar64Vec3,
|
|
||||||
}
|
|
||||||
impl FromLuaMulti for MinimumDifference{
|
|
||||||
fn from_lua_multi(mut values:mlua::MultiValue,_lua:&Lua)->LuaResult<Self>{
|
|
||||||
match values.make_contiguous(){
|
|
||||||
&mut [
|
|
||||||
mlua::Value::Boolean(hits),
|
|
||||||
mlua::Value::Nil,
|
|
||||||
mlua::Value::Nil,
|
|
||||||
mlua::Value::Nil,
|
|
||||||
mlua::Value::Nil,
|
|
||||||
mlua::Value::Nil,
|
|
||||||
]=>Ok(Self{hits,details:None}),
|
|
||||||
&mut [
|
|
||||||
mlua::Value::Boolean(hits),
|
|
||||||
mlua::Value::Number(distance),
|
|
||||||
mlua::Value::Vector(p_pos),
|
|
||||||
mlua::Value::Vector(p_norm),
|
|
||||||
mlua::Value::Vector(q_pos),
|
|
||||||
mlua::Value::Vector(q_norm),
|
|
||||||
]=>Ok(Self{
|
|
||||||
hits,
|
|
||||||
details:Some(Details{
|
|
||||||
distance:distance.try_into().unwrap(),
|
|
||||||
p_pos:p64v3(p_pos).unwrap(),
|
|
||||||
p_norm:p64v3(p_norm).unwrap(),
|
|
||||||
q_pos:p64v3(q_pos).unwrap(),
|
|
||||||
q_norm:p64v3(q_norm).unwrap(),
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
values=>Err(mlua::Error::runtime(format!("Invalid return values: {values:?}"))),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct Args{
|
|
||||||
query_p:Function,
|
|
||||||
radius_p:f64,
|
|
||||||
query_q:Function,
|
|
||||||
radius_q:f64,
|
|
||||||
exit_radius:f64,
|
|
||||||
test_intersection:bool,
|
|
||||||
}
|
|
||||||
impl Args{
|
|
||||||
fn new(
|
|
||||||
lua:&Lua,
|
|
||||||
mesh:&'static MinkowskiMesh<'static>,
|
|
||||||
point:Planar64Vec3,
|
|
||||||
test_intersection:bool,
|
|
||||||
)->LuaResult<Self>{
|
|
||||||
let radius_p=0.0;
|
|
||||||
let radius_q=0.0;
|
|
||||||
let exit_radius=0.0;
|
|
||||||
// Query the farthest point on the mesh in the given direction.
|
|
||||||
let query_p=lua.create_function(move|_,dir:Option<Vector>|{
|
|
||||||
let Some(dir)=dir else{
|
|
||||||
return Ok(vec(mesh.mesh0.hint_point()));
|
|
||||||
};
|
|
||||||
let dir=p64v3(dir).unwrap();
|
|
||||||
let vert_id=mesh.mesh0.farthest_vert(dir);
|
|
||||||
let dir=mesh.mesh0.vert(vert_id);
|
|
||||||
Ok(vec(dir))
|
|
||||||
})?;
|
|
||||||
// query_q is different since it includes the test point offset.
|
|
||||||
let query_q=lua.create_function(move|_,dir:Option<Vector>|{
|
|
||||||
let Some(dir)=dir else{
|
|
||||||
return Ok(vec(mesh.mesh1.hint_point()+point));
|
|
||||||
};
|
|
||||||
let dir=p64v3(dir).unwrap();
|
|
||||||
let vert_id=mesh.mesh1.farthest_vert(dir);
|
|
||||||
let dir=mesh.mesh1.vert(vert_id)+point;
|
|
||||||
Ok(vec(dir))
|
|
||||||
})?;
|
|
||||||
Ok(Args{
|
|
||||||
query_p,
|
|
||||||
radius_p,
|
|
||||||
query_q,
|
|
||||||
radius_q,
|
|
||||||
exit_radius,
|
|
||||||
test_intersection,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
impl IntoLuaMulti for Args{
|
|
||||||
fn into_lua_multi(self,lua:&Lua)->LuaResult<mlua::MultiValue>{
|
|
||||||
use mlua::IntoLua;
|
|
||||||
Ok(mlua::MultiValue::from_vec(vec![
|
|
||||||
self.query_p.into_lua(lua)?,
|
|
||||||
self.radius_p.into_lua(lua)?,
|
|
||||||
self.query_q.into_lua(lua)?,
|
|
||||||
self.radius_q.into_lua(lua)?,
|
|
||||||
self.exit_radius.into_lua(lua)?,
|
|
||||||
self.test_intersection.into_lua(lua)?,
|
|
||||||
]))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn minimum_difference(
|
|
||||||
mesh:&MinkowskiMesh,
|
|
||||||
point:Planar64Vec3,
|
|
||||||
test_intersection:bool,
|
|
||||||
)->LuaResult<MinimumDifference>{
|
|
||||||
let ctx=init_lua()?;
|
|
||||||
// SAFETY: mesh lifetime must outlive args usages
|
|
||||||
let mesh=unsafe{core::mem::transmute(mesh)};
|
|
||||||
let args=Args::new(&ctx.lua,mesh,point,test_intersection)?;
|
|
||||||
ctx.f.call(args)
|
|
||||||
}
|
|
||||||
|
|
||||||
struct Ctx{
|
|
||||||
lua:Lua,
|
|
||||||
f:Function,
|
|
||||||
}
|
|
||||||
fn init_lua()->LuaResult<Ctx>{
|
|
||||||
static SOURCE:std::sync::LazyLock<String>=std::sync::LazyLock::new(||std::fs::read_to_string("/home/quat/strafesnet/game/src/ReplicatedStorage/Shared/Trey-MinimumDifference.lua").unwrap());
|
|
||||||
let lua=Lua::new();
|
|
||||||
lua.sandbox(true)?;
|
|
||||||
let lib_f=lua.load(SOURCE.as_str()).set_name("Trey-MinimumDifference").into_function()?;
|
|
||||||
let lib:mlua::Table=lib_f.call(())?;
|
|
||||||
let f=lib.raw_get("difference")?;
|
|
||||||
Ok(Ctx{lua,f})
|
|
||||||
}
|
|
||||||
@@ -92,6 +92,7 @@ pub trait MeshQuery{
|
|||||||
}
|
}
|
||||||
/// This must return a point inside the mesh.
|
/// This must return a point inside the mesh.
|
||||||
fn hint_point(&self)->Planar64Vec3;
|
fn hint_point(&self)->Planar64Vec3;
|
||||||
|
fn farthest_vert(&self,dir:Planar64Vec3)->Self::Vert;
|
||||||
fn vert(&self,vert_id:Self::Vert)->Planar64Vec3;
|
fn vert(&self,vert_id:Self::Vert)->Planar64Vec3;
|
||||||
fn face_nd(&self,face_id:Self::Face)->(Self::Normal,Self::Offset);
|
fn face_nd(&self,face_id:Self::Face)->(Self::Normal,Self::Offset);
|
||||||
fn face_edges(&self,face_id:Self::Face)->impl AsRef<[Self::Edge]>;
|
fn face_edges(&self,face_id:Self::Face)->impl AsRef<[Self::Edge]>;
|
||||||
@@ -102,17 +103,21 @@ pub trait MeshQuery{
|
|||||||
}
|
}
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct FaceRefs{
|
struct FaceRefs{
|
||||||
|
// I didn't write it down, but I assume the edges are directed
|
||||||
|
// clockwise when looking towards the face normal, i.e. right hand rule.
|
||||||
edges:Vec<SubmeshDirectedEdgeId>,
|
edges:Vec<SubmeshDirectedEdgeId>,
|
||||||
|
//verts are redundant, use edge[i].verts[0]
|
||||||
//verts:Vec<VertId>,
|
//verts:Vec<VertId>,
|
||||||
}
|
}
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct EdgeRefs{
|
struct EdgeRefs{
|
||||||
faces:[SubmeshFaceId;2],//left, right
|
faces:[SubmeshFaceId;2],//left, right
|
||||||
verts:[SubmeshVertId;2],//bottom, top
|
verts:[SubmeshVertId;2],//start, end
|
||||||
}
|
}
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct VertRefs{
|
struct VertRefs{
|
||||||
faces:Vec<SubmeshFaceId>,
|
faces:Vec<SubmeshFaceId>,
|
||||||
|
// edges are always directed away from the vert
|
||||||
edges:Vec<SubmeshDirectedEdgeId>,
|
edges:Vec<SubmeshDirectedEdgeId>,
|
||||||
}
|
}
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
@@ -434,7 +439,7 @@ impl TryFrom<&model::Mesh> for PhysicsMesh{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug,Clone,Copy)]
|
||||||
pub struct PhysicsMeshView<'a>{
|
pub struct PhysicsMeshView<'a>{
|
||||||
data:&'a PhysicsMeshData,
|
data:&'a PhysicsMeshData,
|
||||||
topology:&'a PhysicsMeshTopology,
|
topology:&'a PhysicsMeshTopology,
|
||||||
@@ -453,6 +458,18 @@ impl MeshQuery for PhysicsMeshView<'_>{
|
|||||||
// invariant: meshes always encompass the origin
|
// invariant: meshes always encompass the origin
|
||||||
vec3::zero()
|
vec3::zero()
|
||||||
}
|
}
|
||||||
|
fn farthest_vert(&self,dir:Planar64Vec3)->SubmeshVertId{
|
||||||
|
//this happens to be well-defined. there are no virtual virtices
|
||||||
|
SubmeshVertId::new(
|
||||||
|
self.topology.verts.iter()
|
||||||
|
.enumerate()
|
||||||
|
.max_by_key(|&(_,&vert_id)|
|
||||||
|
dir.dot(self.data.verts[vert_id.get() as usize].0)
|
||||||
|
)
|
||||||
|
//assume there is more than zero vertices.
|
||||||
|
.unwrap().0 as u32
|
||||||
|
)
|
||||||
|
}
|
||||||
//ideally I never calculate the vertex position, but I have to for the graphical meshes...
|
//ideally I never calculate the vertex position, but I have to for the graphical meshes...
|
||||||
fn vert(&self,vert_id:SubmeshVertId)->Planar64Vec3{
|
fn vert(&self,vert_id:SubmeshVertId)->Planar64Vec3{
|
||||||
let vert_idx=self.topology.verts[vert_id.get() as usize].get() as usize;
|
let vert_idx=self.topology.verts[vert_id.get() as usize].get() as usize;
|
||||||
@@ -491,7 +508,7 @@ impl PhysicsMeshTransform{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug,Clone,Copy)]
|
||||||
pub struct TransformedMesh<'a>{
|
pub struct TransformedMesh<'a>{
|
||||||
view:PhysicsMeshView<'a>,
|
view:PhysicsMeshView<'a>,
|
||||||
transform:&'a PhysicsMeshTransform,
|
transform:&'a PhysicsMeshTransform,
|
||||||
@@ -509,18 +526,6 @@ impl TransformedMesh<'_>{
|
|||||||
pub fn verts<'a>(&'a self)->impl Iterator<Item=Vector3<Fixed<2,64>>>+'a{
|
pub fn verts<'a>(&'a self)->impl Iterator<Item=Vector3<Fixed<2,64>>>+'a{
|
||||||
self.view.data.verts.iter().map(|&Vert(pos)|self.transform.vertex.transform_point3(pos))
|
self.view.data.verts.iter().map(|&Vert(pos)|self.transform.vertex.transform_point3(pos))
|
||||||
}
|
}
|
||||||
pub fn farthest_vert(&self,dir:Planar64Vec3)->SubmeshVertId{
|
|
||||||
//this happens to be well-defined. there are no virtual virtices
|
|
||||||
SubmeshVertId::new(
|
|
||||||
self.view.topology.verts.iter()
|
|
||||||
.enumerate()
|
|
||||||
.max_by_key(|&(_,&vert_id)|
|
|
||||||
dir.dot(self.transform.vertex.transform_point3(self.view.data.verts[vert_id.get() as usize].0))
|
|
||||||
)
|
|
||||||
//assume there is more than zero vertices.
|
|
||||||
.unwrap().0 as u32
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
impl MeshQuery for TransformedMesh<'_>{
|
impl MeshQuery for TransformedMesh<'_>{
|
||||||
type Face=SubmeshFaceId;
|
type Face=SubmeshFaceId;
|
||||||
@@ -541,6 +546,18 @@ impl MeshQuery for TransformedMesh<'_>{
|
|||||||
fn hint_point(&self)->Planar64Vec3{
|
fn hint_point(&self)->Planar64Vec3{
|
||||||
self.transform.vertex.translation
|
self.transform.vertex.translation
|
||||||
}
|
}
|
||||||
|
fn farthest_vert(&self,dir:Planar64Vec3)->SubmeshVertId{
|
||||||
|
//this happens to be well-defined. there are no virtual virtices
|
||||||
|
SubmeshVertId::new(
|
||||||
|
self.view.topology.verts.iter()
|
||||||
|
.enumerate()
|
||||||
|
.max_by_key(|&(_,&vert_id)|
|
||||||
|
dir.dot(self.transform.vertex.transform_point3(self.view.data.verts[vert_id.get() as usize].0))
|
||||||
|
)
|
||||||
|
//assume there is more than zero vertices.
|
||||||
|
.unwrap().0 as u32
|
||||||
|
)
|
||||||
|
}
|
||||||
#[inline]
|
#[inline]
|
||||||
fn face_edges(&self,face_id:SubmeshFaceId)->impl AsRef<[SubmeshDirectedEdgeId]>{
|
fn face_edges(&self,face_id:SubmeshFaceId)->impl AsRef<[SubmeshDirectedEdgeId]>{
|
||||||
self.view.face_edges(face_id)
|
self.view.face_edges(face_id)
|
||||||
@@ -571,7 +588,16 @@ impl MeshQuery for TransformedMesh<'_>{
|
|||||||
pub enum MinkowskiVert{
|
pub enum MinkowskiVert{
|
||||||
VertVert(SubmeshVertId,SubmeshVertId),
|
VertVert(SubmeshVertId,SubmeshVertId),
|
||||||
}
|
}
|
||||||
#[derive(Clone,Copy,Debug)]
|
// TODO: remove this
|
||||||
|
impl core::ops::Neg for MinkowskiVert{
|
||||||
|
type Output=Self;
|
||||||
|
fn neg(self)->Self::Output{
|
||||||
|
match self{
|
||||||
|
MinkowskiVert::VertVert(v0,v1)=>MinkowskiVert::VertVert(v1,v0),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#[derive(Clone,Copy,Debug,Eq,PartialEq)]
|
||||||
pub enum MinkowskiEdge{
|
pub enum MinkowskiEdge{
|
||||||
VertEdge(SubmeshVertId,SubmeshEdgeId),
|
VertEdge(SubmeshVertId,SubmeshEdgeId),
|
||||||
EdgeVert(SubmeshEdgeId,SubmeshVertId),
|
EdgeVert(SubmeshEdgeId,SubmeshVertId),
|
||||||
@@ -607,7 +633,7 @@ impl DirectedEdge for MinkowskiDirectedEdge{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#[derive(Clone,Copy,Debug,Hash,Eq,PartialEq)]
|
#[derive(Clone,Copy,Debug,Hash)]
|
||||||
pub enum MinkowskiFace{
|
pub enum MinkowskiFace{
|
||||||
VertFace(SubmeshVertId,SubmeshFaceId),
|
VertFace(SubmeshVertId,SubmeshFaceId),
|
||||||
EdgeEdge(SubmeshEdgeId,SubmeshEdgeId,bool),
|
EdgeEdge(SubmeshEdgeId,SubmeshEdgeId,bool),
|
||||||
@@ -619,19 +645,8 @@ pub enum MinkowskiFace{
|
|||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct MinkowskiMesh<'a>{
|
pub struct MinkowskiMesh<'a>{
|
||||||
pub mesh0:TransformedMesh<'a>,
|
mesh0:TransformedMesh<'a>,
|
||||||
pub mesh1:TransformedMesh<'a>,
|
mesh1:TransformedMesh<'a>,
|
||||||
}
|
|
||||||
|
|
||||||
//infinity fev algorithm state transition
|
|
||||||
#[derive(Debug)]
|
|
||||||
enum Transition{
|
|
||||||
Done,//found closest vert, no edges are better
|
|
||||||
Vert(MinkowskiVert),//transition to vert
|
|
||||||
}
|
|
||||||
enum EV{
|
|
||||||
Vert(MinkowskiVert),
|
|
||||||
Edge(MinkowskiEdge),
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub type GigaTime=Ratio<Fixed<4,128>,Fixed<4,128>>;
|
pub type GigaTime=Ratio<Fixed<4,128>,Fixed<4,128>>;
|
||||||
@@ -640,6 +655,14 @@ pub fn into_giga_time(time:Time,relative_to:Time)->GigaTime{
|
|||||||
Ratio::new(r.num.widen_4(),r.den.widen_4())
|
Ratio::new(r.num.widen_4(),r.den.widen_4())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: remove this
|
||||||
|
impl<'a> core::ops::Neg for &MinkowskiMesh<'a>{
|
||||||
|
type Output=MinkowskiMesh<'a>;
|
||||||
|
fn neg(self)->Self::Output{
|
||||||
|
MinkowskiMesh::minkowski_sum(self.mesh1,self.mesh0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl MinkowskiMesh<'_>{
|
impl MinkowskiMesh<'_>{
|
||||||
pub fn minkowski_sum<'a>(mesh0:TransformedMesh<'a>,mesh1:TransformedMesh<'a>)->MinkowskiMesh<'a>{
|
pub fn minkowski_sum<'a>(mesh0:TransformedMesh<'a>,mesh1:TransformedMesh<'a>)->MinkowskiMesh<'a>{
|
||||||
MinkowskiMesh{
|
MinkowskiMesh{
|
||||||
@@ -647,140 +670,27 @@ impl MinkowskiMesh<'_>{
|
|||||||
mesh1,
|
mesh1,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pub fn farthest_vert(&self,dir:Planar64Vec3)->MinkowskiVert{
|
|
||||||
MinkowskiVert::VertVert(self.mesh0.farthest_vert(dir),self.mesh1.farthest_vert(-dir))
|
|
||||||
}
|
|
||||||
fn next_transition_vert(&self,vert_id:MinkowskiVert,best_distance_squared:&mut Fixed<2,64>,infinity_dir:Planar64Vec3,point:Planar64Vec3)->Transition{
|
|
||||||
let mut best_transition=Transition::Done;
|
|
||||||
for &directed_edge_id in self.vert_edges(vert_id).as_ref(){
|
|
||||||
let edge_n=self.directed_edge_n(directed_edge_id);
|
|
||||||
//is boundary uncrossable by a crawl from infinity
|
|
||||||
let edge_verts=self.edge_verts(directed_edge_id.as_undirected());
|
|
||||||
//select opposite vertex
|
|
||||||
let test_vert_id=edge_verts.as_ref()[directed_edge_id.parity() as usize];
|
|
||||||
//test if it's closer
|
|
||||||
let diff=point-self.vert(test_vert_id);
|
|
||||||
if edge_n.dot(infinity_dir).is_zero(){
|
|
||||||
let distance_squared=diff.dot(diff);
|
|
||||||
if distance_squared<*best_distance_squared{
|
|
||||||
best_transition=Transition::Vert(test_vert_id);
|
|
||||||
*best_distance_squared=distance_squared;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
best_transition
|
|
||||||
}
|
|
||||||
fn final_ev(&self,vert_id:MinkowskiVert,best_distance_squared:&mut Fixed<2,64>,infinity_dir:Planar64Vec3,point:Planar64Vec3)->EV{
|
|
||||||
let mut best_transition=EV::Vert(vert_id);
|
|
||||||
let diff=point-self.vert(vert_id);
|
|
||||||
for &directed_edge_id in self.vert_edges(vert_id).as_ref(){
|
|
||||||
let edge_n=self.directed_edge_n(directed_edge_id);
|
|
||||||
//is boundary uncrossable by a crawl from infinity
|
|
||||||
//check if time of collision is outside Time::MIN..Time::MAX
|
|
||||||
if edge_n.dot(infinity_dir).is_zero(){
|
|
||||||
let d=edge_n.dot(diff);
|
|
||||||
//test the edge
|
|
||||||
let edge_nn=edge_n.dot(edge_n);
|
|
||||||
if !d.is_negative()&&d<=edge_nn{
|
|
||||||
let distance_squared={
|
|
||||||
let c=diff.cross(edge_n);
|
|
||||||
//wrap for speed
|
|
||||||
(c.dot(c)/edge_nn).divide().wrap_2()
|
|
||||||
};
|
|
||||||
if distance_squared<=*best_distance_squared{
|
|
||||||
best_transition=EV::Edge(directed_edge_id.as_undirected());
|
|
||||||
*best_distance_squared=distance_squared;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
best_transition
|
|
||||||
}
|
|
||||||
fn crawl_boundaries(&self,mut vert_id:MinkowskiVert,infinity_dir:Planar64Vec3,point:Planar64Vec3)->EV{
|
|
||||||
let mut best_distance_squared={
|
|
||||||
let diff=point-self.vert(vert_id);
|
|
||||||
diff.dot(diff)
|
|
||||||
};
|
|
||||||
loop{
|
|
||||||
match self.next_transition_vert(vert_id,&mut best_distance_squared,infinity_dir,point){
|
|
||||||
Transition::Done=>return self.final_ev(vert_id,&mut best_distance_squared,infinity_dir,point),
|
|
||||||
Transition::Vert(new_vert_id)=>vert_id=new_vert_id,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
/// This function drops a vertex down to an edge or a face if the path from infinity did not cross any vertex-edge boundaries but the point is supposed to have already crossed a boundary down from a vertex
|
|
||||||
fn infinity_fev(&self,infinity_dir:Planar64Vec3,point:Planar64Vec3)->FEV::<MinkowskiMesh<'_>>{
|
|
||||||
//start on any vertex
|
|
||||||
//cross uncrossable vertex-edge boundaries until you find the closest vertex or edge
|
|
||||||
//cross edge-face boundary if it's uncrossable
|
|
||||||
match self.crawl_boundaries(self.farthest_vert(infinity_dir),infinity_dir,point){
|
|
||||||
//if a vert is returned, it is the closest point to the infinity point
|
|
||||||
EV::Vert(vert_id)=>FEV::Vert(vert_id),
|
|
||||||
EV::Edge(edge_id)=>{
|
|
||||||
//cross to face if the boundary is not crossable and we are on the wrong side
|
|
||||||
let edge_n=self.edge_n(edge_id);
|
|
||||||
// point is multiplied by two because vert_sum sums two vertices.
|
|
||||||
let delta_pos=point*2-{
|
|
||||||
let &[v0,v1]=self.edge_verts(edge_id).as_ref();
|
|
||||||
self.vert(v0)+self.vert(v1)
|
|
||||||
};
|
|
||||||
for (i,&face_id) in self.edge_faces(edge_id).as_ref().iter().enumerate(){
|
|
||||||
let face_n=self.face_nd(face_id).0;
|
|
||||||
//edge-face boundary nd, n facing out of the face towards the edge
|
|
||||||
let boundary_n=face_n.cross(edge_n)*(i as i64*2-1);
|
|
||||||
let boundary_d=boundary_n.dot(delta_pos);
|
|
||||||
//check if time of collision is outside Time::MIN..Time::MAX
|
|
||||||
//infinity_dir can always be treated as a velocity
|
|
||||||
if !boundary_d.is_positive()&&boundary_n.dot(infinity_dir).is_zero(){
|
|
||||||
//both faces cannot pass this condition, return early if one does.
|
|
||||||
return FEV::Face(face_id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
FEV::Edge(edge_id)
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// TODO: fundamentally improve this algorithm.
|
|
||||||
// All it needs to do is find the closest point on the mesh
|
|
||||||
// and return the FEV which the point resides on.
|
|
||||||
//
|
|
||||||
// What it actually does is use the above functions to trace a ray in from infinity,
|
|
||||||
// crawling the closest point along the mesh surface until the ray reaches
|
|
||||||
// the starting point to discover the final FEV.
|
|
||||||
//
|
|
||||||
// The actual collision prediction probably does a single test
|
|
||||||
// and then immediately returns with 0 FEV transitions on average,
|
|
||||||
// because of the strict time_limit constraint.
|
|
||||||
//
|
|
||||||
// Most of the calculation time is just calculating the starting point
|
|
||||||
// for the "actual" crawling algorithm below (predict_collision_{in|out}).
|
|
||||||
fn closest_fev_not_inside(&self,mut infinity_body:Body,start_time:Bound<&Time>)->Option<FEV<MinkowskiMesh<'_>>>{
|
|
||||||
infinity_body.infinity_dir().and_then(|dir|{
|
|
||||||
let infinity_fev=self.infinity_fev(-dir,infinity_body.position);
|
|
||||||
//a line is simpler to solve than a parabola
|
|
||||||
infinity_body.velocity=dir;
|
|
||||||
infinity_body.acceleration=vec3::zero();
|
|
||||||
//crawl in from negative infinity along a tangent line to get the closest fev
|
|
||||||
infinity_fev.crawl(self,&infinity_body,Bound::Unbounded,start_time).miss()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
pub fn predict_collision_in(&self,relative_body:&Body,range:impl RangeBounds<Time>)->Option<(MinkowskiFace,GigaTime)>{
|
pub fn predict_collision_in(&self,relative_body:&Body,range:impl RangeBounds<Time>)->Option<(MinkowskiFace,GigaTime)>{
|
||||||
self.closest_fev_not_inside(*relative_body,range.start_bound()).and_then(|fev|{
|
let fev=crate::minimum_difference::closest_fev_not_inside(self,relative_body.position)?;
|
||||||
//continue forwards along the body parabola
|
//continue forwards along the body parabola
|
||||||
fev.crawl(self,relative_body,range.start_bound(),range.end_bound()).hit()
|
fev.crawl(self,relative_body,range.start_bound(),range.end_bound()).hit()
|
||||||
})
|
|
||||||
}
|
}
|
||||||
pub fn predict_collision_out(&self,relative_body:&Body,range:impl RangeBounds<Time>)->Option<(MinkowskiFace,GigaTime)>{
|
pub fn predict_collision_out(&self,relative_body:&Body,range:impl RangeBounds<Time>)->Option<(MinkowskiFace,GigaTime)>{
|
||||||
let (lower_bound,upper_bound)=(range.start_bound(),range.end_bound());
|
let (lower_bound,upper_bound)=(range.start_bound(),range.end_bound());
|
||||||
|
// TODO: handle unbounded collision using infinity fev
|
||||||
|
let time=match upper_bound{
|
||||||
|
Bound::Included(&time)=>time,
|
||||||
|
Bound::Excluded(&time)=>time,
|
||||||
|
Bound::Unbounded=>unimplemented!("unbounded collision out"),
|
||||||
|
};
|
||||||
|
let fev=crate::minimum_difference::closest_fev_not_inside(self,relative_body.extrapolated_position(time))?;
|
||||||
// swap and negate bounds to do a time inversion
|
// swap and negate bounds to do a time inversion
|
||||||
let (lower_bound,upper_bound)=(upper_bound.map(|&t|-t),lower_bound.map(|&t|-t));
|
let (lower_bound,upper_bound)=(upper_bound.map(|&t|-t),lower_bound.map(|&t|-t));
|
||||||
let infinity_body=-relative_body;
|
let infinity_body=-relative_body;
|
||||||
self.closest_fev_not_inside(infinity_body,lower_bound.as_ref()).and_then(|fev|{
|
//continue backwards along the body parabola
|
||||||
//continue backwards along the body parabola
|
fev.crawl(self,&infinity_body,lower_bound.as_ref(),upper_bound.as_ref()).hit()
|
||||||
fev.crawl(self,&infinity_body,lower_bound.as_ref(),upper_bound.as_ref()).hit()
|
//no need to test -time<time_limit because of the first step
|
||||||
//no need to test -time<time_limit because of the first step
|
.map(|(face,time)|(face,-time))
|
||||||
.map(|(face,time)|(face,-time))
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
pub fn predict_collision_face_out(&self,relative_body:&Body,range:impl RangeBounds<Time>,contact_face_id:MinkowskiFace)->Option<(MinkowskiDirectedEdge,GigaTime)>{
|
pub fn predict_collision_face_out(&self,relative_body:&Body,range:impl RangeBounds<Time>,contact_face_id:MinkowskiFace)->Option<(MinkowskiDirectedEdge,GigaTime)>{
|
||||||
// TODO: make better
|
// TODO: make better
|
||||||
@@ -811,10 +721,7 @@ impl MinkowskiMesh<'_>{
|
|||||||
best_edge
|
best_edge
|
||||||
}
|
}
|
||||||
pub fn contains_point(&self,point:Planar64Vec3)->bool{
|
pub fn contains_point(&self,point:Planar64Vec3)->bool{
|
||||||
let contains_point_lua=crate::minimum_difference_lua::contains_point(self,point).unwrap();
|
crate::minimum_difference::contains_point(self,point)
|
||||||
let contains_point=crate::minimum_difference::contains_point(self,point);
|
|
||||||
println!("contains_point={contains_point} contains_point_lua={contains_point_lua}");
|
|
||||||
contains_point
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
impl MeshQuery for MinkowskiMesh<'_>{
|
impl MeshQuery for MinkowskiMesh<'_>{
|
||||||
@@ -856,6 +763,9 @@ impl MeshQuery for MinkowskiMesh<'_>{
|
|||||||
fn hint_point(&self)->Planar64Vec3{
|
fn hint_point(&self)->Planar64Vec3{
|
||||||
self.mesh0.transform.vertex.translation-self.mesh1.transform.vertex.translation
|
self.mesh0.transform.vertex.translation-self.mesh1.transform.vertex.translation
|
||||||
}
|
}
|
||||||
|
fn farthest_vert(&self,dir:Planar64Vec3)->MinkowskiVert{
|
||||||
|
MinkowskiVert::VertVert(self.mesh0.farthest_vert(dir),self.mesh1.farthest_vert(-dir))
|
||||||
|
}
|
||||||
fn face_edges(&self,face_id:MinkowskiFace)->impl AsRef<[MinkowskiDirectedEdge]>{
|
fn face_edges(&self,face_id:MinkowskiFace)->impl AsRef<[MinkowskiDirectedEdge]>{
|
||||||
match face_id{
|
match face_id{
|
||||||
MinkowskiFace::VertFace(v0,f1)=>{
|
MinkowskiFace::VertFace(v0,f1)=>{
|
||||||
|
|||||||
@@ -729,7 +729,7 @@ struct IntersectModel{
|
|||||||
transform:PhysicsMeshTransform,
|
transform:PhysicsMeshTransform,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug,Clone,Copy,Eq,Hash,PartialEq)]
|
#[derive(Debug,Clone,Copy,Hash)]
|
||||||
pub struct ContactCollision{
|
pub struct ContactCollision{
|
||||||
convex_mesh_id:ConvexMeshId<ContactModelId>,
|
convex_mesh_id:ConvexMeshId<ContactModelId>,
|
||||||
face_id:model_physics::MinkowskiFace,
|
face_id:model_physics::MinkowskiFace,
|
||||||
@@ -738,7 +738,7 @@ pub struct ContactCollision{
|
|||||||
pub struct IntersectCollision{
|
pub struct IntersectCollision{
|
||||||
convex_mesh_id:ConvexMeshId<IntersectModelId>,
|
convex_mesh_id:ConvexMeshId<IntersectModelId>,
|
||||||
}
|
}
|
||||||
#[derive(Debug,Clone,Eq,Hash,PartialEq)]
|
#[derive(Debug,Clone,Hash)]
|
||||||
pub enum Collision{
|
pub enum Collision{
|
||||||
Contact(ContactCollision),
|
Contact(ContactCollision),
|
||||||
Intersect(IntersectCollision),
|
Intersect(IntersectCollision),
|
||||||
|
|||||||
@@ -366,7 +366,7 @@ impl_additive_operator!( Fixed, BitXor, bitxor, Self );
|
|||||||
// non-wide operators. The result is the same width as the inputs.
|
// non-wide operators. The result is the same width as the inputs.
|
||||||
|
|
||||||
// This macro is not used in the default configuration.
|
// This macro is not used in the default configuration.
|
||||||
#[allow(unused_macros)]
|
#[expect(unused_macros)]
|
||||||
macro_rules! impl_multiplicative_operator_not_const_generic {
|
macro_rules! impl_multiplicative_operator_not_const_generic {
|
||||||
( ($struct: ident, $trait: ident, $method: ident, $output: ty ), $width:expr ) => {
|
( ($struct: ident, $trait: ident, $method: ident, $output: ty ), $width:expr ) => {
|
||||||
impl<const F:usize> core::ops::$trait for $struct<$width,F>{
|
impl<const F:usize> core::ops::$trait for $struct<$width,F>{
|
||||||
@@ -545,7 +545,7 @@ impl_shift_operator!( Fixed, Shr, shr, Self );
|
|||||||
|
|
||||||
// wide operators. The result width is the sum of the input widths, i.e. none of the multiplication
|
// wide operators. The result width is the sum of the input widths, i.e. none of the multiplication
|
||||||
|
|
||||||
#[allow(unused_macros)]
|
#[expect(unused_macros)]
|
||||||
macro_rules! impl_wide_operators{
|
macro_rules! impl_wide_operators{
|
||||||
($lhs:expr,$rhs:expr)=>{
|
($lhs:expr,$rhs:expr)=>{
|
||||||
impl core::ops::Mul<Fixed<$rhs,{$rhs*32}>> for Fixed<$lhs,{$lhs*32}>{
|
impl core::ops::Mul<Fixed<$rhs,{$rhs*32}>> for Fixed<$lhs,{$lhs*32}>{
|
||||||
|
|||||||
@@ -13,16 +13,16 @@ authors = ["Rhys Lloyd <krakow20@gmail.com>"]
|
|||||||
bytemuck = "1.14.3"
|
bytemuck = "1.14.3"
|
||||||
glam = "0.30.0"
|
glam = "0.30.0"
|
||||||
regex = { version = "1.11.3", default-features = false }
|
regex = { version = "1.11.3", default-features = false }
|
||||||
rbx_binary = { version = "1.0.1-sn5", registry = "strafesnet" }
|
|
||||||
rbx_dom_weak = { version = "3.0.1-sn5", registry = "strafesnet" }
|
|
||||||
rbx_mesh = "0.5.0"
|
rbx_mesh = "0.5.0"
|
||||||
rbx_reflection = "5.0.0"
|
|
||||||
rbx_reflection_database = "1.0.0"
|
|
||||||
rbx_xml = { version = "1.0.1-sn5", registry = "strafesnet" }
|
|
||||||
rbxassetid = { version = "0.1.0", path = "../rbxassetid", registry = "strafesnet" }
|
rbxassetid = { version = "0.1.0", path = "../rbxassetid", registry = "strafesnet" }
|
||||||
roblox_emulator = { version = "0.5.1", path = "../roblox_emulator", default-features = false, registry = "strafesnet" }
|
roblox_emulator = { version = "0.5.1", path = "../roblox_emulator", default-features = false, registry = "strafesnet" }
|
||||||
strafesnet_common = { version = "0.7.0", path = "../common", registry = "strafesnet" }
|
strafesnet_common = { version = "0.7.0", path = "../common", registry = "strafesnet" }
|
||||||
strafesnet_deferred_loader = { version = "0.5.1", path = "../deferred_loader", registry = "strafesnet" }
|
strafesnet_deferred_loader = { version = "0.5.1", path = "../deferred_loader", registry = "strafesnet" }
|
||||||
|
rbx_binary = "2.0.1"
|
||||||
|
rbx_dom_weak = "4.1.0"
|
||||||
|
rbx_reflection = "6.1.0"
|
||||||
|
rbx_reflection_database = "2.0.2"
|
||||||
|
rbx_xml = "2.0.1"
|
||||||
|
|
||||||
[lints]
|
[lints]
|
||||||
workspace = true
|
workspace = true
|
||||||
|
|||||||
@@ -559,7 +559,7 @@ pub fn convert<'a>(
|
|||||||
//just going to leave it like this for now instead of reworking the data structures for this whole thing
|
//just going to leave it like this for now instead of reworking the data structures for this whole thing
|
||||||
let textureless_render_group=render_config_deferred_loader.acquire_render_config_id(None);
|
let textureless_render_group=render_config_deferred_loader.acquire_render_config_id(None);
|
||||||
|
|
||||||
let db=rbx_reflection_database::get();
|
let db=rbx_reflection_database::get().unwrap();
|
||||||
let basepart=&db.classes["BasePart"];
|
let basepart=&db.classes["BasePart"];
|
||||||
let baseparts=dom.descendants().filter(|&instance|
|
let baseparts=dom.descendants().filter(|&instance|
|
||||||
db.classes.get(instance.class.as_str()).is_some_and(|class|
|
db.classes.get(instance.class.as_str()).is_some_and(|class|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "roblox_emulator"
|
name = "roblox_emulator"
|
||||||
version = "0.5.1"
|
version = "0.5.2"
|
||||||
edition = "2024"
|
edition = "2024"
|
||||||
repository = "https://git.itzana.me/StrafesNET/strafe-project"
|
repository = "https://git.itzana.me/StrafesNET/strafe-project"
|
||||||
license = "MIT OR Apache-2.0"
|
license = "MIT OR Apache-2.0"
|
||||||
@@ -15,10 +15,10 @@ run-service=[]
|
|||||||
glam = "0.30.0"
|
glam = "0.30.0"
|
||||||
mlua = { version = "0.11.3", features = ["luau"] }
|
mlua = { version = "0.11.3", features = ["luau"] }
|
||||||
phf = { version = "0.13.1", features = ["macros"] }
|
phf = { version = "0.13.1", features = ["macros"] }
|
||||||
rbx_dom_weak = { version = "3.0.1-sn5", registry = "strafesnet" }
|
rbx_dom_weak = "4.1.0"
|
||||||
rbx_reflection = "5.0.0"
|
rbx_reflection = "6.1.0"
|
||||||
rbx_reflection_database = "1.0.0"
|
rbx_reflection_database = "2.0.2"
|
||||||
rbx_types = "2.0.0"
|
rbx_types = "3.1.0"
|
||||||
|
|
||||||
[lints]
|
[lints]
|
||||||
workspace = true
|
workspace = true
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ impl Context{
|
|||||||
}
|
}
|
||||||
/// Creates an iterator over all items of a particular class.
|
/// Creates an iterator over all items of a particular class.
|
||||||
pub fn superclass_iter<'a>(&'a self,superclass:&'a str)->impl Iterator<Item=Ref>+'a{
|
pub fn superclass_iter<'a>(&'a self,superclass:&'a str)->impl Iterator<Item=Ref>+'a{
|
||||||
let db=rbx_reflection_database::get();
|
let db=rbx_reflection_database::get().unwrap();
|
||||||
let Some(superclass)=db.classes.get(superclass)else{
|
let Some(superclass)=db.classes.get(superclass)else{
|
||||||
panic!("Invalid class");
|
panic!("Invalid class");
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ impl PartialEq for EnumItem<'_>{
|
|||||||
pub struct Enums;
|
pub struct Enums;
|
||||||
impl Enums{
|
impl Enums{
|
||||||
pub fn get(&self,index:&str)->Option<EnumItems<'static>>{
|
pub fn get(&self,index:&str)->Option<EnumItems<'static>>{
|
||||||
let db=rbx_reflection_database::get();
|
let db=rbx_reflection_database::get().unwrap();
|
||||||
db.enums.get(index).map(|ed|EnumItems{ed})
|
db.enums.get(index).map(|ed|EnumItems{ed})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ pub fn dom_mut<T>(lua:&mlua::Lua,mut f:impl FnMut(&mut WeakDom)->mlua::Result<T>
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn class_is_a(class:&str,superclass:&str)->bool{
|
pub fn class_is_a(class:&str,superclass:&str)->bool{
|
||||||
let db=rbx_reflection_database::get();
|
let db=rbx_reflection_database::get().unwrap();
|
||||||
let (Some(class),Some(superclass))=(db.classes.get(class),db.classes.get(superclass))else{
|
let (Some(class),Some(superclass))=(db.classes.get(class),db.classes.get(superclass))else{
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
@@ -80,14 +80,14 @@ pub fn find_first_descendant_of_class<'a>(dom:&'a WeakDom,instance:&rbx_dom_weak
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn find_first_child_which_is_a<'a>(dom:&'a WeakDom,instance:&rbx_dom_weak::Instance,superclass:&str)->Option<&'a rbx_dom_weak::Instance>{
|
pub fn find_first_child_which_is_a<'a>(dom:&'a WeakDom,instance:&rbx_dom_weak::Instance,superclass:&str)->Option<&'a rbx_dom_weak::Instance>{
|
||||||
let db=rbx_reflection_database::get();
|
let db=rbx_reflection_database::get().unwrap();
|
||||||
let superclass_descriptor=db.classes.get(superclass)?;
|
let superclass_descriptor=db.classes.get(superclass)?;
|
||||||
instance.children().iter().filter_map(|&r|dom.get_by_ref(r)).find(|inst|{
|
instance.children().iter().filter_map(|&r|dom.get_by_ref(r)).find(|inst|{
|
||||||
db.classes.get(inst.class.as_str()).is_some_and(|descriptor|db.has_superclass(descriptor,superclass_descriptor))
|
db.classes.get(inst.class.as_str()).is_some_and(|descriptor|db.has_superclass(descriptor,superclass_descriptor))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
pub fn find_first_descendant_which_is_a<'a>(dom:&'a WeakDom,instance:&rbx_dom_weak::Instance,superclass:&str)->Option<&'a rbx_dom_weak::Instance>{
|
pub fn find_first_descendant_which_is_a<'a>(dom:&'a WeakDom,instance:&rbx_dom_weak::Instance,superclass:&str)->Option<&'a rbx_dom_weak::Instance>{
|
||||||
let db=rbx_reflection_database::get();
|
let db=rbx_reflection_database::get().unwrap();
|
||||||
let superclass_descriptor=db.classes.get(superclass)?;
|
let superclass_descriptor=db.classes.get(superclass)?;
|
||||||
dom.descendants_of(instance.referent()).find(|inst|{
|
dom.descendants_of(instance.referent()).find(|inst|{
|
||||||
db.classes.get(inst.class.as_str()).is_some_and(|descriptor|db.has_superclass(descriptor,superclass_descriptor))
|
db.classes.get(inst.class.as_str()).is_some_and(|descriptor|db.has_superclass(descriptor,superclass_descriptor))
|
||||||
@@ -282,7 +282,7 @@ impl mlua::UserData for Instance{
|
|||||||
dom_mut(lua,|dom|{
|
dom_mut(lua,|dom|{
|
||||||
let instance=this.get(dom)?;
|
let instance=this.get(dom)?;
|
||||||
//println!("__index t={} i={index:?}",instance.name);
|
//println!("__index t={} i={index:?}",instance.name);
|
||||||
let db=rbx_reflection_database::get();
|
let db=rbx_reflection_database::get().unwrap();
|
||||||
let class=db.classes.get(instance.class.as_str()).ok_or_else(||mlua::Error::runtime("Class missing"))?;
|
let class=db.classes.get(instance.class.as_str()).ok_or_else(||mlua::Error::runtime("Class missing"))?;
|
||||||
// Find existing property
|
// Find existing property
|
||||||
// Interestingly, ustr can know ahead of time if
|
// Interestingly, ustr can know ahead of time if
|
||||||
@@ -344,7 +344,7 @@ impl mlua::UserData for Instance{
|
|||||||
let index_str=&*index.to_str()?;
|
let index_str=&*index.to_str()?;
|
||||||
dom_mut(lua,|dom|{
|
dom_mut(lua,|dom|{
|
||||||
let instance=this.get_mut(dom)?;
|
let instance=this.get_mut(dom)?;
|
||||||
let db=rbx_reflection_database::get();
|
let db=rbx_reflection_database::get().unwrap();
|
||||||
let class=db.classes.get(instance.class.as_str()).ok_or_else(||mlua::Error::runtime("Class missing"))?;
|
let class=db.classes.get(instance.class.as_str()).ok_or_else(||mlua::Error::runtime("Class missing"))?;
|
||||||
let property=db.superclasses_iter(class).find_map(|cls|
|
let property=db.superclasses_iter(class).find_map(|cls|
|
||||||
cls.properties.get(index_str)
|
cls.properties.get(index_str)
|
||||||
|
|||||||
@@ -13,10 +13,10 @@ futures = "0.3.31"
|
|||||||
image = "0.25.2"
|
image = "0.25.2"
|
||||||
image_dds = "0.7.1"
|
image_dds = "0.7.1"
|
||||||
rbx_asset = { version = "0.5.0", registry = "strafesnet" }
|
rbx_asset = { version = "0.5.0", registry = "strafesnet" }
|
||||||
rbx_binary = { version = "1.0.1-sn5", registry = "strafesnet" }
|
rbx_binary = "2.0.1"
|
||||||
rbx_dom_weak = { version = "3.0.1-sn5", registry = "strafesnet" }
|
rbx_dom_weak = "4.1.0"
|
||||||
rbx_reflection_database = "1.0.0"
|
rbx_reflection_database = "2.0.2"
|
||||||
rbx_xml = { version = "1.0.1-sn5", registry = "strafesnet" }
|
rbx_xml = "2.0.1"
|
||||||
rbxassetid = { version = "0.1.0", registry = "strafesnet" }
|
rbxassetid = { version = "0.1.0", registry = "strafesnet" }
|
||||||
strafesnet_bsp_loader = { version = "0.3.1", path = "../lib/bsp_loader", registry = "strafesnet" }
|
strafesnet_bsp_loader = { version = "0.3.1", path = "../lib/bsp_loader", registry = "strafesnet" }
|
||||||
strafesnet_deferred_loader = { version = "0.5.1", path = "../lib/deferred_loader", registry = "strafesnet" }
|
strafesnet_deferred_loader = { version = "0.5.1", path = "../lib/deferred_loader", registry = "strafesnet" }
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ strafesnet_rbx_loader = { path = "../lib/rbx_loader", registry = "strafesnet", o
|
|||||||
strafesnet_session = { path = "../engine/session", registry = "strafesnet" }
|
strafesnet_session = { path = "../engine/session", registry = "strafesnet" }
|
||||||
strafesnet_settings = { path = "../engine/settings", registry = "strafesnet" }
|
strafesnet_settings = { path = "../engine/settings", registry = "strafesnet" }
|
||||||
strafesnet_snf = { path = "../lib/snf", registry = "strafesnet", optional = true }
|
strafesnet_snf = { path = "../lib/snf", registry = "strafesnet", optional = true }
|
||||||
wgpu = "27.0.0"
|
wgpu = "28.0.0"
|
||||||
winit = "0.30.7"
|
winit = "0.30.7"
|
||||||
|
|
||||||
[profile.dev]
|
[profile.dev]
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ impl<'a> SetupContextPartial2<'a>{
|
|||||||
let required_features=required_features();
|
let required_features=required_features();
|
||||||
|
|
||||||
//no helper function smh gotta write it myself
|
//no helper function smh gotta write it myself
|
||||||
let adapters=self.instance.enumerate_adapters(self.backends);
|
let adapters=pollster::block_on(self.instance.enumerate_adapters(self.backends));
|
||||||
|
|
||||||
let mut chosen_adapter=None;
|
let mut chosen_adapter=None;
|
||||||
let mut chosen_adapter_score=0;
|
let mut chosen_adapter_score=0;
|
||||||
|
|||||||
Reference in New Issue
Block a user