strafe-project/src/model_physics.rs

766 lines
25 KiB
Rust
Raw Normal View History

2023-10-26 16:18:32 -07:00
use crate::integer::{Planar64,Planar64Vec3};
use std::borrow::{Borrow,Cow};
2023-10-26 16:18:32 -07:00
2023-10-30 22:11:54 -07:00
#[derive(Debug,Clone,Copy,Hash,Eq,PartialEq)]
2023-10-26 16:18:32 -07:00
pub struct VertId(usize);
2023-10-30 22:11:54 -07:00
#[derive(Debug,Clone,Copy,Hash,Eq,PartialEq)]
2023-10-26 16:18:32 -07:00
pub struct EdgeId(usize);
2023-11-10 15:23:28 -08:00
impl EdgeId{
fn as_directed_edge_id(&self,parity:bool)->DirectedEdgeId{
DirectedEdgeId(self.0|((parity as usize)<<(usize::BITS-1)))
}
}
/// DirectedEdgeId refers to an EdgeId when undirected.
#[derive(Debug,Clone,Copy,Hash,Eq,PartialEq)]
pub struct DirectedEdgeId(usize);
impl DirectedEdgeId{
fn as_edge_id(&self)->EdgeId{
EdgeId(self.0&!(1<<(usize::BITS-1)))
}
fn signum(&self)->isize{
((self.0&(1<<(usize::BITS-1))!=0) as isize)*2-1
}
}
2023-10-30 22:11:54 -07:00
#[derive(Debug,Clone,Copy,Hash,Eq,PartialEq)]
2023-10-26 16:18:32 -07:00
pub struct FaceId(usize);
//Vertex <-> Edge <-> Face -> Collide
2023-10-27 14:18:50 -07:00
pub enum FEV<F,E,V>{
Face(F),
Edge(E),
Vert(V),
2023-10-26 16:18:32 -07:00
}
//use Unit32 #[repr(C)] for map files
struct Face{
normal:Planar64Vec3,
dot:Planar64,
}
2023-11-02 16:44:14 -07:00
impl Face{
fn nd(&self)->(Planar64Vec3,Planar64){
(self.normal,self.dot)
}
}
2023-10-27 14:18:50 -07:00
struct Vert(Planar64Vec3);
2023-10-26 16:18:32 -07:00
struct FaceRefs{
2023-10-26 19:38:34 -07:00
edges:Vec<(EdgeId,FaceId)>,
//verts:Vec<VertId>,
2023-10-26 16:18:32 -07:00
}
struct EdgeRefs{
2023-10-26 16:28:07 -07:00
faces:[FaceId;2],//left, right
verts:[VertId;2],//bottom, top
2023-10-26 16:18:32 -07:00
}
struct VertRefs{
2023-11-10 14:02:17 -08:00
faces:Vec<FaceId>,
2023-11-10 15:23:28 -08:00
edges:Vec<DirectedEdgeId>,
2023-10-26 16:18:32 -07:00
}
pub struct PhysicsMesh{
faces:Vec<Face>,
2023-10-27 14:18:50 -07:00
verts:Vec<Vert>,
2023-10-26 16:18:32 -07:00
face_topology:Vec<FaceRefs>,
edge_topology:Vec<EdgeRefs>,
vert_topology:Vec<VertRefs>,
}
2023-10-27 14:18:50 -07:00
2023-10-31 15:25:06 -07:00
#[derive(Default,Clone)]
struct VertRefGuy{
2023-11-10 15:23:28 -08:00
edges:std::collections::HashSet<DirectedEdgeId>,
2023-11-10 14:02:17 -08:00
faces:std::collections::HashSet<FaceId>,
2023-10-31 15:25:06 -07:00
}
2023-11-03 18:57:34 -07:00
#[derive(Clone,Hash,Eq,PartialEq)]
2023-10-31 15:25:06 -07:00
struct EdgeIdGuy([VertId;2]);
impl EdgeIdGuy{
2023-11-10 15:23:28 -08:00
fn new(v0:VertId,v1:VertId)->(Self,bool){
(if v0.0<v1.0{
2023-10-31 15:25:06 -07:00
Self([v0,v1])
}else{
Self([v1,v0])
2023-11-10 15:23:28 -08:00
},v0.0<v1.0)
2023-10-31 15:25:06 -07:00
}
}
struct EdgeRefGuy([FaceId;2]);
impl EdgeRefGuy{
fn new()->Self{
Self([FaceId(0);2])
}
fn push(&mut self,i:usize,face_id:FaceId){
self.0[i]=face_id;
}
}
struct FaceRefGuy(Vec<EdgeId>);
#[derive(Default)]
struct EdgePool{
edge_guys:Vec<(EdgeIdGuy,EdgeRefGuy)>,
edge_id_from_guy:std::collections::HashMap<EdgeIdGuy,usize>,
}
impl EdgePool{
2023-11-15 17:59:24 -08:00
fn push(&mut self,edge_id_guy:EdgeIdGuy)->(&mut EdgeRefGuy,EdgeId){
let edge_id=if let Some(&edge_id)=self.edge_id_from_guy.get(&edge_id_guy){
edge_id
2023-10-31 15:25:06 -07:00
}else{
let edge_id=self.edge_guys.len();
2023-11-03 18:57:34 -07:00
self.edge_guys.push((edge_id_guy.clone(),EdgeRefGuy::new()));
2023-10-31 15:25:06 -07:00
self.edge_id_from_guy.insert(edge_id_guy,edge_id);
2023-11-15 17:59:24 -08:00
edge_id
};
(&mut unsafe{self.edge_guys.get_unchecked_mut(edge_id)}.1,EdgeId(edge_id))
2023-10-31 15:25:06 -07:00
}
}
impl From<&crate::model::IndexedModel> for PhysicsMesh{
fn from(indexed_model:&crate::model::IndexedModel)->Self{
let verts=indexed_model.unique_pos.iter().map(|v|Vert(v.clone())).collect();
2023-11-10 15:23:28 -08:00
let mut vert_ref_guys=vec![VertRefGuy::default();indexed_model.unique_pos.len()];
2023-10-31 15:25:06 -07:00
let mut edge_pool=EdgePool::default();
2023-11-06 15:57:55 -08:00
let mut face_i=0;
let mut faces=Vec::new();
let mut face_ref_guys=Vec::new();
for group in indexed_model.groups.iter(){for poly in group.polys.iter(){
let face_id=FaceId(face_i);
2023-10-31 15:25:06 -07:00
//one face per poly
let mut normal=Planar64Vec3::ZERO;
let len=poly.vertices.len();
let face_edges=poly.vertices.iter().enumerate().map(|(i,&vert_id)|{
2023-11-03 19:08:34 -07:00
let vert0_id=indexed_model.unique_vertices[vert_id as usize].pos as usize;
let vert1_id=indexed_model.unique_vertices[poly.vertices[(i+1)%len] as usize].pos as usize;
2023-10-31 15:25:06 -07:00
//https://www.khronos.org/opengl/wiki/Calculating_a_Surface_Normal (Newell's Method)
let v0=indexed_model.unique_pos[vert0_id];
let v1=indexed_model.unique_pos[vert1_id];
normal+=Planar64Vec3::new(
(v0.y()-v1.y())*(v0.z()+v1.z()),
(v0.z()-v1.z())*(v0.x()+v1.x()),
(v0.x()-v1.x())*(v0.y()+v1.y()),
);
//get/create edge and push face into it
2023-11-10 15:23:28 -08:00
let (edge_id_guy,is_sorted)=EdgeIdGuy::new(VertId(vert0_id),VertId(vert1_id));
2023-11-15 17:59:24 -08:00
let (edge_ref_guy,edge_id)=edge_pool.push(edge_id_guy);
//polygon vertices as assumed to be listed clockwise
//populate the edge face on the left or right depending on how the edge vertices got sorted
edge_ref_guy.push(is_sorted as usize,face_id);
2023-11-10 14:02:17 -08:00
//index edges & face into vertices
{
2023-11-10 15:23:28 -08:00
let vert_ref_guy=unsafe{vert_ref_guys.get_unchecked_mut(vert0_id)};
vert_ref_guy.edges.insert(edge_id.as_directed_edge_id(!is_sorted));
2023-11-10 14:02:17 -08:00
vert_ref_guy.faces.insert(face_id);
2023-11-10 15:23:28 -08:00
unsafe{vert_ref_guys.get_unchecked_mut(vert1_id)}.edges.insert(edge_id.as_directed_edge_id(is_sorted));
2023-11-10 14:02:17 -08:00
}
2023-10-31 15:25:06 -07:00
//return edge_id
edge_id
}).collect();
//choose precision loss randomly idk
normal=normal/len as i64;
let mut dot=Planar64::ZERO;
for &v in poly.vertices.iter(){
2023-11-03 19:08:34 -07:00
dot+=normal.dot(indexed_model.unique_pos[indexed_model.unique_vertices[v as usize].pos as usize]);
2023-10-31 15:25:06 -07:00
}
2023-11-06 15:57:55 -08:00
faces.push(Face{normal,dot:dot/len as i64});
face_ref_guys.push(FaceRefGuy(face_edges));
face_i+=1;
}}
2023-10-31 15:25:06 -07:00
//conceivably faces, edges, and vertices exist now
Self{
faces,
verts,
face_topology:face_ref_guys.into_iter().enumerate().map(|(i,face_ref_guy)|{
let face_id=FaceId(i);
FaceRefs{edges:face_ref_guy.0.into_iter().map(|edge_id|{
//get the edge face that's not this face
2023-11-03 19:43:40 -07:00
let edge_faces=&edge_pool.edge_guys[edge_id.0].1.0;
2023-10-31 15:25:06 -07:00
if edge_faces[0]==face_id{
(edge_id,edge_faces[1])
}else if edge_faces[1]==face_id{
(edge_id,edge_faces[0])
}else{
2023-11-03 19:43:57 -07:00
panic!("edge does not contain face edge_faces={:?} face={:?}",edge_faces,face_id)
2023-10-31 15:25:06 -07:00
}
}).collect()}
}).collect(),
edge_topology:edge_pool.edge_guys.into_iter().map(|(edge_id_guy,edge_ref_guy)|
EdgeRefs{faces:edge_ref_guy.0,verts:edge_id_guy.0}
).collect(),
2023-11-10 15:23:28 -08:00
vert_topology:vert_ref_guys.into_iter().map(|vert_ref_guy|
2023-11-10 14:02:17 -08:00
VertRefs{
edges:vert_ref_guy.edges.into_iter().collect(),
faces:vert_ref_guy.faces.into_iter().collect(),
}
2023-10-31 15:25:06 -07:00
).collect(),
}
}
}
2023-10-27 15:10:38 -07:00
pub trait MeshQuery<FACE:Clone,EDGE:Clone,VERT:Clone>{
2023-10-27 14:18:50 -07:00
fn closest_fev(&self,point:Planar64Vec3)->FEV<FACE,EDGE,VERT>;
2023-11-01 15:08:12 -07:00
fn edge_n(&self,edge_id:EDGE)->Planar64Vec3{
let verts=self.edge_verts(edge_id);
2023-11-03 18:57:34 -07:00
self.vert(verts[1].clone())-self.vert(verts[0].clone())
2023-11-01 15:08:12 -07:00
}
2023-10-27 14:18:50 -07:00
fn vert(&self,vert_id:VERT)->Planar64Vec3;
2023-11-01 15:08:12 -07:00
fn face_nd(&self,face_id:FACE)->(Planar64Vec3,Planar64);
2023-10-27 15:10:38 -07:00
fn face_edges(&self,face_id:FACE)->Cow<Vec<(EDGE,FACE)>>;
2023-10-27 16:04:44 -07:00
fn edge_faces(&self,edge_id:EDGE)->Cow<[FACE;2]>;
fn edge_verts(&self,edge_id:EDGE)->Cow<[VERT;2]>;
fn vert_edges(&self,vert_id:VERT)->Cow<Vec<EDGE>>;
2023-11-10 14:02:17 -08:00
fn vert_faces(&self,vert_id:VERT)->Cow<Vec<FACE>>;
2023-10-27 14:18:50 -07:00
}
impl PhysicsMesh{
pub fn verts<'a>(&'a self)->impl Iterator<Item=Planar64Vec3>+'a{
self.verts.iter().map(|Vert(pos)|*pos)
}
2023-11-02 16:44:14 -07:00
pub fn brute(&self,body:&crate::physics::Body,time_limit:crate::integer::Time)->Option<(FaceId,crate::integer::Time)>{
2023-11-01 17:26:50 -07:00
//check each face
let mut best_time=time_limit;
let mut best_face=None;
2023-11-02 16:44:14 -07:00
for (i,face) in self.faces.iter().enumerate(){
let face_id=FaceId(i);
let (n,d)=face.nd();
2023-11-01 17:26:50 -07:00
for t in crate::zeroes::zeroes2((n.dot(body.position)-d)*2,n.dot(body.velocity)*2,n.dot(body.acceleration)){
let t=body.time+crate::integer::Time::from(t);
if body.time<t&&t<best_time&&n.dot(body.extrapolated_velocity(t))<Planar64::ZERO{
2023-11-02 16:44:14 -07:00
let p=body.extrapolated_position(t);
if self.face_edges(face_id).iter().all(|&(_,face_id)|{
2023-11-01 17:26:50 -07:00
let (n,d)=self.face_nd(face_id);
n.dot(p)<=d
}){
best_time=t;
2023-11-02 16:44:14 -07:00
best_face=Some(face_id);
2023-11-01 17:26:50 -07:00
}
}
}
}
best_face.map(|f|(f,best_time))
}
2023-11-10 14:02:39 -08:00
fn vert_directed_edges(&self,vert_id:VertId)->Cow<Vec<DirectedEdgeId>>{
Cow::Borrowed(&self.vert_topology[vert_id.0].edges)
}
fn directed_edge_n(&self,directed_edge_id:DirectedEdgeId)->Planar64Vec3{
let verts=self.edge_verts(directed_edge_id.as_edge_id());
(self.vert(verts[1].clone())-self.vert(verts[0].clone()))*(directed_edge_id.signum() as i64)
}
}
2023-10-27 14:18:50 -07:00
impl MeshQuery<FaceId,EdgeId,VertId> for PhysicsMesh{
fn closest_fev(&self,point:Planar64Vec3)->FEV<FaceId,EdgeId,VertId>{
2023-11-01 15:52:19 -07:00
//TODO: put some genius code right here
2023-10-29 00:16:07 -07:00
2023-11-01 15:52:19 -07:00
//brute force for now
2023-10-29 00:16:07 -07:00
let mut best_distance_squared=Planar64::MAX;
2023-11-01 15:52:19 -07:00
//make something up as default ret
//hopefully empty meshes don't make their way through here
let mut best_fev=FEV::<FaceId,EdgeId,VertId>::Vert(VertId(0));
2023-10-29 00:16:07 -07:00
//check each vert
for (i,v) in self.verts.iter().enumerate(){
let d=(v.0-point).dot(v.0-point);
if d<best_distance_squared{
best_distance_squared=d;
best_fev=FEV::<FaceId,EdgeId,VertId>::Vert(VertId(i));
}
}
//check each edge
2023-11-01 15:52:19 -07:00
for (i,e) in self.edge_topology.iter().enumerate(){
let v0=self.vert(e.verts[0]);
let v1=self.vert(e.verts[1]);
let n=v1-v0;
//n.cross(point-v0)=sin(t)*n*dis
let d=n.dot(point-v0);
if d<n.dot(v1)&&n.dot(v0)<d{
let c=n.cross(point-v0);
let edge_distance_squared=c.dot(c)/n.dot(n);
if edge_distance_squared<best_distance_squared{
best_distance_squared=edge_distance_squared;
best_fev=FEV::<FaceId,EdgeId,VertId>::Edge(EdgeId(i));
}
2023-10-29 00:16:07 -07:00
}
}
2023-11-01 15:52:19 -07:00
let face_dots:Vec<Planar64>=self.faces.iter().map(|f|f.normal.dot(point)).collect();
2023-10-29 00:16:07 -07:00
//check each face
2023-11-01 15:52:19 -07:00
for (i,f) in self.face_topology.iter().enumerate(){
if face_dots[i]<best_distance_squared&&f.edges.iter().all(|&(_,face_id)|face_dots[face_id.0]<=Planar64::ZERO){
best_distance_squared=face_dots[i];
best_fev=FEV::<FaceId,EdgeId,VertId>::Face(FaceId(i));
}
}
best_fev
2023-10-27 14:18:50 -07:00
}
fn face_nd(&self,face_id:FaceId)->(Planar64Vec3,Planar64){
2023-10-26 19:38:34 -07:00
(self.faces[face_id.0].normal,self.faces[face_id.0].dot)
2023-10-26 16:43:58 -07:00
}
//ideally I never calculate the vertex position, but I have to for the graphical meshes...
2023-10-27 14:18:50 -07:00
fn vert(&self,vert_id:VertId)->Planar64Vec3{
2023-10-29 00:15:51 -07:00
self.verts[vert_id.0].0
2023-10-27 14:18:50 -07:00
}
2023-10-27 15:10:38 -07:00
fn face_edges(&self,face_id:FaceId)->Cow<Vec<(EdgeId,FaceId)>>{
Cow::Borrowed(&self.face_topology[face_id.0].edges)
2023-10-26 16:18:32 -07:00
}
2023-10-27 16:04:44 -07:00
fn edge_faces(&self,edge_id:EdgeId)->Cow<[FaceId;2]>{
2023-10-27 15:10:38 -07:00
Cow::Borrowed(&self.edge_topology[edge_id.0].faces)
2023-10-26 16:18:32 -07:00
}
fn edge_verts(&self,edge_id:EdgeId)->Cow<[VertId;2]>{
2023-10-27 15:10:38 -07:00
Cow::Borrowed(&self.edge_topology[edge_id.0].verts)
2023-10-26 16:18:32 -07:00
}
fn vert_edges(&self,vert_id:VertId)->Cow<Vec<EdgeId>>{
2023-11-10 14:02:39 -08:00
//not poggers
Cow::Owned(self.vert_topology[vert_id.0].edges.iter().map(|directed_edge_id|directed_edge_id.as_edge_id()).collect())
2023-10-26 16:18:32 -07:00
}
2023-11-10 14:02:17 -08:00
fn vert_faces(&self,vert_id:VertId)->Cow<Vec<FaceId>>{
Cow::Borrowed(&self.vert_topology[vert_id.0].faces)
}
2023-10-26 16:18:32 -07:00
}
pub struct TransformedMesh<'a>{
2023-10-30 22:11:54 -07:00
mesh:&'a PhysicsMesh,
transform:&'a crate::integer::Planar64Affine3,
normal_transform:&'a crate::integer::Planar64Mat3,
}
2023-11-02 16:44:14 -07:00
impl TransformedMesh<'_>{
2023-11-03 18:57:49 -07:00
pub fn new<'a>(
mesh:&'a PhysicsMesh,
transform:&'a crate::integer::Planar64Affine3,
normal_transform:&'a crate::integer::Planar64Mat3,
)->TransformedMesh<'a>{
TransformedMesh{
mesh,
transform,
normal_transform,
}
}
2023-11-02 16:44:14 -07:00
pub fn brute_in(&self,body:&crate::physics::Body,time_limit:crate::integer::Time)->Option<(FaceId,crate::integer::Time)>{
//check each face
let mut best_time=time_limit;
let mut best_face=None;
for i in 0..self.mesh.faces.len(){
let face_id=FaceId(i);
let (n,d)=self.face_nd(face_id);
for t in crate::zeroes::zeroes2((n.dot(body.position)-d)*2,n.dot(body.velocity)*2,n.dot(body.acceleration)){
let t=body.time+crate::integer::Time::from(t);
if body.time<t&&t<best_time&&n.dot(body.extrapolated_velocity(t))<Planar64::ZERO{
let p=body.extrapolated_position(t);
if self.face_edges(face_id).iter().all(|&(_,face_id)|{
let (n,d)=self.face_nd(face_id);
n.dot(p)<=d
}){
best_time=t;
best_face=Some(face_id);
}
}
}
}
best_face.map(|f|(f,best_time))
}
pub fn brute_out(&self,body:&crate::physics::Body,time_limit:crate::integer::Time)->Option<(FaceId,crate::integer::Time)>{
//check each face
let mut best_time=time_limit;
let mut best_face=None;
for i in 0..self.mesh.faces.len(){
let face_id=FaceId(i);
let (n,d)=self.face_nd(face_id);
for t in crate::zeroes::zeroes2((n.dot(body.position)-d)*2,n.dot(body.velocity)*2,n.dot(body.acceleration)){
let t=body.time+crate::integer::Time::from(t);
if body.time<t&&t<best_time&&n.dot(body.extrapolated_velocity(t))>Planar64::ZERO{
let p=body.extrapolated_position(t);
2023-11-08 19:46:00 -08:00
if self.face_edges(face_id).iter().all(|&(_,test_face_id)|{
let (n,d)=self.face_nd(test_face_id);
2023-11-02 16:44:14 -07:00
n.dot(p)<=d
}){
best_time=t;
best_face=Some(face_id);
}
}
}
}
best_face.map(|f|(f,best_time))
}
pub fn brute_out_face(&self,body:&crate::physics::Body,time_limit:crate::integer::Time,face_id:FaceId)->Option<(FaceId,crate::integer::Time)>{
//check each face
let mut best_time=time_limit;
let mut best_face=None;
2023-11-08 19:46:00 -08:00
for &(_,test_face_id) in self.mesh.face_edges(face_id).iter(){
let (n,d)=self.face_nd(test_face_id);
2023-11-02 16:44:14 -07:00
for t in crate::zeroes::zeroes2((n.dot(body.position)-d)*2,n.dot(body.velocity)*2,n.dot(body.acceleration)){
let t=body.time+crate::integer::Time::from(t);
if body.time<t&&t<best_time&&n.dot(body.extrapolated_velocity(t))>Planar64::ZERO{
best_time=t;
2023-11-08 19:46:00 -08:00
best_face=Some(test_face_id);
2023-11-02 16:44:14 -07:00
}
}
}
best_face.map(|f|(f,best_time))
}
2023-11-10 14:02:39 -08:00
#[inline]
fn vert_directed_edges(&self,vert_id:VertId)->Cow<Vec<DirectedEdgeId>>{
self.mesh.vert_directed_edges(vert_id)
}
#[inline]
fn directed_edge_n(&self,directed_edge_id:DirectedEdgeId)->Planar64Vec3{
self.mesh.directed_edge_n(directed_edge_id)
}
2023-11-02 16:44:14 -07:00
}
impl MeshQuery<FaceId,EdgeId,VertId> for TransformedMesh<'_>{
2023-10-30 22:11:54 -07:00
fn closest_fev(&self,point:Planar64Vec3)->FEV<FaceId,EdgeId,VertId>{
2023-11-01 16:04:48 -07:00
//TODO: put some genius code right here
//brute force for now
let mut best_distance_squared=Planar64::MAX;
//make something up as default ret
//hopefully empty meshes don't make their way through here
let mut best_fev=FEV::<FaceId,EdgeId,VertId>::Vert(VertId(0));
//check each vert
for i in 0..self.mesh.verts.len(){
let v=self.vert(VertId(i));
let d=(v-point).dot(v-point);
if d<best_distance_squared{
best_distance_squared=d;
best_fev=FEV::<FaceId,EdgeId,VertId>::Vert(VertId(i));
}
}
//check each edge
for (i,e) in self.mesh.edge_topology.iter().enumerate(){
let v0=self.vert(e.verts[0]);
let v1=self.vert(e.verts[1]);
let n=v1-v0;
//n.cross(point-v0)=sin(t)*n*dis
let d=n.dot(point-v0);
if d<n.dot(v1)&&n.dot(v0)<d{
let c=n.cross(point-v0);
let edge_distance_squared=c.dot(c)/n.dot(n);
if edge_distance_squared<best_distance_squared{
best_distance_squared=edge_distance_squared;
best_fev=FEV::<FaceId,EdgeId,VertId>::Edge(EdgeId(i));
}
}
}
let face_dots:Vec<Planar64>=self.mesh.faces.iter().map(|f|(*self.normal_transform*f.normal).dot(point)).collect();
//check each face
for (i,f) in self.mesh.face_topology.iter().enumerate(){
if face_dots[i]<best_distance_squared&&f.edges.iter().all(|&(_,face_id)|face_dots[face_id.0]<=Planar64::ZERO){
best_distance_squared=face_dots[i];
best_fev=FEV::<FaceId,EdgeId,VertId>::Face(FaceId(i));
}
}
best_fev
2023-10-30 22:11:54 -07:00
}
fn face_nd(&self,face_id:FaceId)->(Planar64Vec3,Planar64){
let (n,d)=self.mesh.face_nd(face_id);
2023-11-06 15:57:24 -08:00
let transformed_n=*self.normal_transform*n;
2023-11-09 17:27:07 -08:00
let transformed_d=Planar64::raw(((transformed_n.dot128(self.transform.matrix3*(n*d))<<32)/n.dot128(n)) as i64)+transformed_n.dot(self.transform.translation);
2023-11-06 15:57:24 -08:00
(transformed_n,transformed_d)
2023-10-30 22:11:54 -07:00
}
fn vert(&self,vert_id:VertId)->Planar64Vec3{
self.transform.transform_point3(self.mesh.vert(vert_id))
}
#[inline]
fn face_edges(&self,face_id:FaceId)->Cow<Vec<(EdgeId,FaceId)>>{
self.mesh.face_edges(face_id)
}
#[inline]
fn edge_faces(&self,edge_id:EdgeId)->Cow<[FaceId;2]>{
self.mesh.edge_faces(edge_id)
}
#[inline]
fn edge_verts(&self,edge_id:EdgeId)->Cow<[VertId;2]>{
2023-10-30 22:11:54 -07:00
self.mesh.edge_verts(edge_id)
}
#[inline]
fn vert_edges(&self,vert_id:VertId)->Cow<Vec<EdgeId>>{
2023-10-30 22:11:54 -07:00
self.mesh.vert_edges(vert_id)
}
2023-11-10 14:02:17 -08:00
#[inline]
fn vert_faces(&self,vert_id:VertId)->Cow<Vec<FaceId>>{
self.mesh.vert_faces(vert_id)
}
2023-10-30 22:11:54 -07:00
}
2023-10-26 16:18:32 -07:00
//Note that a face on a minkowski mesh refers to a pair of fevs on the meshes it's summed from
//(face,vertex)
//(edge,edge)
//(vertex,face)
2023-10-27 14:18:50 -07:00
#[derive(Clone,Copy)]
enum MinkowskiVert{
VertVert(VertId,VertId),
}
#[derive(Clone,Copy)]
enum MinkowskiEdge{
VertEdge(VertId,EdgeId),
EdgeVert(EdgeId,VertId),
}
2023-10-30 22:11:54 -07:00
#[derive(Debug,Clone,Copy,Hash,Eq,PartialEq)]
pub enum MinkowskiFace{
2023-10-27 14:18:50 -07:00
VertFace(VertId,FaceId),
2023-11-15 16:05:42 -08:00
EdgeEdge(EdgeId,EdgeId),
FaceVert(FaceId,VertId),
2023-10-27 14:18:50 -07:00
}
2023-10-26 16:18:32 -07:00
2023-10-27 14:18:50 -07:00
pub struct MinkowskiMesh<'a>{
2023-10-31 19:33:16 -07:00
mesh0:&'a TransformedMesh<'a>,
mesh1:&'a TransformedMesh<'a>,
2023-10-26 16:18:32 -07:00
}
2023-10-27 14:18:50 -07:00
impl MinkowskiMesh<'_>{
2023-10-31 19:33:16 -07:00
pub fn minkowski_sum<'a>(mesh0:&'a TransformedMesh,mesh1:&'a TransformedMesh)->MinkowskiMesh<'a>{
2023-10-27 14:18:50 -07:00
MinkowskiMesh{
2023-10-26 16:18:32 -07:00
mesh0,
mesh1,
}
}
2023-10-27 14:18:50 -07:00
}
impl MeshQuery<MinkowskiFace,MinkowskiEdge,MinkowskiVert> for MinkowskiMesh<'_>{
fn closest_fev(&self,point:Planar64Vec3)->FEV<MinkowskiFace,MinkowskiEdge,MinkowskiVert>{
2023-10-26 16:18:32 -07:00
//put some genius code right here
todo!()
}
2023-10-27 14:18:50 -07:00
fn face_nd(&self,face_id:MinkowskiFace)->(Planar64Vec3,Planar64){
2023-10-27 14:19:17 -07:00
match face_id{
2023-11-15 16:05:42 -08:00
MinkowskiFace::VertFace(v0,f1)=>{
let (n,d)=self.mesh1.face_nd(f1);
(-n,d-n.dot(self.mesh0.vert(v0)))
2023-10-27 14:19:17 -07:00
},
MinkowskiFace::EdgeEdge(e0,e1)=>{
let edge0_n=self.mesh0.edge_n(e0);
let edge1_n=self.mesh1.edge_n(e1);
let &[e0v0,e0v1]=self.mesh0.edge_verts(e0).borrow();
let &[e1v0,e1v1]=self.mesh1.edge_verts(e1).borrow();
let n=edge0_n.cross(edge1_n);
let e0d=n.dot(self.mesh0.vert(e0v0)+self.mesh0.vert(e0v1));
let e1d=n.dot(self.mesh0.vert(e1v0)+self.mesh0.vert(e1v1));
let sign=e0d.signum_i64();
(n*(sign*2),(e0d-e1d)*sign)
2023-10-27 14:19:17 -07:00
},
2023-11-15 16:05:42 -08:00
MinkowskiFace::FaceVert(f0,v1)=>{
let (n,d)=self.mesh0.face_nd(f0);
(n,d+n.dot(self.mesh1.vert(v1)))
2023-10-27 14:19:17 -07:00
},
}
2023-10-26 20:52:28 -07:00
}
2023-10-27 14:18:50 -07:00
fn vert(&self,vert_id:MinkowskiVert)->Planar64Vec3{
2023-10-27 15:09:53 -07:00
match vert_id{
MinkowskiVert::VertVert(v0,v1)=>{
self.mesh0.vert(v0)-self.mesh1.vert(v1)
},
}
2023-10-27 14:18:50 -07:00
}
2023-10-27 15:10:38 -07:00
fn face_edges(&self,face_id:MinkowskiFace)->Cow<Vec<(MinkowskiEdge,MinkowskiFace)>>{
2023-10-27 15:09:53 -07:00
match face_id{
2023-11-15 16:05:42 -08:00
MinkowskiFace::VertFace(v0,f1)=>{
let face1_n=self.mesh1.face_nd(f1).0;
Cow::Owned(self.mesh1.face_edges(f1).iter().map(|&(edge_id1,edge_face_id1)|{
//same as above
(MinkowskiEdge::VertEdge(v0,edge_id1),{
let mut best_edge=None;
let mut best_d=Planar64::MAX;
2023-11-15 16:05:42 -08:00
let edge_face1_n=self.mesh1.face_nd(edge_face_id1).0;
let v0e=self.mesh0.vert_directed_edges(v0);
for &directed_edge_id0 in v0e.iter(){
let edge0_n=self.mesh0.directed_edge_n(directed_edge_id0);
if edge_face1_n.dot(edge0_n)<Planar64::ZERO{
let d=face1_n.dot(edge0_n);
if d<best_d{
best_d=d;
2023-11-15 19:45:38 -08:00
best_edge=Some(directed_edge_id0);
}
}
}
best_edge.map_or(
2023-11-15 16:05:42 -08:00
MinkowskiFace::VertFace(v0,edge_face_id1),
|directed_edge_id0|MinkowskiFace::EdgeEdge(directed_edge_id0.as_edge_id(),edge_id1)
)
})
2023-10-27 15:09:53 -07:00
}).collect())
},
MinkowskiFace::EdgeEdge(e0,e1)=>{
let e0v=self.mesh0.edge_verts(e0);
let e0f=self.mesh0.edge_faces(e0);
let edge0_n=self.mesh0.edge_n(e0);
2023-10-27 15:09:53 -07:00
let e1v=self.mesh1.edge_verts(e1);
let e1f=self.mesh1.edge_faces(e1);
let edge1_n=self.mesh1.edge_n(e1);
//populate algorithm variables based on known parity
//I don't like that this arbitrarily picks a face to test against!
let (e0v0_face_id1,e0v1_face_id1)=if edge0_n.dot(self.mesh1.face_nd(e1f[0]).0)<=Planar64::ZERO{
(e1f[1],e1f[0])
}else{
(e1f[0],e1f[1])
};
let [r0,r1]=[(e0v[0],e0v0_face_id1),(e0v[1],e0v1_face_id1)].map(|(vert_id0,edge_face_id1)|{
(MinkowskiEdge::VertEdge(vert_id0,e1),{
let mut best_edge=None;
let mut best_d=Planar64::MAX;
let edge_face1_n=self.mesh1.face_nd(edge_face_id1).0;
let v0e=self.mesh0.vert_directed_edges(vert_id0);
for &directed_edge_id0 in v0e.iter(){
let edge0_n=self.mesh0.directed_edge_n(directed_edge_id0);
if edge_face1_n.dot(edge0_n)<Planar64::ZERO{
//I think face_id1 == edge_face_id1 in this case
let d=edge_face1_n.dot(edge0_n);
if d<best_d{
best_d=d;
2023-11-15 19:45:38 -08:00
best_edge=Some(directed_edge_id0);
}
}
}
best_edge.map_or(
MinkowskiFace::VertFace(vert_id0,edge_face_id1),
|directed_edge_id0|MinkowskiFace::EdgeEdge(directed_edge_id0.as_edge_id(),e1)
)
})
2023-10-27 15:09:53 -07:00
});
let (e1v0_face_id0,e1v1_face_id0)=if edge1_n.dot(self.mesh0.face_nd(e0f[0]).0)<=Planar64::ZERO{
(e0f[1],e0f[0])
}else{
(e0f[0],e0f[1])
};
let [r2,r3]=[(e1v0_face_id0,e1v[0]),(e1v1_face_id0,e1v[1])].map(|(edge_face_id0,vert_id1)|{
(MinkowskiEdge::EdgeVert(e0,vert_id1),{
let mut best_edge=None;
let mut best_d=Planar64::MAX;
let edge_face0_n=self.mesh0.face_nd(edge_face_id0).0;
let v1e=self.mesh1.vert_directed_edges(vert_id1);
for &directed_edge_id1 in v1e.iter(){
let edge1_n=self.mesh1.directed_edge_n(directed_edge_id1);
if edge_face0_n.dot(edge1_n)<Planar64::ZERO{
let d=edge_face0_n.dot(edge1_n);
if d<best_d{
best_d=d;
2023-11-15 19:45:38 -08:00
best_edge=Some(directed_edge_id1);
}
}
}
best_edge.map_or(
MinkowskiFace::FaceVert(edge_face_id0,vert_id1),
|directed_edge_id1|MinkowskiFace::EdgeEdge(e0,directed_edge_id1.as_edge_id())
)
})
2023-10-27 15:09:53 -07:00
});
//could sort this if ordered edges are needed
2023-10-27 15:09:53 -07:00
Cow::Owned(vec![r0,r1,r2,r3])
},
2023-11-15 16:05:42 -08:00
MinkowskiFace::FaceVert(f0,v1)=>{
let face0_n=self.mesh0.face_nd(f0).0;
Cow::Owned(self.mesh0.face_edges(f0).iter().map(|&(edge_id0,edge_face_id0)|{
//compare v1 edges
//candidate edges have negative dot with edge_face_id0 normal
//choose the edge with the smallest edgedir dot with f0 normal
//MinkowskiFace::EdgeEdge(edge_id0,edge_id1)
//if there is no candidate edges
//MinkowskiFace::FaceVert(edge_face_id0,v1)
(MinkowskiEdge::EdgeVert(edge_id0,v1),{
let mut best_edge=None;
let mut best_d=Planar64::MAX;
2023-11-15 16:05:42 -08:00
let edge_face0_n=self.mesh0.face_nd(edge_face_id0).0;
let v1e=self.mesh1.vert_directed_edges(v1);
for &directed_edge_id1 in v1e.iter(){
let edge1_n=self.mesh1.directed_edge_n(directed_edge_id1);
if edge_face0_n.dot(edge1_n)<Planar64::ZERO{
let d=face0_n.dot(edge1_n);
if d<best_d{
best_d=d;
2023-11-15 19:45:38 -08:00
best_edge=Some(directed_edge_id1);
}
}
}
best_edge.map_or(
2023-11-15 16:05:42 -08:00
MinkowskiFace::FaceVert(edge_face_id0,v1),
|directed_edge_id1|MinkowskiFace::EdgeEdge(edge_id0,directed_edge_id1.as_edge_id())
)
})
2023-10-27 15:09:53 -07:00
}).collect())
},
}
2023-10-26 20:52:28 -07:00
}
2023-10-27 16:04:44 -07:00
fn edge_faces(&self,edge_id:MinkowskiEdge)->Cow<[MinkowskiFace;2]>{
2023-10-27 16:23:35 -07:00
match edge_id{
MinkowskiEdge::VertEdge(v0,e1)=>{
2023-11-15 17:59:47 -08:00
let e1f=self.mesh1.edge_faces(e1);
Cow::Owned([(e1f[0],e1f[1]),(e1f[1],e1f[0])].map(|(edge_face_id1,other_edge_face_id1)|{
let mut best_edge=None;
let mut best_d=Planar64::MAX;
let edge_face1_n=self.mesh1.face_nd(edge_face_id1).0;
let other_edge_face1_n=self.mesh1.face_nd(other_edge_face_id1).0;
let v0e=self.mesh0.vert_directed_edges(v0);
for &directed_edge_id0 in v0e.iter(){
let edge0_n=self.mesh0.directed_edge_n(directed_edge_id0);
if edge_face1_n.dot(edge0_n)<Planar64::ZERO{
let d=other_edge_face1_n.dot(edge0_n);
if d<best_d{
best_d=d;
2023-11-15 19:45:38 -08:00
best_edge=Some(directed_edge_id0);
2023-11-15 17:59:47 -08:00
}
}
}
best_edge.map_or(
MinkowskiFace::VertFace(v0,edge_face_id1),
|directed_edge_id0|MinkowskiFace::EdgeEdge(directed_edge_id0.as_edge_id(),e1)
)
2023-10-27 16:23:35 -07:00
}))
},
MinkowskiEdge::EdgeVert(e0,v1)=>{
2023-11-15 17:59:47 -08:00
let e0f=self.mesh0.edge_faces(e0);
Cow::Owned([(e0f[0],e0f[1]),(e0f[1],e0f[0])].map(|(edge_face_id0,other_edge_face_id0)|{
let mut best_edge=None;
let mut best_d=Planar64::MAX;
let edge_face0_n=self.mesh0.face_nd(edge_face_id0).0;
let other_edge_face0_n=self.mesh0.face_nd(other_edge_face_id0).0;
let v1e=self.mesh1.vert_directed_edges(v1);
for &directed_edge_id1 in v1e.iter(){
let edge1_n=self.mesh1.directed_edge_n(directed_edge_id1);
if edge_face0_n.dot(edge1_n)<Planar64::ZERO{
let d=other_edge_face0_n.dot(edge1_n);
if d<best_d{
best_d=d;
2023-11-15 19:45:38 -08:00
best_edge=Some(directed_edge_id1);
2023-11-15 17:59:47 -08:00
}
}
}
best_edge.map_or(
MinkowskiFace::FaceVert(edge_face_id0,v1),
|directed_edge_id1|MinkowskiFace::EdgeEdge(e0,directed_edge_id1.as_edge_id())
)
2023-10-27 16:23:35 -07:00
}))
},
}
2023-10-26 20:52:28 -07:00
}
fn edge_verts(&self,edge_id:MinkowskiEdge)->Cow<[MinkowskiVert;2]>{
2023-10-27 16:27:57 -07:00
match edge_id{
MinkowskiEdge::VertEdge(v0,e1)=>{
2023-11-01 18:03:11 -07:00
Cow::Owned(self.mesh1.edge_verts(e1).map(|vert_id1|{
MinkowskiVert::VertVert(v0,vert_id1)
2023-10-27 16:27:57 -07:00
}))
},
MinkowskiEdge::EdgeVert(e0,v1)=>{
2023-11-01 18:03:11 -07:00
Cow::Owned(self.mesh0.edge_verts(e0).map(|vert_id0|{
MinkowskiVert::VertVert(vert_id0,v1)
2023-10-27 16:27:57 -07:00
}))
},
}
2023-10-26 20:52:28 -07:00
}
fn vert_edges(&self,vert_id:MinkowskiVert)->Cow<Vec<MinkowskiEdge>>{
2023-10-27 16:44:00 -07:00
match vert_id{
MinkowskiVert::VertVert(v0,v1)=>{
2023-11-10 14:02:39 -08:00
let mut edges=Vec::new();
let v0e=self.mesh0.vert_directed_edges(v0);
let v1f=self.mesh1.vert_faces(v1);
for &directed_edge_id in v0e.iter(){
let n=self.mesh0.directed_edge_n(directed_edge_id);
if v1f.iter().all(|&face_id|n.dot(self.mesh1.face_nd(face_id).0)<Planar64::ZERO){
edges.push(MinkowskiEdge::EdgeVert(directed_edge_id.as_edge_id(),v1));
}
}
let v1e=self.mesh1.vert_directed_edges(v1);
let v0f=self.mesh0.vert_faces(v0);
for &directed_edge_id in v1e.iter(){
let n=self.mesh1.directed_edge_n(directed_edge_id);
if v0f.iter().all(|&face_id|n.dot(self.mesh0.face_nd(face_id).0)<Planar64::ZERO){
edges.push(MinkowskiEdge::VertEdge(v0,directed_edge_id.as_edge_id()));
}
}
Cow::Owned(edges)
2023-10-27 16:44:00 -07:00
},
}
2023-10-26 20:52:28 -07:00
}
2023-11-10 14:02:17 -08:00
fn vert_faces(&self,vert_id:MinkowskiVert)->Cow<Vec<MinkowskiFace>>{
todo!()
}
2023-10-26 16:18:32 -07:00
}
2023-11-06 15:58:22 -08:00
#[test]
fn build_me_a_cube(){
let unit_cube=crate::primitives::unit_cube();
let mesh=PhysicsMesh::from(&unit_cube);
println!("mesh={:?}",mesh);
}