use crate::{profiles::ProfileProcessor, Error, Mesh, Point3, Result, Vector3};
use ifc_lite_core::{DecodedEntity, EntityDecoder, IfcSchema, IfcType};
use crate::router::GeometryProcessor;
fn build_tube_rmf(
curve_points: &[Point3<f64>],
) -> (Vec<Vector3<f64>>, Vec<Vector3<f64>>, Vec<Vector3<f64>>) {
let n = curve_points.len();
let mut tangents = Vec::with_capacity(n);
let mut perp1s = Vec::with_capacity(n);
let mut perp2s = Vec::with_capacity(n);
if n < 2 {
return (tangents, perp1s, perp2s);
}
for i in 0..n {
let t = if i == 0 {
(curve_points[1] - curve_points[0]).normalize()
} else if i == n - 1 {
(curve_points[i] - curve_points[i - 1]).normalize()
} else {
((curve_points[i + 1] - curve_points[i - 1]) / 2.0).normalize()
};
tangents.push(t);
}
let up0 = if tangents[0].x.abs() < 0.9 {
Vector3::new(1.0, 0.0, 0.0)
} else {
Vector3::new(0.0, 1.0, 0.0)
};
let mut perp1 = tangents[0].cross(&up0).normalize();
let mut perp2 = tangents[0].cross(&perp1).normalize();
perp1s.push(perp1);
perp2s.push(perp2);
for i in 1..n {
let prev = tangents[i - 1];
let curr = tangents[i];
let cos_a = prev.dot(&curr).clamp(-1.0, 1.0);
let axis = prev.cross(&curr);
let axis_norm = axis.norm();
if axis_norm > 1e-9 && cos_a < 1.0 - 1e-12 {
let axis = axis / axis_norm;
let sin_a = (1.0 - cos_a * cos_a).max(0.0).sqrt();
perp1 = perp1 * cos_a
+ axis.cross(&perp1) * sin_a
+ axis * axis.dot(&perp1) * (1.0 - cos_a);
perp1 = perp1.normalize();
perp2 = curr.cross(&perp1).normalize();
}
perp1s.push(perp1);
perp2s.push(perp2);
}
(tangents, perp1s, perp2s)
}
pub struct SweptDiskSolidProcessor {
profile_processor: ProfileProcessor,
}
impl SweptDiskSolidProcessor {
pub fn new(schema: IfcSchema) -> Self {
Self {
profile_processor: ProfileProcessor::new(schema),
}
}
}
impl GeometryProcessor for SweptDiskSolidProcessor {
fn process(
&self,
entity: &DecodedEntity,
decoder: &mut EntityDecoder,
_schema: &IfcSchema,
) -> Result<Mesh> {
let directrix_attr = entity
.get(0)
.ok_or_else(|| Error::geometry("SweptDiskSolid missing Directrix".to_string()))?;
let radius = entity
.get_float(1)
.ok_or_else(|| Error::geometry("SweptDiskSolid missing Radius".to_string()))?;
let _inner_radius = entity.get_float(2);
let start_param = entity.get_float(3);
let end_param = entity.get_float(4);
let directrix = decoder
.resolve_ref(directrix_attr)?
.ok_or_else(|| Error::geometry("Failed to resolve Directrix".to_string()))?;
let has_trim = start_param.is_some() || end_param.is_some();
let curve_points = if has_trim
&& directrix.ifc_type.is_subtype_of(IfcType::IfcCompositeCurve)
{
self.profile_processor
.get_composite_curve_points_trimmed(
&directrix,
decoder,
start_param,
end_param,
)?
} else if has_trim && directrix.ifc_type == IfcType::IfcPolyline {
self.profile_processor
.get_polyline_points_trimmed(&directrix, decoder, start_param, end_param)?
} else {
self.profile_processor.get_curve_points(&directrix, decoder)?
};
if curve_points.len() < 2 {
return Ok(Mesh::new()); }
let segments = 24; let mut positions = Vec::new();
let mut indices = Vec::new();
let (_, perp1s, perp2s) = build_tube_rmf(&curve_points);
for i in 0..curve_points.len() {
let p = curve_points[i];
let perp1 = perp1s[i];
let perp2 = perp2s[i];
for j in 0..segments {
let angle = 2.0 * std::f64::consts::PI * j as f64 / segments as f64;
let offset = perp1 * (radius * angle.cos()) + perp2 * (radius * angle.sin());
let vertex = p + offset;
positions.push(vertex.x as f32);
positions.push(vertex.y as f32);
positions.push(vertex.z as f32);
}
if i < curve_points.len() - 1 {
let base = (i * segments) as u32;
let next_base = ((i + 1) * segments) as u32;
for j in 0..segments {
let j_next = (j + 1) % segments;
indices.push(base + j as u32);
indices.push(next_base + j as u32);
indices.push(next_base + j_next as u32);
indices.push(base + j as u32);
indices.push(next_base + j_next as u32);
indices.push(base + j_next as u32);
}
}
}
let center_idx = (positions.len() / 3) as u32;
let start = curve_points[0];
positions.push(start.x as f32);
positions.push(start.y as f32);
positions.push(start.z as f32);
for j in 0..segments {
let j_next = (j + 1) % segments;
indices.push(center_idx);
indices.push(j_next as u32);
indices.push(j as u32);
}
let end_center_idx = (positions.len() / 3) as u32;
let end_base = ((curve_points.len() - 1) * segments) as u32;
let end = curve_points[curve_points.len() - 1];
positions.push(end.x as f32);
positions.push(end.y as f32);
positions.push(end.z as f32);
for j in 0..segments {
let j_next = (j + 1) % segments;
indices.push(end_center_idx);
indices.push(end_base + j as u32);
indices.push(end_base + j_next as u32);
}
Ok(Mesh {
positions,
normals: Vec::new(),
indices,
rtc_applied: false,
})
}
fn supported_types(&self) -> Vec<IfcType> {
vec![IfcType::IfcSweptDiskSolid]
}
}
impl Default for SweptDiskSolidProcessor {
fn default() -> Self {
Self::new(IfcSchema::new())
}
}
pub struct RevolvedAreaSolidProcessor {
profile_processor: ProfileProcessor,
}
impl RevolvedAreaSolidProcessor {
pub fn new(schema: IfcSchema) -> Self {
Self {
profile_processor: ProfileProcessor::new(schema),
}
}
}
impl GeometryProcessor for RevolvedAreaSolidProcessor {
fn process(
&self,
entity: &DecodedEntity,
decoder: &mut EntityDecoder,
_schema: &IfcSchema,
) -> Result<Mesh> {
let profile_attr = entity
.get(0)
.ok_or_else(|| Error::geometry("RevolvedAreaSolid missing SweptArea".to_string()))?;
let profile = decoder
.resolve_ref(profile_attr)?
.ok_or_else(|| Error::geometry("Failed to resolve SweptArea".to_string()))?;
let axis_attr = entity
.get(2)
.ok_or_else(|| Error::geometry("RevolvedAreaSolid missing Axis".to_string()))?;
let axis_placement = decoder
.resolve_ref(axis_attr)?
.ok_or_else(|| Error::geometry("Failed to resolve Axis".to_string()))?;
let angle = entity
.get_float(3)
.ok_or_else(|| Error::geometry("RevolvedAreaSolid missing Angle".to_string()))?;
let profile_2d = self.profile_processor.process(&profile, decoder)?;
if profile_2d.outer.is_empty() {
return Ok(Mesh::new());
}
let axis_location = {
let loc_attr = axis_placement
.get(0)
.ok_or_else(|| Error::geometry("Axis1Placement missing Location".to_string()))?;
let loc = decoder
.resolve_ref(loc_attr)?
.ok_or_else(|| Error::geometry("Failed to resolve axis location".to_string()))?;
let coords = loc
.get(0)
.and_then(|v| v.as_list())
.ok_or_else(|| Error::geometry("Axis location missing coordinates".to_string()))?;
Point3::new(
coords.first().and_then(|v| v.as_float()).unwrap_or(0.0),
coords.get(1).and_then(|v| v.as_float()).unwrap_or(0.0),
coords.get(2).and_then(|v| v.as_float()).unwrap_or(0.0),
)
};
let axis_direction = {
if let Some(dir_attr) = axis_placement.get(1) {
if !dir_attr.is_null() {
let dir = decoder.resolve_ref(dir_attr)?.ok_or_else(|| {
Error::geometry("Failed to resolve axis direction".to_string())
})?;
let coords = dir.get(0).and_then(|v| v.as_list()).ok_or_else(|| {
Error::geometry("Axis direction missing coordinates".to_string())
})?;
Vector3::new(
coords.first().and_then(|v| v.as_float()).unwrap_or(0.0),
coords.get(1).and_then(|v| v.as_float()).unwrap_or(1.0),
coords.get(2).and_then(|v| v.as_float()).unwrap_or(0.0),
)
.normalize()
} else {
Vector3::new(0.0, 1.0, 0.0) }
} else {
Vector3::new(0.0, 1.0, 0.0) }
};
let full_circle = angle.abs() >= std::f64::consts::PI * 1.99;
let segments = if full_circle {
24 } else {
((angle.abs() / std::f64::consts::PI * 12.0).ceil() as usize).max(4)
};
let profile_points = &profile_2d.outer;
let num_profile_points = profile_points.len();
let mut positions = Vec::new();
let mut indices = Vec::new();
for i in 0..=segments {
let t = if full_circle && i == segments {
0.0 } else {
angle * i as f64 / segments as f64
};
let cos_t = t.cos();
let sin_t = t.sin();
let (ax, ay, az) = (axis_direction.x, axis_direction.y, axis_direction.z);
let k_matrix = |v: Vector3<f64>| -> Vector3<f64> {
Vector3::new(
ay * v.z - az * v.y,
az * v.x - ax * v.z,
ax * v.y - ay * v.x,
)
};
for (j, p2d) in profile_points.iter().enumerate() {
let radius = p2d.x;
let height = p2d.y;
let v = Vector3::new(radius, 0.0, 0.0);
let k_cross_v = k_matrix(v);
let k_dot_v = ax * v.x + ay * v.y + az * v.z;
let v_rot =
v * cos_t + k_cross_v * sin_t + axis_direction * k_dot_v * (1.0 - cos_t);
let pos = axis_location + axis_direction * height + v_rot;
positions.push(pos.x as f32);
positions.push(pos.y as f32);
positions.push(pos.z as f32);
if i < segments && j < num_profile_points - 1 {
let current = (i * num_profile_points + j) as u32;
let next_seg = ((i + 1) * num_profile_points + j) as u32;
let current_next = current + 1;
let next_seg_next = next_seg + 1;
indices.push(current);
indices.push(next_seg);
indices.push(next_seg_next);
indices.push(current);
indices.push(next_seg_next);
indices.push(current_next);
}
}
}
if !full_circle {
let start_center_idx = (positions.len() / 3) as u32;
let start_center = axis_location
+ axis_direction
* (profile_points.iter().map(|p| p.y).sum::<f64>()
/ profile_points.len() as f64);
positions.push(start_center.x as f32);
positions.push(start_center.y as f32);
positions.push(start_center.z as f32);
for j in 0..num_profile_points - 1 {
indices.push(start_center_idx);
indices.push(j as u32 + 1);
indices.push(j as u32);
}
let end_center_idx = (positions.len() / 3) as u32;
let end_base = (segments * num_profile_points) as u32;
positions.push(start_center.x as f32);
positions.push(start_center.y as f32);
positions.push(start_center.z as f32);
for j in 0..num_profile_points - 1 {
indices.push(end_center_idx);
indices.push(end_base + j as u32);
indices.push(end_base + j as u32 + 1);
}
}
Ok(Mesh {
positions,
normals: Vec::new(),
indices,
rtc_applied: false,
})
}
fn supported_types(&self) -> Vec<IfcType> {
vec![IfcType::IfcRevolvedAreaSolid]
}
}
impl Default for RevolvedAreaSolidProcessor {
fn default() -> Self {
Self::new(IfcSchema::new())
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn rmf_is_constant_on_a_straight_line() {
let pts = vec![
Point3::new(0.0, 0.0, 0.0),
Point3::new(1.0, 0.0, 0.0),
Point3::new(2.0, 0.0, 0.0),
];
let (tangents, perp1s, perp2s) = build_tube_rmf(&pts);
assert_eq!(tangents.len(), 3);
for i in 1..3 {
assert!((tangents[i] - tangents[0]).norm() < 1e-9);
assert!((perp1s[i] - perp1s[0]).norm() < 1e-9);
assert!((perp2s[i] - perp2s[0]).norm() < 1e-9);
}
}
#[test]
fn rmf_does_not_flip_at_sharp_bends() {
let pts = vec![
Point3::new(0.0, 0.0, 0.0),
Point3::new(1.0, 0.0, 0.0),
Point3::new(1.0, 1.0, 0.0),
];
let (_, perp1s, _) = build_tube_rmf(&pts);
assert_eq!(perp1s.len(), 3);
for (i, p) in perp1s.iter().enumerate() {
assert!(
p.z > 0.5,
"perp1 at i={i} flipped or rotated out of +Z half-space: {p:?}"
);
}
}
#[test]
fn rmf_handles_degenerate_inputs() {
let empty: Vec<Point3<f64>> = Vec::new();
let (t, p1, p2) = build_tube_rmf(&empty);
assert!(t.is_empty() && p1.is_empty() && p2.is_empty());
let single = vec![Point3::new(0.0, 0.0, 0.0)];
let (t, p1, p2) = build_tube_rmf(&single);
assert!(t.is_empty() && p1.is_empty() && p2.is_empty());
}
}