use anyhow::Result;
use kcmc::ModelingCmd;
use kcmc::each_cmd as mcmd;
use kcmc::length_unit::LengthUnit;
use kcmc::shared::CutStrategy;
use kcmc::shared::CutTypeV2;
use kittycad_modeling_cmds::shared::Angle;
use kittycad_modeling_cmds::{self as kcmc};
use super::args::TyF64;
use crate::errors::KclError;
use crate::errors::KclErrorDetails;
use crate::execution::ChamferSurface;
use crate::execution::EdgeCut;
use crate::execution::ExecState;
use crate::execution::ExtrudeSurface;
use crate::execution::GeoMeta;
use crate::execution::KclValue;
use crate::execution::ModelingCmdMeta;
use crate::execution::Sketch;
use crate::execution::Solid;
use crate::execution::types::RuntimeType;
use crate::parsing::ast::types::TagNode;
use crate::std::Args;
use crate::std::csg::CsgAlgorithm;
use crate::std::fillet::EdgeReference;
pub(crate) const DEFAULT_TOLERANCE: f64 = 0.0000001;
pub async fn chamfer(exec_state: &mut ExecState, args: Args) -> Result<KclValue, KclError> {
let solid = args.get_unlabeled_kw_arg("solid", &RuntimeType::solid(), exec_state)?;
let length: TyF64 = args.get_kw_arg("length", &RuntimeType::length(), exec_state)?;
let tags = args.kw_arg_edge_array_and_source("tags")?;
let second_length = args.get_kw_arg_opt("secondLength", &RuntimeType::length(), exec_state)?;
let angle = args.get_kw_arg_opt("angle", &RuntimeType::angle(), exec_state)?;
let legacy_csg: Option<bool> = args.get_kw_arg_opt("legacyMethod", &RuntimeType::bool(), exec_state)?;
let csg_algorithm = CsgAlgorithm::legacy(legacy_csg.unwrap_or_default());
let tag = args.get_kw_arg_opt("tag", &RuntimeType::tag_decl(), exec_state)?;
super::fillet::validate_unique(&tags)?;
let tags: Vec<EdgeReference> = tags.into_iter().map(|item| item.0).collect();
let value = inner_chamfer(
solid,
length,
tags,
second_length,
angle,
None,
tag,
csg_algorithm,
exec_state,
args,
)
.await?;
Ok(KclValue::Solid { value })
}
#[allow(clippy::too_many_arguments)]
async fn inner_chamfer(
solid: Box<Solid>,
length: TyF64,
tags: Vec<EdgeReference>,
second_length: Option<TyF64>,
angle: Option<TyF64>,
custom_profile: Option<Sketch>,
tag: Option<TagNode>,
csg_algorithm: CsgAlgorithm,
exec_state: &mut ExecState,
args: Args,
) -> Result<Box<Solid>, KclError> {
if tag.is_some() && tags.len() > 1 {
return Err(KclError::new_type(KclErrorDetails::new(
"You can only tag one edge at a time with a tagged chamfer. Either delete the tag for the chamfer fn if you don't need it OR separate into individual chamfer functions for each tag.".to_string(),
vec![args.source_range],
)));
}
if angle.is_some() && second_length.is_some() {
return Err(KclError::new_semantic(KclErrorDetails::new(
"Cannot specify both an angle and a second length. Specify only one.".to_string(),
vec![args.source_range],
)));
}
let strategy = if second_length.is_some() || angle.is_some() || custom_profile.is_some() {
CutStrategy::Csg
} else {
Default::default()
};
let second_distance = second_length.map(|x| LengthUnit(x.to_mm()));
let angle = angle.map(|x| Angle::from_degrees(x.to_degrees(exec_state, args.source_range)));
if let Some(angle) = angle
&& (angle.ge(&Angle::quarter_circle()) || angle.le(&Angle::zero()))
{
return Err(KclError::new_semantic(KclErrorDetails::new(
"The angle of a chamfer must be greater than zero and less than 90 degrees.".to_string(),
vec![args.source_range],
)));
}
let cut_type = if let Some(custom_profile) = custom_profile {
exec_state
.batch_modeling_cmd(
ModelingCmdMeta::from_args(exec_state, &args),
ModelingCmd::from(
mcmd::ObjectVisible::builder()
.object_id(custom_profile.id)
.hidden(true)
.build(),
),
)
.await?;
CutTypeV2::Custom {
path: custom_profile.id,
}
} else {
CutTypeV2::Chamfer {
distance: LengthUnit(length.to_mm()),
second_distance,
angle,
swap: false,
}
};
let mut solid = solid.clone();
for edge_tag in tags {
let edge_ids = edge_tag.get_all_engine_ids(exec_state, &args)?;
for edge_id in edge_ids {
let id = exec_state.next_uuid();
exec_state
.batch_end_cmd(
ModelingCmdMeta::from_args_id(exec_state, &args, id),
ModelingCmd::from(
mcmd::Solid3dCutEdges::builder()
.use_legacy(csg_algorithm.is_legacy())
.edge_ids(vec![edge_id])
.extra_face_ids(vec![])
.strategy(strategy)
.object_id(solid.id)
.tolerance(LengthUnit(DEFAULT_TOLERANCE)) .cut_type(cut_type)
.build(),
),
)
.await?;
solid.edge_cuts.push(EdgeCut::Chamfer {
id,
edge_id,
length: length.clone(),
tag: Box::new(tag.clone()),
});
if let Some(ref tag) = tag {
solid.value.push(ExtrudeSurface::Chamfer(ChamferSurface {
face_id: id,
tag: Some(tag.clone()),
geo_meta: GeoMeta {
id,
metadata: args.source_range.into(),
},
}));
}
}
}
Ok(solid)
}