use crate::extrusion::apply_transform;
use crate::{Error, Mesh, Result, Vector3};
use ifc_lite_core::{DecodedEntity, EntityDecoder, IfcSchema, IfcType};
use nalgebra::Point3;
use super::boolean::BooleanClippingProcessor;
use super::helpers::parse_axis2_placement_3d;
use crate::router::GeometryProcessor;
pub struct BlockProcessor;
impl BlockProcessor {
pub fn new() -> Self {
Self
}
}
impl Default for BlockProcessor {
fn default() -> Self {
Self::new()
}
}
impl GeometryProcessor for BlockProcessor {
fn process(
&self,
entity: &DecodedEntity,
decoder: &mut EntityDecoder,
_schema: &IfcSchema,
) -> Result<Mesh> {
let x = entity
.get_float(1)
.ok_or_else(|| Error::geometry("IfcBlock missing XLength".to_string()))?;
let y = entity
.get_float(2)
.ok_or_else(|| Error::geometry("IfcBlock missing YLength".to_string()))?;
let z = entity
.get_float(3)
.ok_or_else(|| Error::geometry("IfcBlock missing ZLength".to_string()))?;
if !(x.is_finite() && y.is_finite() && z.is_finite() && x > 0.0 && y > 0.0 && z > 0.0) {
return Err(Error::geometry(format!(
"IfcBlock requires finite positive lengths, got ({}, {}, {})",
x, y, z
)));
}
let mut mesh = build_axis_aligned_box(x, y, z);
if let Some(pos_attr) = entity.get(0) {
if !pos_attr.is_null() {
if let Some(pos_entity) = decoder.resolve_ref(pos_attr)? {
if pos_entity.ifc_type == IfcType::IfcAxis2Placement3D {
let transform = parse_axis2_placement_3d(&pos_entity, decoder)?;
apply_transform(&mut mesh, &transform);
}
}
}
}
Ok(mesh)
}
fn supported_types(&self) -> Vec<IfcType> {
vec![IfcType::IfcBlock]
}
}
pub struct CsgSolidProcessor;
impl CsgSolidProcessor {
pub fn new() -> Self {
Self
}
}
impl Default for CsgSolidProcessor {
fn default() -> Self {
Self::new()
}
}
impl GeometryProcessor for CsgSolidProcessor {
fn process(
&self,
entity: &DecodedEntity,
decoder: &mut EntityDecoder,
schema: &IfcSchema,
) -> Result<Mesh> {
let root_attr = entity.get(0).ok_or_else(|| {
Error::geometry("IfcCsgSolid missing TreeRootExpression".to_string())
})?;
let root = decoder.resolve_ref(root_attr)?.ok_or_else(|| {
Error::geometry("IfcCsgSolid TreeRootExpression unresolved".to_string())
})?;
match root.ifc_type {
IfcType::IfcBooleanResult | IfcType::IfcBooleanClippingResult => {
BooleanClippingProcessor::new().process(&root, decoder, schema)
}
IfcType::IfcBlock => BlockProcessor::new().process(&root, decoder, schema),
IfcType::IfcCsgSolid => Err(Error::geometry(
"IfcCsgSolid TreeRootExpression must be IfcBooleanResult or \
IfcCsgPrimitive3D, not another IfcCsgSolid (spec violation)"
.to_string(),
)),
other => Err(Error::geometry(format!(
"Unsupported IfcCsgSolid TreeRootExpression: {}",
other
))),
}
}
fn supported_types(&self) -> Vec<IfcType> {
vec![IfcType::IfcCsgSolid]
}
}
fn build_axis_aligned_box(x: f64, y: f64, z: f64) -> Mesh {
let mut mesh = Mesh::with_capacity(24, 36);
let faces: [([Point3<f64>; 4], Vector3<f64>); 6] = [
(
[
Point3::new(0.0, 0.0, 0.0),
Point3::new(0.0, y, 0.0),
Point3::new(x, y, 0.0),
Point3::new(x, 0.0, 0.0),
],
Vector3::new(0.0, 0.0, -1.0),
),
(
[
Point3::new(0.0, 0.0, z),
Point3::new(x, 0.0, z),
Point3::new(x, y, z),
Point3::new(0.0, y, z),
],
Vector3::new(0.0, 0.0, 1.0),
),
(
[
Point3::new(0.0, 0.0, 0.0),
Point3::new(x, 0.0, 0.0),
Point3::new(x, 0.0, z),
Point3::new(0.0, 0.0, z),
],
Vector3::new(0.0, -1.0, 0.0),
),
(
[
Point3::new(x, y, 0.0),
Point3::new(0.0, y, 0.0),
Point3::new(0.0, y, z),
Point3::new(x, y, z),
],
Vector3::new(0.0, 1.0, 0.0),
),
(
[
Point3::new(0.0, y, 0.0),
Point3::new(0.0, 0.0, 0.0),
Point3::new(0.0, 0.0, z),
Point3::new(0.0, y, z),
],
Vector3::new(-1.0, 0.0, 0.0),
),
(
[
Point3::new(x, 0.0, 0.0),
Point3::new(x, y, 0.0),
Point3::new(x, y, z),
Point3::new(x, 0.0, z),
],
Vector3::new(1.0, 0.0, 0.0),
),
];
for (corners, normal) in faces {
let base = (mesh.positions.len() / 3) as u32;
for p in &corners {
mesh.add_vertex(*p, normal);
}
mesh.add_triangle(base, base + 1, base + 2);
mesh.add_triangle(base, base + 2, base + 3);
}
mesh
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn axis_aligned_box_is_closed_and_has_outward_normals() {
let mesh = build_axis_aligned_box(2.0, 3.0, 4.0);
assert_eq!(mesh.positions.len() / 3, 24);
assert_eq!(mesh.indices.len() / 3, 12);
let mut min = [f32::INFINITY; 3];
let mut max = [f32::NEG_INFINITY; 3];
for chunk in mesh.positions.chunks_exact(3) {
for i in 0..3 {
min[i] = min[i].min(chunk[i]);
max[i] = max[i].max(chunk[i]);
}
}
assert_eq!(min, [0.0, 0.0, 0.0]);
assert_eq!(max, [2.0, 3.0, 4.0]);
let mut faces_seen = [false; 6];
for chunk in mesh.normals.chunks_exact(12) {
let nx = chunk[0];
let ny = chunk[1];
let nz = chunk[2];
let label = match (nx, ny, nz) {
(x, _, _) if x > 0.5 => 0,
(x, _, _) if x < -0.5 => 1,
(_, y, _) if y > 0.5 => 2,
(_, y, _) if y < -0.5 => 3,
(_, _, z) if z > 0.5 => 4,
(_, _, z) if z < -0.5 => 5,
_ => panic!("non-axial normal"),
};
faces_seen[label] = true;
}
assert!(faces_seen.iter().all(|&seen| seen), "missing a face");
}
}