1use std::f64::consts::TAU;
2use std::ops::Add;
3use std::ops::AddAssign;
4use std::ops::Mul;
5use std::ops::Sub;
6use std::ops::SubAssign;
7use std::sync::Arc;
8
9use anyhow::Result;
10use indexmap::IndexMap;
11use kcl_error::SourceRange;
12use kittycad_modeling_cmds::ModelingCmd;
13use kittycad_modeling_cmds::each_cmd as mcmd;
14use kittycad_modeling_cmds::length_unit::LengthUnit;
15use kittycad_modeling_cmds::units::UnitLength;
16use kittycad_modeling_cmds::websocket::ModelingCmdReq;
17use kittycad_modeling_cmds::{self as kcmc};
18use parse_display::Display;
19use parse_display::FromStr;
20use serde::Deserialize;
21use serde::Serialize;
22use uuid::Uuid;
23
24use crate::NodePath;
25use crate::engine::DEFAULT_PLANE_INFO;
26use crate::engine::PlaneName;
27use crate::errors::KclError;
28use crate::errors::KclErrorDetails;
29use crate::exec::KclValue;
30use crate::execution::ArtifactId;
31use crate::execution::ExecState;
32use crate::execution::ExecutorContext;
33use crate::execution::Metadata;
34use crate::execution::TagEngineInfo;
35use crate::execution::TagIdentifier;
36use crate::execution::normalize_to_solver_distance_unit;
37use crate::execution::types::NumericType;
38use crate::execution::types::adjust_length;
39use crate::front::ArcCtor;
40use crate::front::CircleCtor;
41use crate::front::ControlPointSplineCtor;
42use crate::front::Freedom;
43use crate::front::LineCtor;
44use crate::front::Number;
45use crate::front::ObjectId;
46use crate::front::Point2d as ApiPoint2d;
47use crate::front::PointCtor;
48use crate::parsing::ast::types::Node;
49use crate::parsing::ast::types::NodeRef;
50use crate::parsing::ast::types::TagDeclarator;
51use crate::parsing::ast::types::TagNode;
52use crate::std::Args;
53use crate::std::args::TyF64;
54use crate::std::sketch::FaceTag;
55use crate::std::sketch::PlaneData;
56use crate::util::MathExt;
57
58type Point3D = kcmc::shared::Point3d<f64>;
59
60#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
62#[ts(export)]
63#[serde(tag = "type", rename_all = "camelCase")]
64pub struct GdtAnnotation {
65 pub id: uuid::Uuid,
67 #[serde(skip)]
68 pub meta: Vec<Metadata>,
69}
70
71#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
73#[ts(export)]
74#[serde(tag = "type")]
75#[allow(clippy::large_enum_variant)]
76pub enum Geometry {
77 Sketch(Sketch),
78 Solid(Solid),
79}
80
81impl Geometry {
82 pub fn id(&self) -> uuid::Uuid {
83 match self {
84 Geometry::Sketch(s) => s.id,
85 Geometry::Solid(e) => e.id,
86 }
87 }
88
89 pub fn original_id(&self) -> uuid::Uuid {
93 match self {
94 Geometry::Sketch(s) => s.original_id,
95 Geometry::Solid(e) => e.original_id(),
96 }
97 }
98}
99
100#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
102#[ts(export)]
103#[serde(tag = "type")]
104#[allow(clippy::large_enum_variant)]
105pub enum GeometryWithImportedGeometry {
106 Sketch(Sketch),
107 Solid(Solid),
108 ImportedGeometry(Box<ImportedGeometry>),
109}
110
111impl GeometryWithImportedGeometry {
112 pub async fn id(&mut self, ctx: &ExecutorContext) -> Result<uuid::Uuid, KclError> {
113 match self {
114 GeometryWithImportedGeometry::Sketch(s) => Ok(s.id),
115 GeometryWithImportedGeometry::Solid(e) => Ok(e.id),
116 GeometryWithImportedGeometry::ImportedGeometry(i) => {
117 let id = i.id(ctx).await?;
118 Ok(id)
119 }
120 }
121 }
122}
123
124#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
126#[ts(export)]
127#[serde(tag = "type")]
128#[allow(clippy::vec_box)]
129pub enum Geometries {
130 Sketches(Vec<Sketch>),
131 Solids(Vec<Solid>),
132}
133
134impl From<Geometry> for Geometries {
135 fn from(value: Geometry) -> Self {
136 match value {
137 Geometry::Sketch(x) => Self::Sketches(vec![x]),
138 Geometry::Solid(x) => Self::Solids(vec![x]),
139 }
140 }
141}
142
143#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
145#[ts(export)]
146#[serde(rename_all = "camelCase")]
147pub struct ImportedGeometry {
148 pub id: uuid::Uuid,
150 pub value: Vec<String>,
152 #[serde(skip)]
153 pub meta: Vec<Metadata>,
154 #[serde(skip)]
156 completed: bool,
157}
158
159impl ImportedGeometry {
160 pub fn new(id: uuid::Uuid, value: Vec<String>, meta: Vec<Metadata>) -> Self {
161 Self {
162 id,
163 value,
164 meta,
165 completed: false,
166 }
167 }
168
169 async fn wait_for_finish(&mut self, ctx: &ExecutorContext) -> Result<(), KclError> {
170 if self.completed {
171 return Ok(());
172 }
173
174 ctx.engine
175 .ensure_async_command_completed(self.id, self.meta.first().map(|m| m.source_range))
176 .await?;
177
178 self.completed = true;
179
180 Ok(())
181 }
182
183 pub async fn id(&mut self, ctx: &ExecutorContext) -> Result<uuid::Uuid, KclError> {
184 if !self.completed {
185 self.wait_for_finish(ctx).await?;
186 }
187
188 Ok(self.id)
189 }
190}
191
192#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
194#[ts(export)]
195#[serde(tag = "type", rename_all = "camelCase")]
196#[allow(clippy::vec_box)]
197pub enum HideableGeometry {
198 ImportedGeometry(Box<ImportedGeometry>),
199 SolidSet(Vec<Solid>),
200 SketchSet(Vec<Sketch>),
201 HelixSet(Vec<Helix>),
202 GdtAnnotationSet(Vec<GdtAnnotation>),
203}
204
205impl From<HideableGeometry> for crate::execution::KclValue {
206 fn from(value: HideableGeometry) -> Self {
207 match value {
208 HideableGeometry::ImportedGeometry(s) => crate::execution::KclValue::ImportedGeometry(*s),
209 HideableGeometry::SolidSet(mut s) => {
210 if s.len() == 1
211 && let Some(s) = s.pop()
212 {
213 crate::execution::KclValue::Solid { value: Box::new(s) }
214 } else {
215 crate::execution::KclValue::HomArray {
216 value: s
217 .into_iter()
218 .map(|s| crate::execution::KclValue::Solid { value: Box::new(s) })
219 .collect(),
220 ty: crate::execution::types::RuntimeType::solid(),
221 }
222 }
223 }
224 HideableGeometry::GdtAnnotationSet(mut s) => {
225 if s.len() == 1
226 && let Some(s) = s.pop()
227 {
228 crate::execution::KclValue::GdtAnnotation { value: Box::new(s) }
229 } else {
230 crate::execution::KclValue::HomArray {
231 value: s
232 .into_iter()
233 .map(|s| crate::execution::KclValue::GdtAnnotation { value: Box::new(s) })
234 .collect(),
235 ty: crate::execution::types::RuntimeType::gdt(),
236 }
237 }
238 }
239 HideableGeometry::SketchSet(mut s) => {
240 if s.len() == 1
241 && let Some(s) = s.pop()
242 {
243 crate::execution::KclValue::Sketch { value: Box::new(s) }
244 } else {
245 crate::execution::KclValue::HomArray {
246 value: s
247 .into_iter()
248 .map(|s| crate::execution::KclValue::Sketch { value: Box::new(s) })
249 .collect(),
250 ty: crate::execution::types::RuntimeType::sketch(),
251 }
252 }
253 }
254 HideableGeometry::HelixSet(mut s) => {
255 if s.len() == 1
256 && let Some(s) = s.pop()
257 {
258 crate::execution::KclValue::Helix { value: Box::new(s) }
259 } else {
260 crate::execution::KclValue::HomArray {
261 value: s
262 .into_iter()
263 .map(|s| crate::execution::KclValue::Helix { value: Box::new(s) })
264 .collect(),
265 ty: crate::execution::types::RuntimeType::helices(),
266 }
267 }
268 }
269 }
270 }
271}
272
273impl HideableGeometry {
274 pub(crate) async fn ids(&mut self, ctx: &ExecutorContext) -> Result<Vec<uuid::Uuid>, KclError> {
275 match self {
276 HideableGeometry::ImportedGeometry(s) => {
277 let id = s.id(ctx).await?;
278
279 Ok(vec![id])
280 }
281 HideableGeometry::SolidSet(s) => Ok(s.iter().map(|s| s.id).collect()),
282 HideableGeometry::GdtAnnotationSet(s) => Ok(s.iter().map(|s| s.id).collect()),
283 HideableGeometry::SketchSet(s) => Ok(s.iter().map(|s| s.id).collect()),
284 HideableGeometry::HelixSet(s) => Ok(s.iter().map(|s| s.value).collect()),
285 }
286 }
287}
288
289#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
291#[ts(export)]
292#[serde(tag = "type", rename_all = "camelCase")]
293#[allow(clippy::vec_box)]
294pub enum SolidOrSketchOrImportedGeometry {
295 ImportedGeometry(Box<ImportedGeometry>),
296 SolidSet(Vec<Solid>),
297 SketchSet(Vec<Sketch>),
298}
299
300impl From<SolidOrSketchOrImportedGeometry> for crate::execution::KclValue {
301 fn from(value: SolidOrSketchOrImportedGeometry) -> Self {
302 match value {
303 SolidOrSketchOrImportedGeometry::ImportedGeometry(s) => crate::execution::KclValue::ImportedGeometry(*s),
304 SolidOrSketchOrImportedGeometry::SolidSet(mut s) => {
305 if s.len() == 1
306 && let Some(s) = s.pop()
307 {
308 crate::execution::KclValue::Solid { value: Box::new(s) }
309 } else {
310 crate::execution::KclValue::HomArray {
311 value: s
312 .into_iter()
313 .map(|s| crate::execution::KclValue::Solid { value: Box::new(s) })
314 .collect(),
315 ty: crate::execution::types::RuntimeType::solid(),
316 }
317 }
318 }
319 SolidOrSketchOrImportedGeometry::SketchSet(mut s) => {
320 if s.len() == 1
321 && let Some(s) = s.pop()
322 {
323 crate::execution::KclValue::Sketch { value: Box::new(s) }
324 } else {
325 crate::execution::KclValue::HomArray {
326 value: s
327 .into_iter()
328 .map(|s| crate::execution::KclValue::Sketch { value: Box::new(s) })
329 .collect(),
330 ty: crate::execution::types::RuntimeType::sketch(),
331 }
332 }
333 }
334 }
335 }
336}
337
338impl SolidOrSketchOrImportedGeometry {
339 pub(crate) async fn ids(&mut self, ctx: &ExecutorContext) -> Result<Vec<uuid::Uuid>, KclError> {
340 match self {
341 SolidOrSketchOrImportedGeometry::ImportedGeometry(s) => {
342 let id = s.id(ctx).await?;
343
344 Ok(vec![id])
345 }
346 SolidOrSketchOrImportedGeometry::SolidSet(s) => Ok(s.iter().map(|s| s.id).collect()),
347 SolidOrSketchOrImportedGeometry::SketchSet(s) => Ok(s.iter().map(|s| s.id).collect()),
348 }
349 }
350}
351
352#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
354#[ts(export)]
355#[serde(tag = "type", rename_all = "camelCase")]
356#[allow(clippy::vec_box)]
357pub enum SolidOrImportedGeometry {
358 ImportedGeometry(Box<ImportedGeometry>),
359 SolidSet(Vec<Solid>),
360}
361
362impl From<SolidOrImportedGeometry> for crate::execution::KclValue {
363 fn from(value: SolidOrImportedGeometry) -> Self {
364 match value {
365 SolidOrImportedGeometry::ImportedGeometry(s) => crate::execution::KclValue::ImportedGeometry(*s),
366 SolidOrImportedGeometry::SolidSet(mut s) => {
367 if s.len() == 1
368 && let Some(s) = s.pop()
369 {
370 crate::execution::KclValue::Solid { value: Box::new(s) }
371 } else {
372 crate::execution::KclValue::HomArray {
373 value: s
374 .into_iter()
375 .map(|s| crate::execution::KclValue::Solid { value: Box::new(s) })
376 .collect(),
377 ty: crate::execution::types::RuntimeType::solid(),
378 }
379 }
380 }
381 }
382 }
383}
384
385impl SolidOrImportedGeometry {
386 pub(crate) async fn ids(&mut self, ctx: &ExecutorContext) -> Result<Vec<uuid::Uuid>, KclError> {
387 match self {
388 SolidOrImportedGeometry::ImportedGeometry(s) => {
389 let id = s.id(ctx).await?;
390
391 Ok(vec![id])
392 }
393 SolidOrImportedGeometry::SolidSet(s) => Ok(s.iter().map(|s| s.id).collect()),
394 }
395 }
396}
397
398#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
400#[ts(export)]
401#[serde(rename_all = "camelCase")]
402pub struct Helix {
403 pub value: uuid::Uuid,
405 pub artifact_id: ArtifactId,
407 pub revolutions: f64,
409 pub angle_start: f64,
411 pub ccw: bool,
413 pub cylinder_id: Option<uuid::Uuid>,
415 pub units: UnitLength,
416 #[serde(skip)]
417 pub meta: Vec<Metadata>,
418}
419
420#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
421#[ts(export)]
422#[serde(rename_all = "camelCase")]
423pub struct Plane {
424 pub id: uuid::Uuid,
426 pub artifact_id: ArtifactId,
428 #[serde(skip_serializing_if = "Option::is_none")]
431 pub object_id: Option<ObjectId>,
432 pub kind: PlaneKind,
434 #[serde(flatten)]
436 pub info: PlaneInfo,
437 #[serde(skip)]
438 pub meta: Vec<Metadata>,
439}
440
441#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, ts_rs::TS)]
442#[ts(export)]
443#[serde(rename_all = "camelCase")]
444pub struct PlaneInfo {
445 pub origin: Point3d,
447 pub x_axis: Point3d,
449 pub y_axis: Point3d,
451 pub z_axis: Point3d,
453}
454
455impl PlaneInfo {
456 pub(crate) fn into_plane_data(self) -> PlaneData {
457 if self.origin.is_zero() {
458 match self {
459 Self {
460 origin:
461 Point3d {
462 x: 0.0,
463 y: 0.0,
464 z: 0.0,
465 units: Some(UnitLength::Millimeters),
466 },
467 x_axis:
468 Point3d {
469 x: 1.0,
470 y: 0.0,
471 z: 0.0,
472 units: _,
473 },
474 y_axis:
475 Point3d {
476 x: 0.0,
477 y: 1.0,
478 z: 0.0,
479 units: _,
480 },
481 z_axis: _,
482 } => return PlaneData::XY,
483 Self {
484 origin:
485 Point3d {
486 x: 0.0,
487 y: 0.0,
488 z: 0.0,
489 units: Some(UnitLength::Millimeters),
490 },
491 x_axis:
492 Point3d {
493 x: -1.0,
494 y: 0.0,
495 z: 0.0,
496 units: _,
497 },
498 y_axis:
499 Point3d {
500 x: 0.0,
501 y: 1.0,
502 z: 0.0,
503 units: _,
504 },
505 z_axis: _,
506 } => return PlaneData::NegXY,
507 Self {
508 origin:
509 Point3d {
510 x: 0.0,
511 y: 0.0,
512 z: 0.0,
513 units: Some(UnitLength::Millimeters),
514 },
515 x_axis:
516 Point3d {
517 x: 1.0,
518 y: 0.0,
519 z: 0.0,
520 units: _,
521 },
522 y_axis:
523 Point3d {
524 x: 0.0,
525 y: 0.0,
526 z: 1.0,
527 units: _,
528 },
529 z_axis: _,
530 } => return PlaneData::XZ,
531 Self {
532 origin:
533 Point3d {
534 x: 0.0,
535 y: 0.0,
536 z: 0.0,
537 units: Some(UnitLength::Millimeters),
538 },
539 x_axis:
540 Point3d {
541 x: -1.0,
542 y: 0.0,
543 z: 0.0,
544 units: _,
545 },
546 y_axis:
547 Point3d {
548 x: 0.0,
549 y: 0.0,
550 z: 1.0,
551 units: _,
552 },
553 z_axis: _,
554 } => return PlaneData::NegXZ,
555 Self {
556 origin:
557 Point3d {
558 x: 0.0,
559 y: 0.0,
560 z: 0.0,
561 units: Some(UnitLength::Millimeters),
562 },
563 x_axis:
564 Point3d {
565 x: 0.0,
566 y: 1.0,
567 z: 0.0,
568 units: _,
569 },
570 y_axis:
571 Point3d {
572 x: 0.0,
573 y: 0.0,
574 z: 1.0,
575 units: _,
576 },
577 z_axis: _,
578 } => return PlaneData::YZ,
579 Self {
580 origin:
581 Point3d {
582 x: 0.0,
583 y: 0.0,
584 z: 0.0,
585 units: Some(UnitLength::Millimeters),
586 },
587 x_axis:
588 Point3d {
589 x: 0.0,
590 y: -1.0,
591 z: 0.0,
592 units: _,
593 },
594 y_axis:
595 Point3d {
596 x: 0.0,
597 y: 0.0,
598 z: 1.0,
599 units: _,
600 },
601 z_axis: _,
602 } => return PlaneData::NegYZ,
603 _ => {}
604 }
605 }
606
607 PlaneData::Plane(Self {
608 origin: self.origin,
609 x_axis: self.x_axis,
610 y_axis: self.y_axis,
611 z_axis: self.z_axis,
612 })
613 }
614
615 pub(crate) fn is_right_handed(&self) -> bool {
616 let lhs = self
619 .x_axis
620 .axes_cross_product(&self.y_axis)
621 .axes_dot_product(&self.z_axis);
622 let rhs_x = self.x_axis.axes_dot_product(&self.x_axis);
623 let rhs_y = self.y_axis.axes_dot_product(&self.y_axis);
624 let rhs_z = self.z_axis.axes_dot_product(&self.z_axis);
625 let rhs = (rhs_x * rhs_y * rhs_z).sqrt();
626 (lhs - rhs).abs() <= 0.0001
628 }
629
630 #[cfg(test)]
631 pub(crate) fn is_left_handed(&self) -> bool {
632 !self.is_right_handed()
633 }
634
635 pub(crate) fn make_right_handed(self) -> Self {
636 if self.is_right_handed() {
637 return self;
638 }
639 Self {
641 origin: self.origin,
642 x_axis: self.x_axis.negated(),
643 y_axis: self.y_axis,
644 z_axis: self.z_axis,
645 }
646 }
647}
648
649impl TryFrom<PlaneData> for PlaneInfo {
650 type Error = KclError;
651
652 fn try_from(value: PlaneData) -> Result<Self, Self::Error> {
653 let name = match value {
654 PlaneData::XY => PlaneName::Xy,
655 PlaneData::NegXY => PlaneName::NegXy,
656 PlaneData::XZ => PlaneName::Xz,
657 PlaneData::NegXZ => PlaneName::NegXz,
658 PlaneData::YZ => PlaneName::Yz,
659 PlaneData::NegYZ => PlaneName::NegYz,
660 PlaneData::Plane(info) => {
661 return Ok(info);
662 }
663 };
664
665 let info = DEFAULT_PLANE_INFO.get(&name).ok_or_else(|| {
666 KclError::new_internal(KclErrorDetails::new(
667 format!("Plane {name} not found"),
668 Default::default(),
669 ))
670 })?;
671
672 Ok(info.clone())
673 }
674}
675
676impl From<&PlaneData> for PlaneKind {
677 fn from(value: &PlaneData) -> Self {
678 match value {
679 PlaneData::XY => PlaneKind::XY,
680 PlaneData::NegXY => PlaneKind::XY,
681 PlaneData::XZ => PlaneKind::XZ,
682 PlaneData::NegXZ => PlaneKind::XZ,
683 PlaneData::YZ => PlaneKind::YZ,
684 PlaneData::NegYZ => PlaneKind::YZ,
685 PlaneData::Plane(_) => PlaneKind::Custom,
686 }
687 }
688}
689
690impl From<&PlaneInfo> for PlaneKind {
691 fn from(value: &PlaneInfo) -> Self {
692 let data = PlaneData::Plane(value.clone());
693 PlaneKind::from(&data)
694 }
695}
696
697impl From<PlaneInfo> for PlaneKind {
698 fn from(value: PlaneInfo) -> Self {
699 let data = PlaneData::Plane(value);
700 PlaneKind::from(&data)
701 }
702}
703
704impl Plane {
705 #[cfg(test)]
706 pub(crate) fn from_plane_data_skipping_engine(
707 value: PlaneData,
708 exec_state: &mut ExecState,
709 ) -> Result<Self, KclError> {
710 let id = exec_state.next_uuid();
711 let kind = PlaneKind::from(&value);
712 Ok(Plane {
713 id,
714 artifact_id: id.into(),
715 info: PlaneInfo::try_from(value)?,
716 object_id: None,
717 kind,
718 meta: vec![],
719 })
720 }
721
722 pub fn is_initialized(&self) -> bool {
724 self.object_id.is_some()
725 }
726
727 pub fn is_uninitialized(&self) -> bool {
729 !self.is_initialized()
730 }
731
732 pub fn is_standard(&self) -> bool {
734 match &self.kind {
735 PlaneKind::XY | PlaneKind::YZ | PlaneKind::XZ => true,
736 PlaneKind::Custom => false,
737 }
738 }
739
740 pub fn project(&self, point: Point3d) -> Point3d {
743 let v = point - self.info.origin;
744 let dot = v.axes_dot_product(&self.info.z_axis);
745
746 point - self.info.z_axis * dot
747 }
748}
749
750#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
752#[ts(export)]
753#[serde(rename_all = "camelCase")]
754pub struct Face {
755 pub id: uuid::Uuid,
757 pub artifact_id: ArtifactId,
759 pub object_id: ObjectId,
761 pub value: String,
763 pub x_axis: Point3d,
765 pub y_axis: Point3d,
767 pub parent_solid: FaceParentSolid,
769 pub units: UnitLength,
770 #[serde(skip)]
771 pub meta: Vec<Metadata>,
772}
773
774#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
776#[ts(export)]
777#[serde(rename_all = "camelCase")]
778pub struct FaceParentSolid {
779 pub solid_id: Uuid,
781 pub creator_sketch_id: Option<Uuid>,
783 #[serde(default, skip_serializing_if = "Vec::is_empty")]
785 pub edge_cut_ids: Vec<Uuid>,
786}
787
788impl FaceParentSolid {
789 pub(crate) fn sketch_or_solid_id(&self) -> Uuid {
790 self.creator_sketch_id.unwrap_or(self.solid_id)
791 }
792}
793
794#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
796#[ts(export)]
797#[serde(rename_all = "camelCase")]
798pub struct BoundedEdge {
799 pub face_id: uuid::Uuid,
801 pub edge_id: uuid::Uuid,
803 pub lower_bound: f32,
806 pub upper_bound: f32,
809}
810
811#[derive(Debug, Clone, Copy, Serialize, PartialEq, Eq, ts_rs::TS, FromStr, Display)]
813#[ts(export)]
814#[display(style = "camelCase")]
815pub enum PlaneKind {
816 #[serde(rename = "XY", alias = "xy")]
817 #[display("XY")]
818 XY,
819 #[serde(rename = "XZ", alias = "xz")]
820 #[display("XZ")]
821 XZ,
822 #[serde(rename = "YZ", alias = "yz")]
823 #[display("YZ")]
824 YZ,
825 #[display("Custom")]
827 Custom,
828}
829
830#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
831#[ts(export)]
832#[serde(tag = "type", rename_all = "camelCase")]
833pub struct Sketch {
834 pub id: uuid::Uuid,
836 pub paths: Vec<Path>,
840 #[serde(default, skip_serializing_if = "Vec::is_empty")]
842 pub inner_paths: Vec<Path>,
843 pub on: SketchSurface,
845 pub start: BasePath,
847 #[serde(default, skip_serializing_if = "IndexMap::is_empty")]
849 pub tags: IndexMap<String, TagIdentifier>,
850 pub artifact_id: ArtifactId,
853 #[ts(skip)]
854 pub original_id: uuid::Uuid,
855 #[serde(skip_serializing_if = "Option::is_none")]
860 #[ts(skip)]
861 pub origin_sketch_id: Option<uuid::Uuid>,
862 #[serde(skip)]
864 pub mirror: Option<uuid::Uuid>,
865 #[serde(skip)]
867 pub clone: Option<uuid::Uuid>,
868 #[serde(skip)]
870 #[ts(skip)]
871 pub synthetic_jump_path_ids: Vec<uuid::Uuid>,
872 pub units: UnitLength,
873 #[serde(skip)]
875 pub meta: Vec<Metadata>,
876 #[serde(
879 default = "ProfileClosed::explicitly",
880 skip_serializing_if = "ProfileClosed::is_explicitly"
881 )]
882 pub is_closed: ProfileClosed,
883}
884
885impl ProfileClosed {
886 #[expect(dead_code, reason = "it's not actually dead, it's called by serde")]
887 fn explicitly() -> Self {
888 Self::Explicitly
889 }
890
891 fn is_explicitly(&self) -> bool {
892 matches!(self, ProfileClosed::Explicitly)
893 }
894}
895
896#[derive(Debug, Serialize, Eq, PartialEq, Clone, Copy, Hash, Ord, PartialOrd, ts_rs::TS)]
898#[serde(rename_all = "camelCase")]
899pub enum ProfileClosed {
900 No,
902 Maybe,
904 Implicitly,
906 Explicitly,
908}
909
910impl Sketch {
911 pub(crate) fn build_sketch_mode_cmds(
914 &self,
915 exec_state: &mut ExecState,
916 inner_cmd: ModelingCmdReq,
917 ) -> Vec<ModelingCmdReq> {
918 vec![
919 ModelingCmdReq {
922 cmd: ModelingCmd::from(
923 mcmd::EnableSketchMode::builder()
924 .animated(false)
925 .ortho(false)
926 .entity_id(self.on.id())
927 .adjust_camera(false)
928 .maybe_planar_normal(if let SketchSurface::Plane(plane) = &self.on {
929 let normal = plane.info.x_axis.axes_cross_product(&plane.info.y_axis);
931 Some(normal.into())
932 } else {
933 None
934 })
935 .build(),
936 ),
937 cmd_id: exec_state.next_uuid().into(),
938 },
939 inner_cmd,
940 ModelingCmdReq {
941 cmd: ModelingCmd::SketchModeDisable(mcmd::SketchModeDisable::builder().build()),
942 cmd_id: exec_state.next_uuid().into(),
943 },
944 ]
945 }
946}
947
948#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
950#[ts(export)]
951#[serde(tag = "type", rename_all = "camelCase")]
952pub enum SketchSurface {
953 Plane(Box<Plane>),
954 Face(Box<Face>),
955}
956
957impl SketchSurface {
958 pub(crate) fn id(&self) -> uuid::Uuid {
959 match self {
960 SketchSurface::Plane(plane) => plane.id,
961 SketchSurface::Face(face) => face.id,
962 }
963 }
964 pub(crate) fn x_axis(&self) -> Point3d {
965 match self {
966 SketchSurface::Plane(plane) => plane.info.x_axis,
967 SketchSurface::Face(face) => face.x_axis,
968 }
969 }
970 pub(crate) fn y_axis(&self) -> Point3d {
971 match self {
972 SketchSurface::Plane(plane) => plane.info.y_axis,
973 SketchSurface::Face(face) => face.y_axis,
974 }
975 }
976
977 pub(crate) fn object_id(&self) -> Option<ObjectId> {
978 match self {
979 SketchSurface::Plane(plane) => plane.object_id,
980 SketchSurface::Face(face) => Some(face.object_id),
981 }
982 }
983
984 pub(crate) fn set_object_id(&mut self, object_id: ObjectId) {
985 match self {
986 SketchSurface::Plane(plane) => plane.object_id = Some(object_id),
987 SketchSurface::Face(face) => face.object_id = object_id,
988 }
989 }
990}
991
992#[derive(Debug, Clone, PartialEq)]
994pub enum Extrudable {
995 Sketch(Box<Sketch>),
997 Face(FaceTag),
999}
1000
1001impl Extrudable {
1002 pub async fn id_to_extrude(
1004 &self,
1005 exec_state: &mut ExecState,
1006 args: &Args,
1007 must_be_planar: bool,
1008 ) -> Result<uuid::Uuid, KclError> {
1009 match self {
1010 Extrudable::Sketch(sketch) => Ok(sketch.id),
1011 Extrudable::Face(face_tag) => face_tag.get_face_id_from_tag(exec_state, args, must_be_planar).await,
1012 }
1013 }
1014
1015 pub fn as_sketch(&self) -> Option<Sketch> {
1016 match self {
1017 Extrudable::Sketch(sketch) => Some((**sketch).clone()),
1018 Extrudable::Face(face_tag) => match face_tag.geometry() {
1019 Some(Geometry::Sketch(sketch)) => Some(sketch),
1020 Some(Geometry::Solid(solid)) => solid.sketch().cloned(),
1021 None => None,
1022 },
1023 }
1024 }
1025
1026 pub fn is_closed(&self) -> ProfileClosed {
1027 match self {
1028 Extrudable::Sketch(sketch) => sketch.is_closed,
1029 Extrudable::Face(face_tag) => match face_tag.geometry() {
1030 Some(Geometry::Sketch(sketch)) => sketch.is_closed,
1031 Some(Geometry::Solid(solid)) => solid
1032 .sketch()
1033 .map(|sketch| sketch.is_closed)
1034 .unwrap_or(ProfileClosed::Maybe),
1035 _ => ProfileClosed::Maybe,
1036 },
1037 }
1038 }
1039}
1040
1041impl From<Sketch> for Extrudable {
1042 fn from(value: Sketch) -> Self {
1043 Extrudable::Sketch(Box::new(value))
1044 }
1045}
1046
1047#[derive(Debug, Clone)]
1048pub(crate) enum GetTangentialInfoFromPathsResult {
1049 PreviousPoint([f64; 2]),
1050 Arc {
1051 center: [f64; 2],
1052 ccw: bool,
1053 },
1054 Circle {
1055 center: [f64; 2],
1056 ccw: bool,
1057 radius: f64,
1058 },
1059 Ellipse {
1060 center: [f64; 2],
1061 ccw: bool,
1062 major_axis: [f64; 2],
1063 _minor_radius: f64,
1064 },
1065}
1066
1067impl GetTangentialInfoFromPathsResult {
1068 pub(crate) fn tan_previous_point(&self, last_arc_end: [f64; 2]) -> [f64; 2] {
1069 match self {
1070 GetTangentialInfoFromPathsResult::PreviousPoint(p) => *p,
1071 GetTangentialInfoFromPathsResult::Arc { center, ccw } => {
1072 crate::std::utils::get_tangent_point_from_previous_arc(*center, *ccw, last_arc_end)
1073 }
1074 GetTangentialInfoFromPathsResult::Circle {
1077 center, radius, ccw, ..
1078 } => [center[0] + radius, center[1] + if *ccw { -1.0 } else { 1.0 }],
1079 GetTangentialInfoFromPathsResult::Ellipse {
1080 center,
1081 major_axis,
1082 ccw,
1083 ..
1084 } => [center[0] + major_axis[0], center[1] + if *ccw { -1.0 } else { 1.0 }],
1085 }
1086 }
1087}
1088
1089impl Sketch {
1090 pub(crate) fn add_tag(
1091 &mut self,
1092 tag: NodeRef<'_, TagDeclarator>,
1093 current_path: &Path,
1094 exec_state: &ExecState,
1095 surface: Option<&ExtrudeSurface>,
1096 ) {
1097 let mut tag_identifier: TagIdentifier = tag.into();
1098 let base = current_path.get_base();
1099 let mut sketch_copy = self.clone();
1100 sketch_copy.tags.clear();
1101 tag_identifier.info.push((
1102 exec_state.stack().current_epoch(),
1103 TagEngineInfo {
1104 id: base.geo_meta.id,
1105 geometry: Geometry::Sketch(sketch_copy),
1106 path: Some(current_path.clone()),
1107 surface: surface.cloned(),
1108 },
1109 ));
1110
1111 self.tags.insert(tag.name.to_string(), tag_identifier);
1112 }
1113
1114 pub(crate) fn merge_tags<'a>(&mut self, tags: impl Iterator<Item = &'a TagIdentifier>) {
1115 for t in tags {
1116 match self.tags.get_mut(&t.value) {
1117 Some(id) => {
1118 id.merge_info(t);
1119 }
1120 None => {
1121 self.tags.insert(t.value.clone(), t.clone());
1122 }
1123 }
1124 }
1125 }
1126
1127 pub(crate) fn latest_path(&self) -> Option<&Path> {
1129 self.paths.last()
1130 }
1131
1132 pub(crate) fn current_pen_position(&self) -> Result<Point2d, KclError> {
1136 let Some(path) = self.latest_path() else {
1137 return Ok(Point2d::new(self.start.to[0], self.start.to[1], self.start.units));
1138 };
1139
1140 let to = path.get_base().to;
1141 Ok(Point2d::new(to[0], to[1], path.get_base().units))
1142 }
1143
1144 pub(crate) fn get_tangential_info_from_paths(&self) -> GetTangentialInfoFromPathsResult {
1145 let Some(path) = self.latest_path() else {
1146 return GetTangentialInfoFromPathsResult::PreviousPoint(self.start.to);
1147 };
1148 path.get_tangential_info()
1149 }
1150}
1151
1152#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
1153#[ts(export)]
1154#[serde(tag = "type", rename_all = "camelCase")]
1155pub struct Solid {
1156 pub id: uuid::Uuid,
1158 #[serde(skip)]
1160 #[ts(skip)]
1161 pub value_id: uuid::Uuid,
1162 pub artifact_id: ArtifactId,
1164 pub value: Vec<ExtrudeSurface>,
1166 #[serde(rename = "sketch")]
1168 pub creator: SolidCreator,
1169 pub start_cap_id: Option<uuid::Uuid>,
1171 pub end_cap_id: Option<uuid::Uuid>,
1173 #[serde(default, skip_serializing_if = "Vec::is_empty")]
1175 pub edge_cuts: Vec<EdgeCut>,
1176 pub units: UnitLength,
1178 pub sectional: bool,
1180 #[serde(skip)]
1182 pub meta: Vec<Metadata>,
1183}
1184
1185#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
1186#[ts(export)]
1187pub struct CreatorFace {
1188 pub face_id: uuid::Uuid,
1190 pub solid_id: uuid::Uuid,
1192 pub sketch: Sketch,
1194}
1195
1196#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
1198#[ts(export)]
1199#[serde(tag = "creatorType", rename_all = "camelCase")]
1200pub enum SolidCreator {
1201 Sketch(Sketch),
1203 Face(CreatorFace),
1205 Procedural,
1207}
1208
1209impl Solid {
1210 pub fn sketch(&self) -> Option<&Sketch> {
1211 match &self.creator {
1212 SolidCreator::Sketch(sketch) => Some(sketch),
1213 SolidCreator::Face(CreatorFace { sketch, .. }) => Some(sketch),
1214 SolidCreator::Procedural => None,
1215 }
1216 }
1217
1218 pub fn sketch_mut(&mut self) -> Option<&mut Sketch> {
1219 match &mut self.creator {
1220 SolidCreator::Sketch(sketch) => Some(sketch),
1221 SolidCreator::Face(CreatorFace { sketch, .. }) => Some(sketch),
1222 SolidCreator::Procedural => None,
1223 }
1224 }
1225
1226 pub fn sketch_id(&self) -> Option<uuid::Uuid> {
1227 self.sketch().map(|sketch| sketch.id)
1228 }
1229
1230 pub fn original_id(&self) -> uuid::Uuid {
1231 self.sketch().map(|sketch| sketch.original_id).unwrap_or(self.id)
1232 }
1233
1234 pub(crate) fn get_all_edge_cut_ids(&self) -> impl Iterator<Item = uuid::Uuid> + '_ {
1235 self.edge_cuts.iter().map(|foc| foc.id())
1236 }
1237}
1238
1239impl From<&Solid> for FaceParentSolid {
1240 fn from(solid: &Solid) -> Self {
1241 Self {
1242 solid_id: solid.id,
1243 creator_sketch_id: solid.sketch_id(),
1244 edge_cut_ids: solid.get_all_edge_cut_ids().collect(),
1245 }
1246 }
1247}
1248
1249#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
1251#[ts(export)]
1252#[serde(tag = "type", rename_all = "camelCase")]
1253pub enum EdgeCut {
1254 Fillet {
1256 id: uuid::Uuid,
1258 radius: TyF64,
1259 #[serde(rename = "edgeId")]
1261 edge_id: uuid::Uuid,
1262 tag: Box<Option<TagNode>>,
1263 },
1264 Chamfer {
1266 id: uuid::Uuid,
1268 length: TyF64,
1269 #[serde(rename = "edgeId")]
1271 edge_id: uuid::Uuid,
1272 tag: Box<Option<TagNode>>,
1273 },
1274}
1275
1276impl EdgeCut {
1277 pub fn id(&self) -> uuid::Uuid {
1278 match self {
1279 EdgeCut::Fillet { id, .. } => *id,
1280 EdgeCut::Chamfer { id, .. } => *id,
1281 }
1282 }
1283
1284 pub fn set_id(&mut self, id: uuid::Uuid) {
1285 match self {
1286 EdgeCut::Fillet { id: i, .. } => *i = id,
1287 EdgeCut::Chamfer { id: i, .. } => *i = id,
1288 }
1289 }
1290
1291 pub fn edge_id(&self) -> uuid::Uuid {
1292 match self {
1293 EdgeCut::Fillet { edge_id, .. } => *edge_id,
1294 EdgeCut::Chamfer { edge_id, .. } => *edge_id,
1295 }
1296 }
1297
1298 pub fn set_edge_id(&mut self, id: uuid::Uuid) {
1299 match self {
1300 EdgeCut::Fillet { edge_id: i, .. } => *i = id,
1301 EdgeCut::Chamfer { edge_id: i, .. } => *i = id,
1302 }
1303 }
1304
1305 pub fn tag(&self) -> Option<TagNode> {
1306 match self {
1307 EdgeCut::Fillet { tag, .. } => *tag.clone(),
1308 EdgeCut::Chamfer { tag, .. } => *tag.clone(),
1309 }
1310 }
1311}
1312
1313#[derive(Debug, Serialize, PartialEq, Clone, Copy, ts_rs::TS)]
1314#[ts(export)]
1315pub struct Point2d {
1316 pub x: f64,
1317 pub y: f64,
1318 pub units: UnitLength,
1319}
1320
1321impl Point2d {
1322 pub const ZERO: Self = Self {
1323 x: 0.0,
1324 y: 0.0,
1325 units: UnitLength::Millimeters,
1326 };
1327
1328 pub fn new(x: f64, y: f64, units: UnitLength) -> Self {
1329 Self { x, y, units }
1330 }
1331
1332 pub fn into_x(self) -> TyF64 {
1333 TyF64::new(self.x, self.units.into())
1334 }
1335
1336 pub fn into_y(self) -> TyF64 {
1337 TyF64::new(self.y, self.units.into())
1338 }
1339
1340 pub fn ignore_units(self) -> [f64; 2] {
1341 [self.x, self.y]
1342 }
1343}
1344
1345#[derive(Debug, Deserialize, Serialize, PartialEq, Clone, Copy, ts_rs::TS, Default)]
1346#[ts(export)]
1347pub struct Point3d {
1348 pub x: f64,
1349 pub y: f64,
1350 pub z: f64,
1351 pub units: Option<UnitLength>,
1352}
1353
1354impl Point3d {
1355 pub const ZERO: Self = Self {
1356 x: 0.0,
1357 y: 0.0,
1358 z: 0.0,
1359 units: Some(UnitLength::Millimeters),
1360 };
1361
1362 pub fn new(x: f64, y: f64, z: f64, units: Option<UnitLength>) -> Self {
1363 Self { x, y, z, units }
1364 }
1365
1366 pub const fn is_zero(&self) -> bool {
1367 self.x == 0.0 && self.y == 0.0 && self.z == 0.0
1368 }
1369
1370 pub fn axes_cross_product(&self, other: &Self) -> Self {
1375 Self {
1376 x: self.y * other.z - self.z * other.y,
1377 y: self.z * other.x - self.x * other.z,
1378 z: self.x * other.y - self.y * other.x,
1379 units: None,
1380 }
1381 }
1382
1383 pub fn axes_dot_product(&self, other: &Self) -> f64 {
1388 let x = self.x * other.x;
1389 let y = self.y * other.y;
1390 let z = self.z * other.z;
1391 x + y + z
1392 }
1393
1394 pub fn normalize(&self) -> Self {
1395 let len = f64::sqrt(self.x * self.x + self.y * self.y + self.z * self.z);
1396 Point3d {
1397 x: self.x / len,
1398 y: self.y / len,
1399 z: self.z / len,
1400 units: None,
1401 }
1402 }
1403
1404 pub fn as_3_dims(&self) -> ([f64; 3], Option<UnitLength>) {
1405 let p = [self.x, self.y, self.z];
1406 let u = self.units;
1407 (p, u)
1408 }
1409
1410 pub(crate) fn negated(self) -> Self {
1411 Self {
1412 x: -self.x,
1413 y: -self.y,
1414 z: -self.z,
1415 units: self.units,
1416 }
1417 }
1418}
1419
1420impl From<[TyF64; 3]> for Point3d {
1421 fn from(p: [TyF64; 3]) -> Self {
1422 Self {
1423 x: p[0].n,
1424 y: p[1].n,
1425 z: p[2].n,
1426 units: p[0].ty.as_length(),
1427 }
1428 }
1429}
1430
1431impl From<Point3d> for Point3D {
1432 fn from(p: Point3d) -> Self {
1433 Self { x: p.x, y: p.y, z: p.z }
1434 }
1435}
1436
1437impl From<Point3d> for kittycad_modeling_cmds::shared::Point3d<LengthUnit> {
1438 fn from(p: Point3d) -> Self {
1439 if let Some(units) = p.units {
1440 Self {
1441 x: LengthUnit(adjust_length(units, p.x, UnitLength::Millimeters).0),
1442 y: LengthUnit(adjust_length(units, p.y, UnitLength::Millimeters).0),
1443 z: LengthUnit(adjust_length(units, p.z, UnitLength::Millimeters).0),
1444 }
1445 } else {
1446 Self {
1447 x: LengthUnit(p.x),
1448 y: LengthUnit(p.y),
1449 z: LengthUnit(p.z),
1450 }
1451 }
1452 }
1453}
1454
1455impl Add for Point3d {
1456 type Output = Point3d;
1457
1458 fn add(self, rhs: Self) -> Self::Output {
1459 Point3d {
1461 x: self.x + rhs.x,
1462 y: self.y + rhs.y,
1463 z: self.z + rhs.z,
1464 units: self.units,
1465 }
1466 }
1467}
1468
1469impl AddAssign for Point3d {
1470 fn add_assign(&mut self, rhs: Self) {
1471 *self = *self + rhs
1472 }
1473}
1474
1475impl Sub for Point3d {
1476 type Output = Point3d;
1477
1478 fn sub(self, rhs: Self) -> Self::Output {
1479 let (x, y, z) = if rhs.units != self.units
1480 && let Some(sunits) = self.units
1481 && let Some(runits) = rhs.units
1482 {
1483 (
1484 adjust_length(runits, rhs.x, sunits).0,
1485 adjust_length(runits, rhs.y, sunits).0,
1486 adjust_length(runits, rhs.z, sunits).0,
1487 )
1488 } else {
1489 (rhs.x, rhs.y, rhs.z)
1490 };
1491 Point3d {
1492 x: self.x - x,
1493 y: self.y - y,
1494 z: self.z - z,
1495 units: self.units,
1496 }
1497 }
1498}
1499
1500impl SubAssign for Point3d {
1501 fn sub_assign(&mut self, rhs: Self) {
1502 *self = *self - rhs
1503 }
1504}
1505
1506impl Mul<f64> for Point3d {
1507 type Output = Point3d;
1508
1509 fn mul(self, rhs: f64) -> Self::Output {
1510 Point3d {
1511 x: self.x * rhs,
1512 y: self.y * rhs,
1513 z: self.z * rhs,
1514 units: self.units,
1515 }
1516 }
1517}
1518
1519#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
1521#[ts(export)]
1522#[serde(rename_all = "camelCase")]
1523pub struct BasePath {
1524 #[ts(type = "[number, number]")]
1526 pub from: [f64; 2],
1527 #[ts(type = "[number, number]")]
1529 pub to: [f64; 2],
1530 pub units: UnitLength,
1531 pub tag: Option<TagNode>,
1533 #[serde(rename = "__geoMeta")]
1535 pub geo_meta: GeoMeta,
1536}
1537
1538impl BasePath {
1539 pub fn get_to(&self) -> [TyF64; 2] {
1540 let ty: NumericType = self.units.into();
1541 [TyF64::new(self.to[0], ty), TyF64::new(self.to[1], ty)]
1542 }
1543
1544 pub fn get_from(&self) -> [TyF64; 2] {
1545 let ty: NumericType = self.units.into();
1546 [TyF64::new(self.from[0], ty), TyF64::new(self.from[1], ty)]
1547 }
1548}
1549
1550#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
1552#[ts(export)]
1553#[serde(rename_all = "camelCase")]
1554pub struct GeoMeta {
1555 pub id: uuid::Uuid,
1557 #[serde(flatten)]
1559 pub metadata: Metadata,
1560}
1561
1562#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
1564#[ts(export)]
1565#[serde(tag = "type")]
1566pub enum Path {
1567 ToPoint {
1569 #[serde(flatten)]
1570 base: BasePath,
1571 },
1572 TangentialArcTo {
1574 #[serde(flatten)]
1575 base: BasePath,
1576 #[ts(type = "[number, number]")]
1578 center: [f64; 2],
1579 ccw: bool,
1581 },
1582 TangentialArc {
1584 #[serde(flatten)]
1585 base: BasePath,
1586 #[ts(type = "[number, number]")]
1588 center: [f64; 2],
1589 ccw: bool,
1591 },
1592 Circle {
1595 #[serde(flatten)]
1596 base: BasePath,
1597 #[ts(type = "[number, number]")]
1599 center: [f64; 2],
1600 radius: f64,
1602 ccw: bool,
1605 },
1606 CircleThreePoint {
1607 #[serde(flatten)]
1608 base: BasePath,
1609 #[ts(type = "[number, number]")]
1611 p1: [f64; 2],
1612 #[ts(type = "[number, number]")]
1614 p2: [f64; 2],
1615 #[ts(type = "[number, number]")]
1617 p3: [f64; 2],
1618 },
1619 ArcThreePoint {
1620 #[serde(flatten)]
1621 base: BasePath,
1622 #[ts(type = "[number, number]")]
1624 p1: [f64; 2],
1625 #[ts(type = "[number, number]")]
1627 p2: [f64; 2],
1628 #[ts(type = "[number, number]")]
1630 p3: [f64; 2],
1631 },
1632 Horizontal {
1634 #[serde(flatten)]
1635 base: BasePath,
1636 x: f64,
1638 },
1639 AngledLineTo {
1641 #[serde(flatten)]
1642 base: BasePath,
1643 x: Option<f64>,
1645 y: Option<f64>,
1647 },
1648 Base {
1650 #[serde(flatten)]
1651 base: BasePath,
1652 },
1653 Arc {
1655 #[serde(flatten)]
1656 base: BasePath,
1657 center: [f64; 2],
1659 radius: f64,
1661 ccw: bool,
1663 },
1664 Ellipse {
1665 #[serde(flatten)]
1666 base: BasePath,
1667 center: [f64; 2],
1668 major_axis: [f64; 2],
1669 minor_radius: f64,
1670 ccw: bool,
1671 },
1672 Conic {
1674 #[serde(flatten)]
1675 base: BasePath,
1676 },
1677 Bezier {
1679 #[serde(flatten)]
1680 base: BasePath,
1681 #[ts(type = "[number, number]")]
1683 control1: [f64; 2],
1684 #[ts(type = "[number, number]")]
1686 control2: [f64; 2],
1687 },
1688}
1689
1690impl Path {
1691 pub fn get_id(&self) -> uuid::Uuid {
1692 match self {
1693 Path::ToPoint { base } => base.geo_meta.id,
1694 Path::Horizontal { base, .. } => base.geo_meta.id,
1695 Path::AngledLineTo { base, .. } => base.geo_meta.id,
1696 Path::Base { base } => base.geo_meta.id,
1697 Path::TangentialArcTo { base, .. } => base.geo_meta.id,
1698 Path::TangentialArc { base, .. } => base.geo_meta.id,
1699 Path::Circle { base, .. } => base.geo_meta.id,
1700 Path::CircleThreePoint { base, .. } => base.geo_meta.id,
1701 Path::Arc { base, .. } => base.geo_meta.id,
1702 Path::ArcThreePoint { base, .. } => base.geo_meta.id,
1703 Path::Ellipse { base, .. } => base.geo_meta.id,
1704 Path::Conic { base, .. } => base.geo_meta.id,
1705 Path::Bezier { base, .. } => base.geo_meta.id,
1706 }
1707 }
1708
1709 pub fn set_id(&mut self, id: uuid::Uuid) {
1710 match self {
1711 Path::ToPoint { base } => base.geo_meta.id = id,
1712 Path::Horizontal { base, .. } => base.geo_meta.id = id,
1713 Path::AngledLineTo { base, .. } => base.geo_meta.id = id,
1714 Path::Base { base } => base.geo_meta.id = id,
1715 Path::TangentialArcTo { base, .. } => base.geo_meta.id = id,
1716 Path::TangentialArc { base, .. } => base.geo_meta.id = id,
1717 Path::Circle { base, .. } => base.geo_meta.id = id,
1718 Path::CircleThreePoint { base, .. } => base.geo_meta.id = id,
1719 Path::Arc { base, .. } => base.geo_meta.id = id,
1720 Path::ArcThreePoint { base, .. } => base.geo_meta.id = id,
1721 Path::Ellipse { base, .. } => base.geo_meta.id = id,
1722 Path::Conic { base, .. } => base.geo_meta.id = id,
1723 Path::Bezier { base, .. } => base.geo_meta.id = id,
1724 }
1725 }
1726
1727 pub fn get_tag(&self) -> Option<TagNode> {
1728 match self {
1729 Path::ToPoint { base } => base.tag.clone(),
1730 Path::Horizontal { base, .. } => base.tag.clone(),
1731 Path::AngledLineTo { base, .. } => base.tag.clone(),
1732 Path::Base { base } => base.tag.clone(),
1733 Path::TangentialArcTo { base, .. } => base.tag.clone(),
1734 Path::TangentialArc { base, .. } => base.tag.clone(),
1735 Path::Circle { base, .. } => base.tag.clone(),
1736 Path::CircleThreePoint { base, .. } => base.tag.clone(),
1737 Path::Arc { base, .. } => base.tag.clone(),
1738 Path::ArcThreePoint { base, .. } => base.tag.clone(),
1739 Path::Ellipse { base, .. } => base.tag.clone(),
1740 Path::Conic { base, .. } => base.tag.clone(),
1741 Path::Bezier { base, .. } => base.tag.clone(),
1742 }
1743 }
1744
1745 pub fn get_base(&self) -> &BasePath {
1746 match self {
1747 Path::ToPoint { base } => base,
1748 Path::Horizontal { base, .. } => base,
1749 Path::AngledLineTo { base, .. } => base,
1750 Path::Base { base } => base,
1751 Path::TangentialArcTo { base, .. } => base,
1752 Path::TangentialArc { base, .. } => base,
1753 Path::Circle { base, .. } => base,
1754 Path::CircleThreePoint { base, .. } => base,
1755 Path::Arc { base, .. } => base,
1756 Path::ArcThreePoint { base, .. } => base,
1757 Path::Ellipse { base, .. } => base,
1758 Path::Conic { base, .. } => base,
1759 Path::Bezier { base, .. } => base,
1760 }
1761 }
1762
1763 pub fn get_from(&self) -> [TyF64; 2] {
1765 let p = &self.get_base().from;
1766 let ty: NumericType = self.get_base().units.into();
1767 [TyF64::new(p[0], ty), TyF64::new(p[1], ty)]
1768 }
1769
1770 pub fn get_to(&self) -> [TyF64; 2] {
1772 let p = &self.get_base().to;
1773 let ty: NumericType = self.get_base().units.into();
1774 [TyF64::new(p[0], ty), TyF64::new(p[1], ty)]
1775 }
1776
1777 pub fn start_point_components(&self) -> ([f64; 2], NumericType) {
1779 let p = &self.get_base().from;
1780 let ty: NumericType = self.get_base().units.into();
1781 (*p, ty)
1782 }
1783
1784 pub fn end_point_components(&self) -> ([f64; 2], NumericType) {
1786 let p = &self.get_base().to;
1787 let ty: NumericType = self.get_base().units.into();
1788 (*p, ty)
1789 }
1790
1791 pub fn length(&self) -> Option<TyF64> {
1794 let n = match self {
1795 Self::ToPoint { .. } | Self::Base { .. } | Self::Horizontal { .. } | Self::AngledLineTo { .. } => {
1796 Some(linear_distance(&self.get_base().from, &self.get_base().to))
1797 }
1798 Self::TangentialArc {
1799 base: _,
1800 center,
1801 ccw: _,
1802 }
1803 | Self::TangentialArcTo {
1804 base: _,
1805 center,
1806 ccw: _,
1807 } => {
1808 let radius = linear_distance(&self.get_base().from, center);
1811 debug_assert_eq!(radius, linear_distance(&self.get_base().to, center));
1812 Some(linear_distance(&self.get_base().from, &self.get_base().to))
1814 }
1815 Self::Circle { radius, .. } => Some(TAU * radius),
1816 Self::CircleThreePoint { .. } => {
1817 let circle_center = crate::std::utils::calculate_circle_from_3_points([
1818 self.get_base().from,
1819 self.get_base().to,
1820 self.get_base().to,
1821 ]);
1822 let radius = linear_distance(
1823 &[circle_center.center[0], circle_center.center[1]],
1824 &self.get_base().from,
1825 );
1826 Some(TAU * radius)
1827 }
1828 Self::Arc { .. } => {
1829 Some(linear_distance(&self.get_base().from, &self.get_base().to))
1831 }
1832 Self::ArcThreePoint { .. } => {
1833 Some(linear_distance(&self.get_base().from, &self.get_base().to))
1835 }
1836 Self::Ellipse { .. } => {
1837 None
1839 }
1840 Self::Conic { .. } => {
1841 None
1843 }
1844 Self::Bezier { .. } => {
1845 None
1847 }
1848 };
1849 n.map(|n| TyF64::new(n, self.get_base().units.into()))
1850 }
1851
1852 pub fn get_base_mut(&mut self) -> &mut BasePath {
1853 match self {
1854 Path::ToPoint { base } => base,
1855 Path::Horizontal { base, .. } => base,
1856 Path::AngledLineTo { base, .. } => base,
1857 Path::Base { base } => base,
1858 Path::TangentialArcTo { base, .. } => base,
1859 Path::TangentialArc { base, .. } => base,
1860 Path::Circle { base, .. } => base,
1861 Path::CircleThreePoint { base, .. } => base,
1862 Path::Arc { base, .. } => base,
1863 Path::ArcThreePoint { base, .. } => base,
1864 Path::Ellipse { base, .. } => base,
1865 Path::Conic { base, .. } => base,
1866 Path::Bezier { base, .. } => base,
1867 }
1868 }
1869
1870 pub(crate) fn get_tangential_info(&self) -> GetTangentialInfoFromPathsResult {
1871 match self {
1872 Path::TangentialArc { center, ccw, .. }
1873 | Path::TangentialArcTo { center, ccw, .. }
1874 | Path::Arc { center, ccw, .. } => GetTangentialInfoFromPathsResult::Arc {
1875 center: *center,
1876 ccw: *ccw,
1877 },
1878 Path::ArcThreePoint { p1, p2, p3, .. } => {
1879 let circle = crate::std::utils::calculate_circle_from_3_points([*p1, *p2, *p3]);
1880 GetTangentialInfoFromPathsResult::Arc {
1881 center: circle.center,
1882 ccw: crate::std::utils::is_points_ccw(&[*p1, *p2, *p3]) > 0,
1883 }
1884 }
1885 Path::Circle {
1886 center, ccw, radius, ..
1887 } => GetTangentialInfoFromPathsResult::Circle {
1888 center: *center,
1889 ccw: *ccw,
1890 radius: *radius,
1891 },
1892 Path::CircleThreePoint { p1, p2, p3, .. } => {
1893 let circle = crate::std::utils::calculate_circle_from_3_points([*p1, *p2, *p3]);
1894 let center_point = [circle.center[0], circle.center[1]];
1895 GetTangentialInfoFromPathsResult::Circle {
1896 center: center_point,
1897 ccw: true,
1899 radius: circle.radius,
1900 }
1901 }
1902 Path::Ellipse {
1904 center,
1905 major_axis,
1906 minor_radius,
1907 ccw,
1908 ..
1909 } => GetTangentialInfoFromPathsResult::Ellipse {
1910 center: *center,
1911 major_axis: *major_axis,
1912 _minor_radius: *minor_radius,
1913 ccw: *ccw,
1914 },
1915 Path::Conic { .. }
1916 | Path::ToPoint { .. }
1917 | Path::Horizontal { .. }
1918 | Path::AngledLineTo { .. }
1919 | Path::Base { .. }
1920 | Path::Bezier { .. } => {
1921 let base = self.get_base();
1922 GetTangentialInfoFromPathsResult::PreviousPoint(base.from)
1923 }
1924 }
1925 }
1926
1927 pub(crate) fn is_straight_line(&self) -> bool {
1929 matches!(self, Path::AngledLineTo { .. } | Path::ToPoint { .. })
1930 }
1931}
1932
1933#[rustfmt::skip]
1935fn linear_distance(
1936 [x0, y0]: &[f64; 2],
1937 [x1, y1]: &[f64; 2]
1938) -> f64 {
1939 let y_sq = (y1 - y0).squared();
1940 let x_sq = (x1 - x0).squared();
1941 (y_sq + x_sq).sqrt()
1942}
1943
1944#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
1946#[ts(export)]
1947#[serde(tag = "type", rename_all = "camelCase")]
1948pub enum ExtrudeSurface {
1949 ExtrudePlane(ExtrudePlane),
1951 ExtrudeArc(ExtrudeArc),
1952 Chamfer(ChamferSurface),
1953 Fillet(FilletSurface),
1954}
1955
1956#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
1958#[ts(export)]
1959#[serde(rename_all = "camelCase")]
1960pub struct ChamferSurface {
1961 pub face_id: uuid::Uuid,
1963 pub tag: Option<Node<TagDeclarator>>,
1965 #[serde(flatten)]
1967 pub geo_meta: GeoMeta,
1968}
1969
1970#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
1972#[ts(export)]
1973#[serde(rename_all = "camelCase")]
1974pub struct FilletSurface {
1975 pub face_id: uuid::Uuid,
1977 pub tag: Option<Node<TagDeclarator>>,
1979 #[serde(flatten)]
1981 pub geo_meta: GeoMeta,
1982}
1983
1984#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
1986#[ts(export)]
1987#[serde(rename_all = "camelCase")]
1988pub struct ExtrudePlane {
1989 pub face_id: uuid::Uuid,
1991 pub tag: Option<Node<TagDeclarator>>,
1993 #[serde(flatten)]
1995 pub geo_meta: GeoMeta,
1996}
1997
1998#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2000#[ts(export)]
2001#[serde(rename_all = "camelCase")]
2002pub struct ExtrudeArc {
2003 pub face_id: uuid::Uuid,
2005 pub tag: Option<Node<TagDeclarator>>,
2007 #[serde(flatten)]
2009 pub geo_meta: GeoMeta,
2010}
2011
2012impl ExtrudeSurface {
2013 pub fn get_id(&self) -> uuid::Uuid {
2014 match self {
2015 ExtrudeSurface::ExtrudePlane(ep) => ep.geo_meta.id,
2016 ExtrudeSurface::ExtrudeArc(ea) => ea.geo_meta.id,
2017 ExtrudeSurface::Fillet(f) => f.geo_meta.id,
2018 ExtrudeSurface::Chamfer(c) => c.geo_meta.id,
2019 }
2020 }
2021
2022 pub fn face_id(&self) -> uuid::Uuid {
2023 match self {
2024 ExtrudeSurface::ExtrudePlane(ep) => ep.face_id,
2025 ExtrudeSurface::ExtrudeArc(ea) => ea.face_id,
2026 ExtrudeSurface::Fillet(f) => f.face_id,
2027 ExtrudeSurface::Chamfer(c) => c.face_id,
2028 }
2029 }
2030
2031 pub fn set_face_id(&mut self, face_id: uuid::Uuid) {
2032 match self {
2033 ExtrudeSurface::ExtrudePlane(ep) => ep.face_id = face_id,
2034 ExtrudeSurface::ExtrudeArc(ea) => ea.face_id = face_id,
2035 ExtrudeSurface::Fillet(f) => f.face_id = face_id,
2036 ExtrudeSurface::Chamfer(c) => c.face_id = face_id,
2037 }
2038 }
2039
2040 pub fn set_surface_tag(&mut self, tag: &TagNode) {
2041 match self {
2042 ExtrudeSurface::ExtrudePlane(extrude_plane) => extrude_plane.tag = Some(tag.clone()),
2043 ExtrudeSurface::ExtrudeArc(extrude_arc) => extrude_arc.tag = Some(tag.clone()),
2044 ExtrudeSurface::Chamfer(chamfer) => chamfer.tag = Some(tag.clone()),
2045 ExtrudeSurface::Fillet(fillet) => fillet.tag = Some(tag.clone()),
2046 }
2047 }
2048
2049 pub fn get_tag(&self) -> Option<Node<TagDeclarator>> {
2050 match self {
2051 ExtrudeSurface::ExtrudePlane(ep) => ep.tag.clone(),
2052 ExtrudeSurface::ExtrudeArc(ea) => ea.tag.clone(),
2053 ExtrudeSurface::Fillet(f) => f.tag.clone(),
2054 ExtrudeSurface::Chamfer(c) => c.tag.clone(),
2055 }
2056 }
2057}
2058
2059#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Serialize, ts_rs::TS)]
2060pub struct SketchVarId(pub usize);
2061
2062impl SketchVarId {
2063 pub const INVALID: Self = Self(usize::MAX);
2064
2065 pub fn to_constraint_id(self, range: SourceRange) -> Result<ezpz::Id, KclError> {
2066 self.0.try_into().map_err(|_| {
2067 KclError::new_type(KclErrorDetails::new(
2068 "Cannot convert to constraint ID since the sketch variable ID is too large".to_owned(),
2069 vec![range],
2070 ))
2071 })
2072 }
2073}
2074
2075#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2076#[ts(export_to = "Geometry.ts")]
2077#[serde(rename_all = "camelCase")]
2078pub struct SketchVar {
2079 pub id: SketchVarId,
2080 pub initial_value: f64,
2081 pub ty: NumericType,
2082 #[serde(skip)]
2083 pub meta: Vec<Metadata>,
2084}
2085
2086impl SketchVar {
2087 pub fn initial_value_to_solver_units(
2088 &self,
2089 exec_state: &mut ExecState,
2090 source_range: SourceRange,
2091 description: &str,
2092 ) -> Result<TyF64, KclError> {
2093 let x_initial_value = KclValue::Number {
2094 value: self.initial_value,
2095 ty: self.ty,
2096 meta: vec![source_range.into()],
2097 };
2098 let normalized_value =
2099 normalize_to_solver_distance_unit(&x_initial_value, source_range, exec_state, description)?;
2100 normalized_value.as_ty_f64().ok_or_else(|| {
2101 let message = format!(
2102 "Expected number after coercion, but found {}",
2103 normalized_value.human_friendly_type()
2104 );
2105 debug_assert!(false, "{}", &message);
2106 KclError::new_internal(KclErrorDetails::new(message, vec![source_range]))
2107 })
2108 }
2109}
2110
2111#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2112#[ts(export_to = "Geometry.ts")]
2113#[serde(tag = "type")]
2114pub enum UnsolvedExpr {
2115 Known(TyF64),
2116 Unknown(SketchVarId),
2117}
2118
2119impl UnsolvedExpr {
2120 pub fn var(&self) -> Option<SketchVarId> {
2121 match self {
2122 UnsolvedExpr::Known(_) => None,
2123 UnsolvedExpr::Unknown(id) => Some(*id),
2124 }
2125 }
2126}
2127
2128pub type UnsolvedPoint2dExpr = [UnsolvedExpr; 2];
2129
2130#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2131#[ts(export_to = "Geometry.ts")]
2132#[serde(rename_all = "camelCase")]
2133pub struct ConstrainablePoint2d {
2134 pub vars: crate::front::Point2d<SketchVarId>,
2135 pub object_id: ObjectId,
2136}
2137
2138#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2139#[ts(export_to = "Geometry.ts")]
2140pub enum ConstrainablePoint2dOrOrigin {
2141 Point(ConstrainablePoint2d),
2142 Origin,
2143}
2144
2145#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2146#[ts(export_to = "Geometry.ts")]
2147#[serde(rename_all = "camelCase")]
2148pub struct ConstrainableLine2d {
2149 pub vars: [crate::front::Point2d<SketchVarId>; 2],
2150 pub object_id: ObjectId,
2151}
2152
2153#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2154#[ts(export_to = "Geometry.ts")]
2155#[serde(rename_all = "camelCase")]
2156pub struct UnsolvedSegment {
2157 pub id: Uuid,
2159 pub object_id: ObjectId,
2160 pub kind: UnsolvedSegmentKind,
2161 #[serde(skip_serializing_if = "Option::is_none")]
2162 pub tag: Option<TagIdentifier>,
2163 #[serde(skip)]
2164 pub node_path: Option<NodePath>,
2165 #[serde(skip)]
2166 pub meta: Vec<Metadata>,
2167}
2168
2169#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2170#[ts(export_to = "Geometry.ts")]
2171#[serde(rename_all = "camelCase")]
2172pub enum UnsolvedSegmentKind {
2173 Point {
2174 position: UnsolvedPoint2dExpr,
2175 ctor: Box<PointCtor>,
2176 },
2177 Line {
2178 start: UnsolvedPoint2dExpr,
2179 end: UnsolvedPoint2dExpr,
2180 ctor: Box<LineCtor>,
2181 start_object_id: ObjectId,
2182 end_object_id: ObjectId,
2183 construction: bool,
2184 },
2185 Arc {
2186 start: UnsolvedPoint2dExpr,
2187 end: UnsolvedPoint2dExpr,
2188 center: UnsolvedPoint2dExpr,
2189 ctor: Box<ArcCtor>,
2190 start_object_id: ObjectId,
2191 end_object_id: ObjectId,
2192 center_object_id: ObjectId,
2193 construction: bool,
2194 },
2195 Circle {
2196 start: UnsolvedPoint2dExpr,
2197 center: UnsolvedPoint2dExpr,
2198 ctor: Box<CircleCtor>,
2199 start_object_id: ObjectId,
2200 center_object_id: ObjectId,
2201 construction: bool,
2202 },
2203 ControlPointSpline {
2204 controls: Vec<UnsolvedPoint2dExpr>,
2205 ctor: Box<ControlPointSplineCtor>,
2206 control_object_ids: Vec<ObjectId>,
2207 control_polygon_edge_object_ids: Vec<ObjectId>,
2208 degree: u32,
2209 construction: bool,
2210 },
2211}
2212
2213impl UnsolvedSegmentKind {
2214 pub fn human_friendly_kind_with_article(&self) -> &'static str {
2217 match self {
2218 Self::Point { .. } => "a Point",
2219 Self::Line { .. } => "a Line",
2220 Self::Arc { .. } => "an Arc",
2221 Self::Circle { .. } => "a Circle",
2222 Self::ControlPointSpline { .. } => "a Control Point Spline",
2223 }
2224 }
2225}
2226
2227#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2228#[ts(export_to = "Geometry.ts")]
2229#[serde(rename_all = "camelCase")]
2230pub struct Segment {
2231 pub id: Uuid,
2233 pub object_id: ObjectId,
2234 pub kind: SegmentKind,
2235 pub surface: SketchSurface,
2236 pub sketch_id: Uuid,
2238 #[serde(skip)]
2239 #[ts(skip)]
2240 pub sketch: Option<Arc<Sketch>>,
2241 #[serde(skip_serializing_if = "Option::is_none")]
2242 pub tag: Option<TagIdentifier>,
2243 #[serde(skip)]
2244 pub node_path: Option<NodePath>,
2245 #[serde(skip)]
2246 pub meta: Vec<Metadata>,
2247}
2248
2249impl Segment {
2250 pub fn is_construction(&self) -> bool {
2251 match &self.kind {
2252 SegmentKind::Point { .. } => true,
2253 SegmentKind::Line { construction, .. } => *construction,
2254 SegmentKind::Arc { construction, .. } => *construction,
2255 SegmentKind::Circle { construction, .. } => *construction,
2256 SegmentKind::ControlPointSpline { construction, .. } => *construction,
2257 }
2258 }
2259}
2260
2261#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2262#[ts(export_to = "Geometry.ts")]
2263#[serde(rename_all = "camelCase")]
2264pub enum SegmentKind {
2265 Point {
2266 position: [TyF64; 2],
2267 ctor: Box<PointCtor>,
2268 #[serde(skip_serializing_if = "Option::is_none")]
2269 freedom: Option<Freedom>,
2270 },
2271 Line {
2272 start: [TyF64; 2],
2273 end: [TyF64; 2],
2274 ctor: Box<LineCtor>,
2275 start_object_id: ObjectId,
2276 end_object_id: ObjectId,
2277 #[serde(skip_serializing_if = "Option::is_none")]
2278 start_freedom: Option<Freedom>,
2279 #[serde(skip_serializing_if = "Option::is_none")]
2280 end_freedom: Option<Freedom>,
2281 construction: bool,
2282 },
2283 Arc {
2284 start: [TyF64; 2],
2285 end: [TyF64; 2],
2286 center: [TyF64; 2],
2287 ctor: Box<ArcCtor>,
2288 start_object_id: ObjectId,
2289 end_object_id: ObjectId,
2290 center_object_id: ObjectId,
2291 #[serde(skip_serializing_if = "Option::is_none")]
2292 start_freedom: Option<Freedom>,
2293 #[serde(skip_serializing_if = "Option::is_none")]
2294 end_freedom: Option<Freedom>,
2295 #[serde(skip_serializing_if = "Option::is_none")]
2296 center_freedom: Option<Freedom>,
2297 construction: bool,
2298 },
2299 Circle {
2300 start: [TyF64; 2],
2301 center: [TyF64; 2],
2302 ctor: Box<CircleCtor>,
2303 start_object_id: ObjectId,
2304 center_object_id: ObjectId,
2305 #[serde(skip_serializing_if = "Option::is_none")]
2306 start_freedom: Option<Freedom>,
2307 #[serde(skip_serializing_if = "Option::is_none")]
2308 center_freedom: Option<Freedom>,
2309 construction: bool,
2310 },
2311 ControlPointSpline {
2312 controls: Vec<[TyF64; 2]>,
2313 ctor: Box<ControlPointSplineCtor>,
2314 control_object_ids: Vec<ObjectId>,
2315 control_polygon_edge_object_ids: Vec<ObjectId>,
2316 #[serde(skip_serializing_if = "Vec::is_empty")]
2317 control_freedoms: Vec<Option<Freedom>>,
2318 degree: u32,
2319 construction: bool,
2320 },
2321}
2322
2323#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2324#[ts(export_to = "Geometry.ts")]
2325#[serde(rename_all = "camelCase")]
2326pub struct AbstractSegment {
2327 pub repr: SegmentRepr,
2328 #[serde(skip)]
2329 pub meta: Vec<Metadata>,
2330}
2331
2332#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2333pub enum SegmentRepr {
2334 Unsolved { segment: Box<UnsolvedSegment> },
2335 Solved { segment: Box<Segment> },
2336}
2337
2338#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2339#[ts(export_to = "Geometry.ts")]
2340#[serde(rename_all = "camelCase")]
2341pub struct SketchConstraint {
2342 pub kind: SketchConstraintKind,
2343 #[serde(skip)]
2344 pub meta: Vec<Metadata>,
2345}
2346
2347#[derive(Debug, Clone, Serialize, PartialEq, ts_rs::TS)]
2348#[ts(export_to = "Geometry.ts")]
2349#[serde(rename_all = "camelCase")]
2350pub enum SketchConstraintKind {
2351 Angle {
2352 line0: ConstrainableLine2d,
2353 line1: ConstrainableLine2d,
2354 },
2355 Distance {
2356 points: [ConstrainablePoint2dOrOrigin; 2],
2357 #[serde(rename = "labelPosition")]
2358 #[serde(skip_serializing_if = "Option::is_none")]
2359 #[ts(rename = "labelPosition")]
2360 #[ts(optional)]
2361 label_position: Option<ApiPoint2d<Number>>,
2362 },
2363 PointLineDistance {
2364 point: ConstrainablePoint2dOrOrigin,
2365 line: ConstrainableLine2d,
2366 input_object_ids: [Option<ObjectId>; 2],
2367 #[serde(rename = "labelPosition")]
2368 #[serde(skip_serializing_if = "Option::is_none")]
2369 #[ts(rename = "labelPosition")]
2370 #[ts(optional)]
2371 label_position: Option<ApiPoint2d<Number>>,
2372 },
2373 LineLineDistance {
2374 line0: ConstrainableLine2d,
2375 line1: ConstrainableLine2d,
2376 input_object_ids: [ObjectId; 2],
2377 #[serde(rename = "labelPosition")]
2378 #[serde(skip_serializing_if = "Option::is_none")]
2379 #[ts(rename = "labelPosition")]
2380 #[ts(optional)]
2381 label_position: Option<ApiPoint2d<Number>>,
2382 },
2383 PointCircularDistance {
2384 point: ConstrainablePoint2dOrOrigin,
2385 center: ConstrainablePoint2d,
2386 start: ConstrainablePoint2d,
2387 end: Option<ConstrainablePoint2d>,
2388 input_object_ids: [Option<ObjectId>; 2],
2389 #[serde(rename = "labelPosition")]
2390 #[serde(skip_serializing_if = "Option::is_none")]
2391 #[ts(rename = "labelPosition")]
2392 #[ts(optional)]
2393 label_position: Option<ApiPoint2d<Number>>,
2394 },
2395 LineCircularDistance {
2396 line: ConstrainableLine2d,
2397 center: ConstrainablePoint2d,
2398 start: ConstrainablePoint2d,
2399 end: Option<ConstrainablePoint2d>,
2400 input_object_ids: [ObjectId; 2],
2401 #[serde(rename = "labelPosition")]
2402 #[serde(skip_serializing_if = "Option::is_none")]
2403 #[ts(rename = "labelPosition")]
2404 #[ts(optional)]
2405 label_position: Option<ApiPoint2d<Number>>,
2406 },
2407 CircularCircularDistance {
2408 center0: ConstrainablePoint2d,
2409 start0: ConstrainablePoint2d,
2410 end0: Option<ConstrainablePoint2d>,
2411 center1: ConstrainablePoint2d,
2412 start1: ConstrainablePoint2d,
2413 end1: Option<ConstrainablePoint2d>,
2414 input_object_ids: [ObjectId; 2],
2415 #[serde(rename = "labelPosition")]
2416 #[serde(skip_serializing_if = "Option::is_none")]
2417 #[ts(rename = "labelPosition")]
2418 #[ts(optional)]
2419 label_position: Option<ApiPoint2d<Number>>,
2420 },
2421 Radius {
2422 points: [ConstrainablePoint2d; 2],
2423 #[serde(rename = "labelPosition")]
2424 #[serde(skip_serializing_if = "Option::is_none")]
2425 #[ts(rename = "labelPosition")]
2426 #[ts(optional)]
2427 label_position: Option<ApiPoint2d<Number>>,
2428 },
2429 Diameter {
2430 points: [ConstrainablePoint2d; 2],
2431 #[serde(rename = "labelPosition")]
2432 #[serde(skip_serializing_if = "Option::is_none")]
2433 #[ts(rename = "labelPosition")]
2434 #[ts(optional)]
2435 label_position: Option<ApiPoint2d<Number>>,
2436 },
2437 HorizontalDistance {
2438 points: [ConstrainablePoint2dOrOrigin; 2],
2439 #[serde(rename = "labelPosition")]
2440 #[serde(skip_serializing_if = "Option::is_none")]
2441 #[ts(rename = "labelPosition")]
2442 #[ts(optional)]
2443 label_position: Option<ApiPoint2d<Number>>,
2444 },
2445 VerticalDistance {
2446 points: [ConstrainablePoint2dOrOrigin; 2],
2447 #[serde(rename = "labelPosition")]
2448 #[serde(skip_serializing_if = "Option::is_none")]
2449 #[ts(rename = "labelPosition")]
2450 #[ts(optional)]
2451 label_position: Option<ApiPoint2d<Number>>,
2452 },
2453}
2454
2455impl SketchConstraintKind {
2456 pub fn name(&self) -> &'static str {
2457 match self {
2458 SketchConstraintKind::Angle { .. } => "angle",
2459 SketchConstraintKind::Distance { .. } => "distance",
2460 SketchConstraintKind::PointLineDistance { .. } => "distance",
2461 SketchConstraintKind::LineLineDistance { .. } => "distance",
2462 SketchConstraintKind::PointCircularDistance { .. } => "distance",
2463 SketchConstraintKind::LineCircularDistance { .. } => "distance",
2464 SketchConstraintKind::CircularCircularDistance { .. } => "distance",
2465 SketchConstraintKind::Radius { .. } => "radius",
2466 SketchConstraintKind::Diameter { .. } => "diameter",
2467 SketchConstraintKind::HorizontalDistance { .. } => "horizontalDistance",
2468 SketchConstraintKind::VerticalDistance { .. } => "verticalDistance",
2469 }
2470 }
2471}