use anyhow::Result;
use kcmc::ModelingCmd;
use kcmc::each_cmd as mcmd;
use kittycad_modeling_cmds::length_unit::LengthUnit;
use kittycad_modeling_cmds::ok_response::OkModelingCmdResponse;
use kittycad_modeling_cmds::shared::Point3d;
use kittycad_modeling_cmds::websocket::OkWebSocketResponseData;
use kittycad_modeling_cmds::{self as kcmc};
use crate::errors::KclError;
use crate::errors::KclErrorDetails;
use crate::execution::ExecState;
use crate::execution::KclValue;
use crate::execution::ModelingCmdMeta;
use crate::execution::Sketch;
use crate::execution::types::PrimitiveType;
use crate::execution::types::RuntimeType;
use crate::std::Args;
use crate::std::axis_or_reference::Axis2dOrEdgeReference;
pub async fn mirror_2d(exec_state: &mut ExecState, args: Args) -> Result<KclValue, KclError> {
let sketches = args.get_unlabeled_kw_arg("sketches", &RuntimeType::sketches(), exec_state)?;
let axis = args.get_kw_arg(
"axis",
&RuntimeType::Union(vec![
RuntimeType::Primitive(PrimitiveType::Edge),
RuntimeType::Primitive(PrimitiveType::Axis2d),
RuntimeType::segment(),
]),
exec_state,
)?;
let sketches = inner_mirror_2d(sketches, axis, exec_state, args).await?;
Ok(sketches.into())
}
async fn inner_mirror_2d(
sketches: Vec<Sketch>,
axis: Axis2dOrEdgeReference,
exec_state: &mut ExecState,
args: Args,
) -> Result<Vec<Sketch>, KclError> {
let mut starting_sketches = sketches.clone();
if args.ctx.no_engine_commands().await {
for sketch in starting_sketches.iter_mut() {
sketch.is_closed = crate::execution::ProfileClosed::Maybe;
}
return Ok(starting_sketches);
}
match axis {
Axis2dOrEdgeReference::Axis { direction, origin } => {
let resp = exec_state
.send_modeling_cmd(
ModelingCmdMeta::from_args(exec_state, &args),
ModelingCmd::from(
mcmd::EntityMirror::builder()
.ids(starting_sketches.iter().map(|sketch| sketch.id).collect())
.axis(Point3d {
x: direction[0].to_mm(),
y: direction[1].to_mm(),
z: 0.0,
})
.point(Point3d {
x: LengthUnit(origin[0].to_mm()),
y: LengthUnit(origin[1].to_mm()),
z: LengthUnit(0.0),
})
.build(),
),
)
.await?;
if let OkWebSocketResponseData::Modeling {
modeling_response: OkModelingCmdResponse::EntityMirror(mirror_info),
} = &resp
{
let face_edge_info = &mirror_info.entity_face_edge_ids;
starting_sketches
.iter_mut()
.zip(face_edge_info.iter())
.try_for_each(|(sketch, info)| {
sketch.id = info.object_id;
let first_edge = info.edges.first().copied();
match first_edge {
Some(edge) => sketch.mirror = Some(edge),
None => {
return Err(KclError::new_engine(KclErrorDetails::new(
"No edges found in mirror info".to_string(),
vec![args.source_range],
)));
}
}
sketch.is_closed = crate::execution::ProfileClosed::Maybe;
Ok(())
})?;
} else {
return Err(KclError::new_engine(KclErrorDetails::new(
format!("EntityMirror response was not as expected: {resp:?}"),
vec![args.source_range],
)));
};
}
Axis2dOrEdgeReference::Edge(edge) => {
let edge_id = edge.get_engine_id(exec_state, &args)?;
let resp = exec_state
.send_modeling_cmd(
ModelingCmdMeta::from_args(exec_state, &args),
ModelingCmd::from(
mcmd::EntityMirrorAcrossEdge::builder()
.ids(starting_sketches.iter().map(|sketch| sketch.id).collect())
.edge_id(edge_id)
.build(),
),
)
.await?;
if let OkWebSocketResponseData::Modeling {
modeling_response: OkModelingCmdResponse::EntityMirrorAcrossEdge(mirror_info),
} = &resp
{
let face_edge_info = &mirror_info.entity_face_edge_ids;
starting_sketches
.iter_mut()
.zip(face_edge_info.iter())
.try_for_each(|(sketch, info)| {
sketch.id = info.object_id;
let first_edge = info.edges.first().copied();
match first_edge {
Some(edge) => sketch.mirror = Some(edge),
None => {
return Err(KclError::new_engine(KclErrorDetails::new(
"No edges found in mirror info".to_string(),
vec![args.source_range],
)));
}
}
Ok(())
})?;
} else {
return Err(KclError::new_engine(KclErrorDetails::new(
format!("EntityMirrorAcrossEdge response was not as expected: {resp:?}"),
vec![args.source_range],
)));
};
}
}
Ok(starting_sketches)
}