* Make tag identifiers monotonic Signed-off-by: Nick Cameron <nrc@ncameron.org> * Use epochs rather than snapshots in memory Signed-off-by: Nick Cameron <nrc@ncameron.org> --------- Signed-off-by: Nick Cameron <nrc@ncameron.org>
466 lines
13 KiB
Rust
466 lines
13 KiB
Rust
//! Standard library shapes.
|
|
|
|
use anyhow::Result;
|
|
use kcl_derive_docs::stdlib;
|
|
use kcmc::{
|
|
each_cmd as mcmd,
|
|
length_unit::LengthUnit,
|
|
shared::{Angle, Point2d as KPoint2d},
|
|
ModelingCmd,
|
|
};
|
|
use kittycad_modeling_cmds as kcmc;
|
|
use kittycad_modeling_cmds::shared::PathSegment;
|
|
use schemars::JsonSchema;
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
use crate::{
|
|
errors::{KclError, KclErrorDetails},
|
|
execution::{BasePath, ExecState, GeoMeta, KclValue, Path, Sketch, SketchSurface},
|
|
parsing::ast::types::TagNode,
|
|
std::{
|
|
sketch::NEW_TAG_KW,
|
|
utils::{calculate_circle_center, distance},
|
|
Args,
|
|
},
|
|
};
|
|
|
|
/// A sketch surface or a sketch.
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema)]
|
|
#[ts(export)]
|
|
#[serde(untagged)]
|
|
pub enum SketchOrSurface {
|
|
SketchSurface(SketchSurface),
|
|
Sketch(Box<Sketch>),
|
|
}
|
|
|
|
/// Sketch a circle.
|
|
pub async fn circle(exec_state: &mut ExecState, args: Args) -> Result<KclValue, KclError> {
|
|
let sketch_or_surface = args.get_unlabeled_kw_arg("sketchOrSurface")?;
|
|
let center = args.get_kw_arg("center")?;
|
|
let radius = args.get_kw_arg("radius")?;
|
|
let tag = args.get_kw_arg_opt(NEW_TAG_KW)?;
|
|
|
|
let sketch = inner_circle(sketch_or_surface, center, radius, tag, exec_state, args).await?;
|
|
Ok(KclValue::Sketch {
|
|
value: Box::new(sketch),
|
|
})
|
|
}
|
|
|
|
/// Construct a 2-dimensional circle, of the specified radius, centered at
|
|
/// the provided (x, y) origin point.
|
|
///
|
|
/// ```no_run
|
|
/// exampleSketch = startSketchOn("-XZ")
|
|
/// |> circle( center = [0, 0], radius = 10 )
|
|
///
|
|
/// example = extrude(exampleSketch, length = 5)
|
|
/// ```
|
|
///
|
|
/// ```no_run
|
|
/// exampleSketch = startSketchOn("XZ")
|
|
/// |> startProfileAt([-15, 0], %)
|
|
/// |> line(end = [30, 0])
|
|
/// |> line(end = [0, 30])
|
|
/// |> line(end = [-30, 0])
|
|
/// |> close()
|
|
/// |> hole(circle( center = [0, 15], radius = 5), %)
|
|
///
|
|
/// example = extrude(exampleSketch, length = 5)
|
|
/// ```
|
|
#[stdlib {
|
|
name = "circle",
|
|
keywords = true,
|
|
unlabeled_first = true,
|
|
args = {
|
|
sketch_or_surface = {docs = "Plane or surface to sketch on."},
|
|
center = {docs = "The center of the circle."},
|
|
radius = {docs = "The radius of the circle."},
|
|
tag = { docs = "Create a new tag which refers to this circle"},
|
|
}
|
|
}]
|
|
async fn inner_circle(
|
|
sketch_or_surface: SketchOrSurface,
|
|
center: [f64; 2],
|
|
radius: f64,
|
|
tag: Option<TagNode>,
|
|
exec_state: &mut ExecState,
|
|
args: Args,
|
|
) -> Result<Sketch, KclError> {
|
|
let sketch_surface = match sketch_or_surface {
|
|
SketchOrSurface::SketchSurface(surface) => surface,
|
|
SketchOrSurface::Sketch(s) => s.on,
|
|
};
|
|
let units = sketch_surface.units();
|
|
let sketch = crate::std::sketch::inner_start_profile_at(
|
|
[center[0] + radius, center[1]],
|
|
sketch_surface,
|
|
None,
|
|
exec_state,
|
|
args.clone(),
|
|
)
|
|
.await?;
|
|
|
|
let from = [center[0] + radius, center[1]];
|
|
let angle_start = Angle::zero();
|
|
let angle_end = Angle::turn();
|
|
|
|
let id = exec_state.next_uuid();
|
|
|
|
args.batch_modeling_cmd(
|
|
id,
|
|
ModelingCmd::from(mcmd::ExtendPath {
|
|
path: sketch.id.into(),
|
|
segment: PathSegment::Arc {
|
|
start: angle_start,
|
|
end: angle_end,
|
|
center: KPoint2d::from(center).map(LengthUnit),
|
|
radius: radius.into(),
|
|
relative: false,
|
|
},
|
|
}),
|
|
)
|
|
.await?;
|
|
|
|
let current_path = Path::Circle {
|
|
base: BasePath {
|
|
from,
|
|
to: from,
|
|
tag: tag.clone(),
|
|
units,
|
|
geo_meta: GeoMeta {
|
|
id,
|
|
metadata: args.source_range.into(),
|
|
},
|
|
},
|
|
radius,
|
|
center,
|
|
ccw: angle_start < angle_end,
|
|
};
|
|
|
|
let mut new_sketch = sketch.clone();
|
|
if let Some(tag) = &tag {
|
|
new_sketch.add_tag(tag, ¤t_path, exec_state);
|
|
}
|
|
|
|
new_sketch.paths.push(current_path);
|
|
|
|
args.batch_modeling_cmd(id, ModelingCmd::from(mcmd::ClosePath { path_id: new_sketch.id }))
|
|
.await?;
|
|
|
|
Ok(new_sketch)
|
|
}
|
|
|
|
/// Sketch a 3-point circle.
|
|
pub async fn circle_three_point(exec_state: &mut ExecState, args: Args) -> Result<KclValue, KclError> {
|
|
let p1 = args.get_kw_arg("p1")?;
|
|
let p2 = args.get_kw_arg("p2")?;
|
|
let p3 = args.get_kw_arg("p3")?;
|
|
let sketch_surface_or_group = args.get_unlabeled_kw_arg("sketch_surface_or_group")?;
|
|
let tag = args.get_kw_arg_opt("tag")?;
|
|
|
|
let sketch = inner_circle_three_point(p1, p2, p3, sketch_surface_or_group, tag, exec_state, args).await?;
|
|
Ok(KclValue::Sketch {
|
|
value: Box::new(sketch),
|
|
})
|
|
}
|
|
|
|
/// Construct a circle derived from 3 points.
|
|
///
|
|
/// ```no_run
|
|
/// exampleSketch = startSketchOn("XY")
|
|
/// |> circleThreePoint(p1 = [10,10], p2 = [20,8], p3 = [15,5])
|
|
/// |> extrude(length = 5)
|
|
/// ```
|
|
#[stdlib {
|
|
name = "circleThreePoint",
|
|
keywords = true,
|
|
unlabeled_first = true,
|
|
args = {
|
|
p1 = {docs = "1st point to derive the circle."},
|
|
p2 = {docs = "2nd point to derive the circle."},
|
|
p3 = {docs = "3rd point to derive the circle."},
|
|
sketch_surface_or_group = {docs = "Plane or surface to sketch on."},
|
|
tag = {docs = "Identifier for the circle to reference elsewhere."},
|
|
}
|
|
}]
|
|
|
|
// Similar to inner_circle, but needs to retain 3-point information in the
|
|
// path so it can be used for other features, otherwise it's lost.
|
|
async fn inner_circle_three_point(
|
|
p1: [f64; 2],
|
|
p2: [f64; 2],
|
|
p3: [f64; 2],
|
|
sketch_surface_or_group: SketchOrSurface,
|
|
tag: Option<TagNode>,
|
|
exec_state: &mut ExecState,
|
|
args: Args,
|
|
) -> Result<Sketch, KclError> {
|
|
let center = calculate_circle_center(p1, p2, p3);
|
|
// It can be the distance to any of the 3 points - they all lay on the circumference.
|
|
let radius = distance(center.into(), p2.into());
|
|
|
|
let sketch_surface = match sketch_surface_or_group {
|
|
SketchOrSurface::SketchSurface(surface) => surface,
|
|
SketchOrSurface::Sketch(group) => group.on,
|
|
};
|
|
let sketch = crate::std::sketch::inner_start_profile_at(
|
|
[center[0] + radius, center[1]],
|
|
sketch_surface,
|
|
None,
|
|
exec_state,
|
|
args.clone(),
|
|
)
|
|
.await?;
|
|
|
|
let from = [center[0] + radius, center[1]];
|
|
let angle_start = Angle::zero();
|
|
let angle_end = Angle::turn();
|
|
|
|
let id = exec_state.next_uuid();
|
|
|
|
args.batch_modeling_cmd(
|
|
id,
|
|
ModelingCmd::from(mcmd::ExtendPath {
|
|
path: sketch.id.into(),
|
|
segment: PathSegment::Arc {
|
|
start: angle_start,
|
|
end: angle_end,
|
|
center: KPoint2d::from(center).map(LengthUnit),
|
|
radius: radius.into(),
|
|
relative: false,
|
|
},
|
|
}),
|
|
)
|
|
.await?;
|
|
|
|
let current_path = Path::CircleThreePoint {
|
|
base: BasePath {
|
|
from,
|
|
to: from,
|
|
tag: tag.clone(),
|
|
units: sketch.units,
|
|
geo_meta: GeoMeta {
|
|
id,
|
|
metadata: args.source_range.into(),
|
|
},
|
|
},
|
|
p1,
|
|
p2,
|
|
p3,
|
|
};
|
|
|
|
let mut new_sketch = sketch.clone();
|
|
if let Some(tag) = &tag {
|
|
new_sketch.add_tag(tag, ¤t_path, exec_state);
|
|
}
|
|
|
|
new_sketch.paths.push(current_path);
|
|
|
|
args.batch_modeling_cmd(id, ModelingCmd::from(mcmd::ClosePath { path_id: new_sketch.id }))
|
|
.await?;
|
|
|
|
Ok(new_sketch)
|
|
}
|
|
|
|
/// Type of the polygon
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema, Default)]
|
|
#[ts(export)]
|
|
#[serde(rename_all = "lowercase")]
|
|
pub enum PolygonType {
|
|
#[default]
|
|
Inscribed,
|
|
Circumscribed,
|
|
}
|
|
|
|
/// Data for drawing a polygon
|
|
#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema)]
|
|
#[ts(export)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct PolygonData {
|
|
/// The radius of the polygon
|
|
pub radius: f64,
|
|
/// The number of sides in the polygon
|
|
pub num_sides: u64,
|
|
/// The center point of the polygon
|
|
pub center: [f64; 2],
|
|
/// The type of the polygon (inscribed or circumscribed)
|
|
#[serde(skip)]
|
|
pub polygon_type: PolygonType,
|
|
/// Whether the polygon is inscribed (true) or circumscribed (false) about a circle with the specified radius
|
|
#[serde(default = "default_inscribed")]
|
|
pub inscribed: bool,
|
|
}
|
|
|
|
fn default_inscribed() -> bool {
|
|
true
|
|
}
|
|
|
|
/// Create a regular polygon with the specified number of sides and radius.
|
|
pub async fn polygon(exec_state: &mut ExecState, args: Args) -> Result<KclValue, KclError> {
|
|
let (data, sketch_surface_or_group, tag): (PolygonData, SketchOrSurface, Option<TagNode>) =
|
|
args.get_polygon_args()?;
|
|
|
|
let sketch = inner_polygon(data, sketch_surface_or_group, tag, exec_state, args).await?;
|
|
Ok(KclValue::Sketch {
|
|
value: Box::new(sketch),
|
|
})
|
|
}
|
|
|
|
/// Create a regular polygon with the specified number of sides that is either inscribed or circumscribed around a circle of the specified radius.
|
|
///
|
|
/// ```no_run
|
|
/// // Create a regular hexagon inscribed in a circle of radius 10
|
|
/// hex = startSketchOn('XY')
|
|
/// |> polygon({
|
|
/// radius = 10,
|
|
/// numSides = 6,
|
|
/// center = [0, 0],
|
|
/// inscribed = true,
|
|
/// }, %)
|
|
///
|
|
/// example = extrude(hex, length = 5)
|
|
/// ```
|
|
///
|
|
/// ```no_run
|
|
/// // Create a square circumscribed around a circle of radius 5
|
|
/// square = startSketchOn('XY')
|
|
/// |> polygon({
|
|
/// radius = 5.0,
|
|
/// numSides = 4,
|
|
/// center = [10, 10],
|
|
/// inscribed = false,
|
|
/// }, %)
|
|
/// example = extrude(square, length = 5)
|
|
/// ```
|
|
#[stdlib {
|
|
name = "polygon",
|
|
}]
|
|
async fn inner_polygon(
|
|
data: PolygonData,
|
|
sketch_surface_or_group: SketchOrSurface,
|
|
tag: Option<TagNode>,
|
|
exec_state: &mut ExecState,
|
|
args: Args,
|
|
) -> Result<Sketch, KclError> {
|
|
if data.num_sides < 3 {
|
|
return Err(KclError::Type(KclErrorDetails {
|
|
message: "Polygon must have at least 3 sides".to_string(),
|
|
source_ranges: vec![args.source_range],
|
|
}));
|
|
}
|
|
|
|
if data.radius <= 0.0 {
|
|
return Err(KclError::Type(KclErrorDetails {
|
|
message: "Radius must be greater than 0".to_string(),
|
|
source_ranges: vec![args.source_range],
|
|
}));
|
|
}
|
|
|
|
let sketch_surface = match sketch_surface_or_group {
|
|
SketchOrSurface::SketchSurface(surface) => surface,
|
|
SketchOrSurface::Sketch(group) => group.on,
|
|
};
|
|
|
|
let half_angle = std::f64::consts::PI / data.num_sides as f64;
|
|
|
|
let radius_to_vertices = match data.polygon_type {
|
|
PolygonType::Inscribed => data.radius,
|
|
PolygonType::Circumscribed => data.radius / half_angle.cos(),
|
|
};
|
|
|
|
let angle_step = 2.0 * std::f64::consts::PI / data.num_sides as f64;
|
|
|
|
let vertices: Vec<[f64; 2]> = (0..data.num_sides)
|
|
.map(|i| {
|
|
let angle = angle_step * i as f64;
|
|
[
|
|
data.center[0] + radius_to_vertices * angle.cos(),
|
|
data.center[1] + radius_to_vertices * angle.sin(),
|
|
]
|
|
})
|
|
.collect();
|
|
|
|
let mut sketch =
|
|
crate::std::sketch::inner_start_profile_at(vertices[0], sketch_surface, None, exec_state, args.clone()).await?;
|
|
|
|
// Draw all the lines with unique IDs and modified tags
|
|
for vertex in vertices.iter().skip(1) {
|
|
let from = sketch.current_pen_position()?;
|
|
let id = exec_state.next_uuid();
|
|
|
|
args.batch_modeling_cmd(
|
|
id,
|
|
ModelingCmd::from(mcmd::ExtendPath {
|
|
path: sketch.id.into(),
|
|
segment: PathSegment::Line {
|
|
end: KPoint2d::from(*vertex).with_z(0.0).map(LengthUnit),
|
|
relative: false,
|
|
},
|
|
}),
|
|
)
|
|
.await?;
|
|
|
|
let current_path = Path::ToPoint {
|
|
base: BasePath {
|
|
from: from.into(),
|
|
to: *vertex,
|
|
tag: tag.clone(),
|
|
units: sketch.units,
|
|
geo_meta: GeoMeta {
|
|
id,
|
|
metadata: args.source_range.into(),
|
|
},
|
|
},
|
|
};
|
|
|
|
if let Some(tag) = &tag {
|
|
sketch.add_tag(tag, ¤t_path, exec_state);
|
|
}
|
|
|
|
sketch.paths.push(current_path);
|
|
}
|
|
|
|
// Close the polygon by connecting back to the first vertex with a new ID
|
|
let from = sketch.current_pen_position()?;
|
|
let close_id = exec_state.next_uuid();
|
|
|
|
args.batch_modeling_cmd(
|
|
close_id,
|
|
ModelingCmd::from(mcmd::ExtendPath {
|
|
path: sketch.id.into(),
|
|
segment: PathSegment::Line {
|
|
end: KPoint2d::from(vertices[0]).with_z(0.0).map(LengthUnit),
|
|
relative: false,
|
|
},
|
|
}),
|
|
)
|
|
.await?;
|
|
|
|
let current_path = Path::ToPoint {
|
|
base: BasePath {
|
|
from: from.into(),
|
|
to: vertices[0],
|
|
tag: tag.clone(),
|
|
units: sketch.units,
|
|
geo_meta: GeoMeta {
|
|
id: close_id,
|
|
metadata: args.source_range.into(),
|
|
},
|
|
},
|
|
};
|
|
|
|
if let Some(tag) = &tag {
|
|
sketch.add_tag(tag, ¤t_path, exec_state);
|
|
}
|
|
|
|
sketch.paths.push(current_path);
|
|
|
|
args.batch_modeling_cmd(
|
|
exec_state.next_uuid(),
|
|
ModelingCmd::from(mcmd::ClosePath { path_id: sketch.id }),
|
|
)
|
|
.await?;
|
|
|
|
Ok(sketch)
|
|
}
|