2023-10-26 04:31:53 +00:00
|
|
|
use crate::physics::Body;
|
2023-10-27 21:18:50 +00:00
|
|
|
use crate::model_physics::{FEV,MeshQuery};
|
2023-10-27 02:38:37 +00:00
|
|
|
use crate::integer::{Time,Planar64,Planar64Vec3};
|
|
|
|
use crate::zeroes::zeroes2;
|
2023-10-26 04:31:53 +00:00
|
|
|
|
2023-10-27 21:18:50 +00:00
|
|
|
struct State<FEV>{
|
2023-10-26 04:22:28 +00:00
|
|
|
fev:FEV,
|
2023-10-27 02:38:34 +00:00
|
|
|
time:Time,
|
2023-10-26 04:22:28 +00:00
|
|
|
}
|
|
|
|
|
2023-10-27 21:18:50 +00:00
|
|
|
enum Transition<F,E,V>{
|
2023-10-26 04:22:28 +00:00
|
|
|
Miss,
|
2023-10-27 21:18:50 +00:00
|
|
|
Next(FEV<F,E,V>,Time),
|
|
|
|
Hit(F,Time),
|
2023-10-26 04:22:28 +00:00
|
|
|
}
|
2023-10-26 04:31:53 +00:00
|
|
|
|
2023-10-27 21:18:50 +00:00
|
|
|
impl<F:Copy,E:Copy,V:Copy> State<FEV<F,E,V>>{
|
|
|
|
fn next_transition(&self,mesh:&impl MeshQuery<F,E,V>,body:&Body,time_limit:Time)->Transition<F,E,V>{
|
2023-10-26 04:31:53 +00:00
|
|
|
//conflicting derivative means it crosses in the wrong direction.
|
2023-10-26 22:51:47 +00:00
|
|
|
//if the transition time is equal to an already tested transition, do not replace the current best.
|
2023-10-27 02:37:59 +00:00
|
|
|
let mut best_time=time_limit;
|
|
|
|
let mut best_transtition=Transition::Miss;
|
2023-10-26 22:51:47 +00:00
|
|
|
match &self.fev{
|
2023-10-27 21:18:50 +00:00
|
|
|
&FEV::<F,E,V>::Face(face_id)=>{
|
2023-10-27 02:37:59 +00:00
|
|
|
//test own face collision time, ignoring roots with zero or conflicting derivative
|
|
|
|
//n=face.normal d=face.dot
|
|
|
|
//n.a t^2+n.v t+n.p-d==0
|
|
|
|
let (n,d)=mesh.face_nd(face_id);
|
|
|
|
for t in zeroes2((n.dot(body.position)-d)*2,n.dot(body.velocity)*2,n.dot(body.acceleration)){
|
|
|
|
let t=body.time+Time::from(t);
|
|
|
|
if self.time<t&&t<best_time&&n.dot(body.extrapolated_velocity(t))<Planar64::ZERO{
|
|
|
|
best_time=t;
|
|
|
|
best_transtition=Transition::Hit(face_id,t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//test each edge collision time, ignoring roots with zero or conflicting derivative
|
2023-11-16 04:34:15 +00:00
|
|
|
for &edge_id in mesh.face_edges(face_id).iter(){
|
2023-11-16 03:48:13 +00:00
|
|
|
let edge_n=mesh.edge_n(edge_id);
|
|
|
|
let n=n.cross(edge_n);
|
|
|
|
//picking a vert randomly is terrible
|
|
|
|
let d=n.dot(mesh.vert(mesh.edge_verts(edge_id)[0]));
|
2023-10-27 02:37:59 +00:00
|
|
|
for t in zeroes2((n.dot(body.position)-d)*2,n.dot(body.velocity)*2,n.dot(body.acceleration)){
|
|
|
|
let t=body.time+Time::from(t);
|
|
|
|
if self.time<t&&t<best_time&&n.dot(body.extrapolated_velocity(t))<Planar64::ZERO{
|
|
|
|
best_time=t;
|
2023-10-27 21:18:50 +00:00
|
|
|
best_transtition=Transition::Next(FEV::<F,E,V>::Edge(edge_id),t);
|
2023-10-27 02:37:59 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-10-26 22:51:47 +00:00
|
|
|
//if none:
|
2023-10-26 04:31:53 +00:00
|
|
|
},
|
2023-10-27 21:18:50 +00:00
|
|
|
&FEV::<F,E,V>::Edge(edge_id)=>{
|
2023-10-27 02:37:59 +00:00
|
|
|
//test each face collision time, ignoring roots with zero or conflicting derivative
|
2023-11-16 03:48:13 +00:00
|
|
|
let edge_n=mesh.edge_n(edge_id);
|
2023-10-27 23:04:44 +00:00
|
|
|
for &test_face_id in mesh.edge_faces(edge_id).iter(){
|
2023-11-16 03:48:13 +00:00
|
|
|
let face_n=mesh.face_nd(test_face_id).0;
|
|
|
|
let n=edge_n.cross(face_n);
|
|
|
|
let d=n.dot(mesh.vert(mesh.edge_verts(edge_id)[0]));
|
2023-10-27 02:37:59 +00:00
|
|
|
for t in zeroes2((n.dot(body.position)-d)*2,n.dot(body.velocity)*2,n.dot(body.acceleration)){
|
|
|
|
let t=body.time+Time::from(t);
|
|
|
|
if self.time<t&&t<best_time&&n.dot(body.extrapolated_velocity(t))<Planar64::ZERO{
|
|
|
|
best_time=t;
|
2023-10-27 21:18:50 +00:00
|
|
|
best_transtition=Transition::Next(FEV::<F,E,V>::Face(test_face_id),t);
|
2023-10-27 02:37:59 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//test each vertex collision time, ignoring roots with zero or conflicting derivative
|
2023-11-01 22:08:27 +00:00
|
|
|
let n=mesh.edge_n(edge_id);
|
|
|
|
for &vert_id in mesh.edge_verts(edge_id).iter(){
|
|
|
|
let d=n.dot(mesh.vert(vert_id));
|
2023-10-27 02:37:59 +00:00
|
|
|
for t in zeroes2((n.dot(body.position)-d)*2,n.dot(body.velocity)*2,n.dot(body.acceleration)){
|
|
|
|
let t=body.time+Time::from(t);
|
|
|
|
if self.time<t&&t<best_time&&n.dot(body.extrapolated_velocity(t))<Planar64::ZERO{
|
|
|
|
best_time=t;
|
2023-10-27 21:18:50 +00:00
|
|
|
best_transtition=Transition::Next(FEV::<F,E,V>::Vert(vert_id),t);
|
2023-10-27 02:37:59 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-10-26 22:51:47 +00:00
|
|
|
//if none:
|
2023-10-26 04:31:53 +00:00
|
|
|
},
|
2023-11-01 22:08:27 +00:00
|
|
|
&FEV::<F,E,V>::Vert(vert_id)=>{
|
2023-10-27 02:37:59 +00:00
|
|
|
//test each edge collision time, ignoring roots with zero or conflicting derivative
|
2023-11-01 22:08:27 +00:00
|
|
|
for &edge_id in mesh.vert_edges(vert_id).iter(){
|
|
|
|
let n=mesh.edge_n(edge_id);
|
|
|
|
let d=n.dot(mesh.vert(vert_id));
|
2023-10-27 02:37:59 +00:00
|
|
|
for t in zeroes2((n.dot(body.position)-d)*2,n.dot(body.velocity)*2,n.dot(body.acceleration)){
|
|
|
|
let t=body.time+Time::from(t);
|
|
|
|
if self.time<t&&t<best_time&&n.dot(body.extrapolated_velocity(t))<Planar64::ZERO{
|
|
|
|
best_time=t;
|
2023-10-27 21:18:50 +00:00
|
|
|
best_transtition=Transition::Next(FEV::<F,E,V>::Edge(edge_id),t);
|
2023-10-27 02:37:59 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-10-26 22:51:47 +00:00
|
|
|
//if none:
|
2023-10-26 04:31:53 +00:00
|
|
|
},
|
|
|
|
}
|
2023-10-27 02:37:59 +00:00
|
|
|
best_transtition
|
2023-10-26 04:31:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-27 21:18:50 +00:00
|
|
|
pub fn predict_collision<F:Copy,E:Copy,V:Copy>(mesh:&impl MeshQuery<F,E,V>,relative_body:&Body,time_limit:Time)->Option<(F,Time)>{
|
2023-10-26 23:18:32 +00:00
|
|
|
let mut state=State{
|
|
|
|
fev:mesh.closest_fev(relative_body.position),
|
2023-10-27 02:38:37 +00:00
|
|
|
time:relative_body.time,
|
2023-10-26 23:18:32 +00:00
|
|
|
};
|
|
|
|
//it would be possible to write down the point of closest approach...
|
|
|
|
loop{
|
|
|
|
match state.next_transition(mesh,relative_body,time_limit){
|
|
|
|
Transition::Miss=>return None,
|
2023-10-27 02:38:37 +00:00
|
|
|
Transition::Next(fev,time)=>(state.fev,state.time)=(fev,time),
|
|
|
|
Transition::Hit(face,time)=>return Some((face,time)),
|
2023-10-26 04:31:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-10-27 03:52:38 +00:00
|
|
|
|
2023-11-01 04:32:00 +00:00
|
|
|
pub fn predict_collision_end<F:Copy,E:Copy,V:Copy>(mesh:&impl MeshQuery<F,E,V>,relative_body:&Body,time_limit:Time,ignore_face_id:F)->Option<(F,Time)>{
|
2023-10-27 03:52:38 +00:00
|
|
|
//imagine the mesh without the collision face
|
|
|
|
//no algorithm needed, there is only one state and three cases (Face,Edge,None)
|
|
|
|
//determine when it passes an edge ("sliding off" case) or if it leaves the surface directly
|
2023-10-31 05:11:54 +00:00
|
|
|
//the state can be constructed from the ContactCollision directly
|
2023-10-27 03:52:38 +00:00
|
|
|
None
|
|
|
|
}
|