From ba79fbcad88ab8242681d2f0e803f53f08cc3305 Mon Sep 17 00:00:00 2001 From: Adam Chalmers Date: Mon, 4 Nov 2024 15:55:43 -0600 Subject: [PATCH] New KCL stdlib function 'segEnd' Part of --- docs/kcl/index.md | 1 + docs/kcl/segEnd.md | 53 + docs/kcl/std.json | 903 ++++++++++++++++++ src/wasm-lib/kcl/src/std/args.rs | 53 +- src/wasm-lib/kcl/src/std/assert.rs | 12 +- src/wasm-lib/kcl/src/std/convert.rs | 2 +- src/wasm-lib/kcl/src/std/math.rs | 2 +- src/wasm-lib/kcl/src/std/mod.rs | 1 + src/wasm-lib/kcl/src/std/segment.rs | 46 + .../serial_test_example_segment_end0.png | Bin 0 -> 66894 bytes 10 files changed, 1040 insertions(+), 33 deletions(-) create mode 100644 docs/kcl/segEnd.md create mode 100644 src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_end0.png diff --git a/docs/kcl/index.md b/docs/kcl/index.md index 75f3c19d4b..5235ea42ff 100644 --- a/docs/kcl/index.md +++ b/docs/kcl/index.md @@ -84,6 +84,7 @@ layout: manual * [`rem`](kcl/rem) * [`revolve`](kcl/revolve) * [`segAng`](kcl/segAng) +* [`segEnd`](kcl/segEnd) * [`segEndX`](kcl/segEndX) * [`segEndY`](kcl/segEndY) * [`segLen`](kcl/segLen) diff --git a/docs/kcl/segEnd.md b/docs/kcl/segEnd.md new file mode 100644 index 0000000000..85ae38b7fc --- /dev/null +++ b/docs/kcl/segEnd.md @@ -0,0 +1,53 @@ +--- +title: "segEnd" +excerpt: "Compute the ending point of the provided line segment." +layout: manual +--- + +Compute the ending point of the provided line segment. + + + +```js +segEnd(tag: TagIdentifier) -> [number] +``` + + +### Arguments + +| Name | Type | Description | Required | +|----------|------|-------------|----------| +| `tag` | [`TagIdentifier`](/docs/kcl/types#tag-identifier) | | Yes | + +### Returns + +`[number]` + + +### Examples + +```js +w = 15 +cube = startSketchAt([0, 0]) + |> line([w, 0], %, $line1) + |> line([0, w], %, $line2) + |> line([-w, 0], %, $line3) + |> line([0, -w], %, $line4) + |> close(%) + |> extrude(5, %) + +fn cylinder = (radius, tag) => { + return startSketchAt([0, 0]) + |> circle({ radius: radius, center: segEnd(tag) }, %) + |> extrude(radius, %) +} + +cylinder(1, line1) +cylinder(2, line2) +cylinder(3, line3) +cylinder(4, line4) +``` + +![Rendered example of segEnd 0]() + + diff --git a/docs/kcl/std.json b/docs/kcl/std.json index 2a49b6a5fc..12f9a36fd0 100644 --- a/docs/kcl/std.json +++ b/docs/kcl/std.json @@ -160120,6 +160120,909 @@ "exampleSketch = startSketchOn('XZ')\n |> startProfileAt([0, 0], %)\n |> line([10, 0], %)\n |> line([5, 10], %, $seg01)\n |> line([-10, 0], %)\n |> angledLine([segAng(seg01), 10], %)\n |> line([-10, 0], %)\n |> angledLine([segAng(seg01), -15], %)\n |> close(%)\n\nexample = extrude(4, exampleSketch)" ] }, + { + "name": "segEnd", + "summary": "Compute the ending point of the provided line segment.", + "description": "", + "tags": [], + "args": [ + { + "name": "tag", + "type": "TagIdentifier", + "schema": { + "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", + "title": "TagIdentifier", + "type": "object", + "required": [ + "__meta", + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "info": { + "allOf": [ + { + "$ref": "#/components/schemas/TagEngineInfo" + } + ], + "nullable": true + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + }, + "definitions": { + "TagEngineInfo": { + "description": "Engine information for a tag.", + "type": "object", + "required": [ + "id", + "sketch" + ], + "properties": { + "id": { + "description": "The id of the tagged object.", + "type": "string", + "format": "uuid" + }, + "sketch": { + "description": "The sketch the tag is on.", + "type": "string", + "format": "uuid" + }, + "path": { + "description": "The path the tag is on.", + "allOf": [ + { + "$ref": "#/components/schemas/Path" + } + ], + "nullable": true + }, + "surface": { + "description": "The surface information for the tag.", + "allOf": [ + { + "$ref": "#/components/schemas/ExtrudeSurface" + } + ], + "nullable": true + } + } + }, + "Path": { + "description": "A path.", + "oneOf": [ + { + "description": "A path that goes to a point.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "ToPoint" + ] + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment that goes to a point", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TangentialArcTo" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TangentialArc" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "a complete arc", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "radius", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Circle" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "radius": { + "description": "the arc's radius", + "type": "number", + "format": "double" + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A path that is horizontal.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type", + "x" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Horizontal" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "An angled line to.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "AngledLineTo" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "y": { + "description": "The y coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A base path.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Base" + ] + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A circular arc, not necessarily tangential to the current point.", + "type": "object", + "required": [ + "__geoMeta", + "center", + "from", + "radius", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Arc" + ] + }, + "center": { + "description": "Center of the circle that this arc is drawn on.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "radius": { + "description": "Radius of the circle that this arc is drawn on.", + "type": "number", + "format": "double" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + } + ] + }, + "TagDeclarator": { + "type": "object", + "required": [ + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + "GeoMeta": { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + "SourceRange": { + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "ExtrudeSurface": { + "description": "An extrude surface.", + "oneOf": [ + { + "description": "An extrude plane.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "extrudePlane" + ] + }, + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "An extruded arc.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "extrudeArc" + ] + }, + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + }, + "faceId": { + "description": "The id for the chamfer surface.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "fillet" + ] + }, + "faceId": { + "description": "The id for the fillet surface.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + } + ] + }, + "Metadata": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + } + } + }, + "required": true + } + ], + "returnValue": { + "name": "", + "type": "[number]", + "schema": { + "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", + "title": "Array_size_2_of_double", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "required": true + }, + "unpublished": false, + "deprecated": false, + "examples": [ + "w = 15\ncube = startSketchAt([0, 0])\n |> line([w, 0], %, $line1)\n |> line([0, w], %, $line2)\n |> line([-w, 0], %, $line3)\n |> line([0, -w], %, $line4)\n |> close(%)\n |> extrude(5, %)\n\nfn cylinder = (radius, tag) => {\n return startSketchAt([0, 0])\n |> circle({ radius: radius, center: segEnd(tag) }, %)\n |> extrude(radius, %)\n}\n\ncylinder(1, line1)\ncylinder(2, line2)\ncylinder(3, line3)\ncylinder(4, line4)" + ] + }, { "name": "segEndX", "summary": "Compute the ending point of the provided line segment along the 'x' axis.", diff --git a/src/wasm-lib/kcl/src/std/args.rs b/src/wasm-lib/kcl/src/std/args.rs index b3ef1c442f..34c233c5ab 100644 --- a/src/wasm-lib/kcl/src/std/args.rs +++ b/src/wasm-lib/kcl/src/std/args.rs @@ -181,47 +181,39 @@ impl Args { Ok(()) } - fn make_user_val_from_json(&self, j: serde_json::Value) -> Result { - Ok(KclValue::UserVal(crate::executor::UserVal { + fn make_user_val_from_json(&self, j: serde_json::Value) -> KclValue { + KclValue::UserVal(crate::executor::UserVal { value: j, meta: vec![Metadata { source_range: self.source_range, }], - })) + }) } - pub(crate) fn make_null_user_val(&self) -> Result { + pub(crate) fn make_null_user_val(&self) -> KclValue { self.make_user_val_from_json(serde_json::Value::Null) } - pub(crate) fn make_user_val_from_i64(&self, n: i64) -> Result { + pub(crate) fn make_user_val_from_i64(&self, n: i64) -> KclValue { self.make_user_val_from_json(serde_json::Value::Number(serde_json::Number::from(n))) } pub(crate) fn make_user_val_from_f64(&self, f: f64) -> Result { - self.make_user_val_from_json(serde_json::Value::Number(serde_json::Number::from_f64(f).ok_or_else( - || { - KclError::Type(KclErrorDetails { - message: format!("Failed to convert `{}` to a number", f), - source_ranges: vec![self.source_range], - }) - }, - )?)) + f64_to_jnum(f, vec![self.source_range]).map(|x| self.make_user_val_from_json(x)) + } + + pub(crate) fn make_user_val_from_point(&self, p: [f64; 2]) -> Result { + let x = f64_to_jnum(p[0], vec![self.source_range])?; + let y = f64_to_jnum(p[1], vec![self.source_range])?; + let array = serde_json::Value::Array(vec![x, y]); + Ok(self.make_user_val_from_json(array)) } pub(crate) fn make_user_val_from_f64_array(&self, f: Vec) -> Result { - let mut arr = Vec::new(); - for n in f { - arr.push(serde_json::Value::Number(serde_json::Number::from_f64(n).ok_or_else( - || { - KclError::Type(KclErrorDetails { - message: format!("Failed to convert `{}` to a number", n), - source_ranges: vec![self.source_range], - }) - }, - )?)); - } - self.make_user_val_from_json(serde_json::Value::Array(arr)) + f.into_iter() + .map(|n| f64_to_jnum(n, vec![self.source_range])) + .collect::, _>>() + .map(|arr| self.make_user_val_from_json(serde_json::Value::Array(arr))) } pub(crate) fn get_number(&self) -> Result { @@ -750,3 +742,14 @@ impl<'a> FromKclValue<'a> for SketchSurface { } } } + +fn f64_to_jnum(f: f64, source_ranges: Vec) -> Result { + serde_json::Number::from_f64(f) + .ok_or_else(|| { + KclError::Type(KclErrorDetails { + message: format!("Failed to convert `{f}` to a number"), + source_ranges, + }) + }) + .map(serde_json::Value::Number) +} diff --git a/src/wasm-lib/kcl/src/std/assert.rs b/src/wasm-lib/kcl/src/std/assert.rs index f4cb15c359..5a71efe73c 100644 --- a/src/wasm-lib/kcl/src/std/assert.rs +++ b/src/wasm-lib/kcl/src/std/assert.rs @@ -24,7 +24,7 @@ async fn _assert(value: bool, message: &str, args: &Args) -> Result<(), KclError pub async fn assert(_exec_state: &mut ExecState, args: Args) -> Result { let (data, description): (bool, String) = args.get_data()?; inner_assert(data, &description, &args).await?; - args.make_null_user_val() + Ok(args.make_null_user_val()) } /// Check a value at runtime, and raise an error if the argument provided @@ -44,7 +44,7 @@ async fn inner_assert(data: bool, message: &str, args: &Args) -> Result<(), KclE pub async fn assert_lt(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, description): (f64, f64, String) = args.get_data()?; inner_assert_lt(left, right, &description, &args).await?; - args.make_null_user_val() + Ok(args.make_null_user_val()) } /// Check that a numerical value is less than to another at runtime, @@ -63,7 +63,7 @@ async fn inner_assert_lt(left: f64, right: f64, message: &str, args: &Args) -> R pub async fn assert_gt(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, description): (f64, f64, String) = args.get_data()?; inner_assert_gt(left, right, &description, &args).await?; - args.make_null_user_val() + Ok(args.make_null_user_val()) } /// Check that a numerical value equals another at runtime, @@ -96,7 +96,7 @@ async fn inner_assert_equal(left: f64, right: f64, epsilon: f64, message: &str, pub async fn assert_equal(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, epsilon, description): (f64, f64, f64, String) = args.get_data()?; inner_assert_equal(left, right, epsilon, &description, &args).await?; - args.make_null_user_val() + Ok(args.make_null_user_val()) } /// Check that a numerical value is greater than another at runtime, @@ -115,7 +115,7 @@ async fn inner_assert_gt(left: f64, right: f64, message: &str, args: &Args) -> R pub async fn assert_lte(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, description): (f64, f64, String) = args.get_data()?; inner_assert_lte(left, right, &description, &args).await?; - args.make_null_user_val() + Ok(args.make_null_user_val()) } /// Check that a numerical value is less than or equal to another at runtime, @@ -135,7 +135,7 @@ async fn inner_assert_lte(left: f64, right: f64, message: &str, args: &Args) -> pub async fn assert_gte(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, description): (f64, f64, String) = args.get_data()?; inner_assert_gte(left, right, &description, &args).await?; - args.make_null_user_val() + Ok(args.make_null_user_val()) } /// Check that a numerical value is greater than or equal to another at runtime, diff --git a/src/wasm-lib/kcl/src/std/convert.rs b/src/wasm-lib/kcl/src/std/convert.rs index 69e8729ff2..777eb0fe50 100644 --- a/src/wasm-lib/kcl/src/std/convert.rs +++ b/src/wasm-lib/kcl/src/std/convert.rs @@ -34,7 +34,7 @@ pub async fn int(_exec_state: &mut ExecState, args: Args) -> Result Result Result { + let tag: TagIdentifier = args.get_data()?; + let result = inner_segment_end(&tag, exec_state, args.clone())?; + + args.make_user_val_from_point(result) +} + +/// Compute the ending point of the provided line segment. +/// +/// ```no_run +/// w = 15 +/// cube = startSketchAt([0, 0]) +/// |> line([w, 0], %, $line1) +/// |> line([0, w], %, $line2) +/// |> line([-w, 0], %, $line3) +/// |> line([0, -w], %, $line4) +/// |> close(%) +/// |> extrude(5, %) +/// +/// fn cylinder = (radius, tag) => { +/// return startSketchAt([0, 0]) +/// |> circle({ radius: radius, center: segEnd(tag) }, %) +/// |> extrude(radius, %) +/// } +/// +/// cylinder(1, line1) +/// cylinder(2, line2) +/// cylinder(3, line3) +/// cylinder(4, line4) +/// ``` +#[stdlib { + name = "segEnd", +}] +fn inner_segment_end(tag: &TagIdentifier, exec_state: &mut ExecState, args: Args) -> Result<[f64; 2], KclError> { + let line = args.get_tag_engine_info(exec_state, tag)?; + let path = line.path.clone().ok_or_else(|| { + KclError::Type(KclErrorDetails { + message: format!("Expected a line segment with a path, found `{:?}`", line), + source_ranges: vec![args.source_range], + }) + })?; + + Ok(path.get_base().to) +} + /// Returns the segment end of x. pub async fn segment_end_x(exec_state: &mut ExecState, args: Args) -> Result { let tag: TagIdentifier = args.get_data()?; diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_end0.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_end0.png new file mode 100644 index 0000000000000000000000000000000000000000..f4a3e8c7ec593825cc275f10c02864eede3b83e9 GIT binary patch literal 66894 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