Compare commits
	
		
			22 Commits
		
	
	
		
			andrewvarg
			...
			kcl-82
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 6358eed7e4 | |||
| fe581ff1d2 | |||
| b301fbba22 | |||
| 0c702e4bab | |||
| 25b9a34640 | |||
| b2152a5684 | |||
| 832bf77c92 | |||
| acb43fc82c | |||
| 7486d25cf1 | |||
| 1a4a030671 | |||
| 3049d939e1 | |||
| ad9822e8ac | |||
| aae34cf1e5 | |||
| d6278cf075 | |||
| 4159cc0047 | |||
| 3936017f10 | |||
| 2b0ced179a | |||
| c2f6ce065d | |||
| b3bdc35da2 | |||
| 8fe4f67a29 | |||
| c6b1d11700 | |||
| 2ef84382a6 | 
							
								
								
									
										3
									
								
								.github/CODEOWNERS
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,3 @@ | ||||
| * @KittyCAD/frontend | ||||
| /src/ @KittyCAD/frontend | ||||
| /rust/ @KittyCAD/kcl | ||||
							
								
								
									
										6
									
								
								.github/workflows/cargo-fmt.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -31,15 +31,15 @@ jobs: | ||||
|       - name: Use correct Rust toolchain | ||||
|         shell: bash | ||||
|         run: | | ||||
|           [ -e rust-toolchain.toml ] || cp rust/rust-toolchain.toml ./ | ||||
|           cp .github/workflows/nightly-rust-toolchain.toml rust-toolchain.toml | ||||
|       - name: Install rust | ||||
|         uses: actions-rust-lang/setup-rust-toolchain@v1 | ||||
|         with: | ||||
|           cache-workspaces: rust | ||||
|           components: rustfmt | ||||
|  | ||||
|       - name: Run cargo fmt | ||||
|       - name: Run nightly cargo fmt | ||||
|         run: | | ||||
|           cd rust | ||||
|           cargo fmt -- --check | ||||
|           cargo +nightly fmt -- --check | ||||
|         shell: bash | ||||
|  | ||||
							
								
								
									
										15
									
								
								.github/workflows/e2e-tests.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -95,7 +95,8 @@ jobs: | ||||
|         shell: bash | ||||
|         run: npm run build:wasm | ||||
|  | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|       - name: Upload compiled wasm artifacts | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: prepared-wasm | ||||
|           path: | | ||||
| @ -176,7 +177,8 @@ jobs: | ||||
|           CI_SUITE: e2e:snapshots | ||||
|           TARGET: web | ||||
|  | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|       - name: Upload playwright report | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         if: ${{ !cancelled() }} | ||||
|         with: | ||||
|           name: playwright-report-snapshot-${{ github.sha }} | ||||
| @ -290,7 +292,8 @@ jobs: | ||||
|           CI_SUITE: e2e:web | ||||
|           TARGET: web | ||||
|  | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|       - name: Upload playwright report | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         if: ${{ !cancelled() && (success() || failure()) }} | ||||
|         with: | ||||
|           name: playwright-report-web-${{ env.OS_NAME }}-${{ matrix.shardIndex }}-${{ github.sha }} | ||||
| @ -415,7 +418,8 @@ jobs: | ||||
|           CI_SUITE: e2e:desktop | ||||
|           TARGET: desktop | ||||
|  | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|       - name: Upload test report | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         if: always() | ||||
|         with: | ||||
|           name: test-results-desktop-${{ env.OS_NAME }}-${{ matrix.shardIndex }}-${{ github.sha }} | ||||
| @ -424,7 +428,8 @@ jobs: | ||||
|           retention-days: 30 | ||||
|           overwrite: true | ||||
|  | ||||
|       - uses: actions/upload-artifact@v4 | ||||
|       - name: Upload playwright report | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         if: always() | ||||
|         with: | ||||
|           name: playwright-report-desktop-${{ env.OS_NAME }}-${{ matrix.shardIndex }}-${{ github.sha }} | ||||
|  | ||||
							
								
								
									
										3
									
								
								.github/workflows/nightly-rust-toolchain.toml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,3 @@ | ||||
| [toolchain] | ||||
| channel = "nightly" | ||||
| components = ["rustfmt"] | ||||
| @ -44,7 +44,7 @@ detail on importing geometry. | ||||
|  | ||||
| Tags are used to give a name (tag) to a specific path. | ||||
|  | ||||
| ### `TagDeclarator` | ||||
| ### Tag declarations - `TagDecl` | ||||
|  | ||||
| The syntax for declaring a tag is `$myTag` you would use it in the following | ||||
| way: | ||||
| @ -67,24 +67,28 @@ startSketchOn(XZ) | ||||
|   |> close() | ||||
| ``` | ||||
|  | ||||
| ### `TagIdentifier` | ||||
| When a function requires declaring a new tag (using the `$` syntax), the argument has type [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl). | ||||
|  | ||||
| As per the example above you can use the tag identifier to get a reference to the  | ||||
| tagged object. The syntax for this is `myTag`. | ||||
| ### Tag identifiers | ||||
|  | ||||
| In the example above we use the tag identifier to get the angle of the segment | ||||
| `segAng(rectangleSegmentA001)`. | ||||
| A tag created using a tag declarator can be used by writing its name without the `$`, e.g., `myTag`. | ||||
| Where necessary to disambiguate from tag declarations, we call these tag identifiers. | ||||
|  | ||||
| ### `Start` | ||||
| In the example above we use the tag identifier `rectangleSegmentA001` to get the angle of the segment | ||||
| using `segAng(rectangleSegmentA001)`. | ||||
|  | ||||
| There is a special tag, `START` (with type `Start`, although under the cover, it's a string) | ||||
| for identifying the face of a solid which was the start of an extrusion (i.e., the surface which | ||||
| is extruded). | ||||
| Tags can identify either an edge or face of a solid, or a line or other edge of a sketch. Functions | ||||
| which take a tag identifier as an argument will use either [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) (for the edge of a | ||||
| solid or sketch) or [`TaggedFace`](/docs/kcl-std/types/std-types-TaggedFace). | ||||
|  | ||||
| ### `End` | ||||
| If a line in a sketch is tagged and then the sketch is extruded, the tag is a `TaggedEdge` before | ||||
| extrusion and a `TaggedFace` after extrusion. | ||||
|  | ||||
| #### `START` and `END` | ||||
|  | ||||
| [`START`](/docs/kcl-std/consts/std-START) and [`END`](/docs/kcl-std/consts/std-END) are special tags | ||||
| for identifying the starting and ending faces of an extruded solid. | ||||
|  | ||||
| There is a special tag, `END` (with type `End`, although under the cover, it's a string) | ||||
| for identifying the face of a solid which was finishes an extrusion. | ||||
|  | ||||
| ### Tag Scope | ||||
|  | ||||
|  | ||||
| @ -8,9 +8,13 @@ layout: manual | ||||
| Identifies the ending face of an extrusion. I.e., the new face created by an extrusion. | ||||
|  | ||||
| ```kcl | ||||
| END: string = 'end' | ||||
| END: TaggedFace | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`TaggedFace`](/docs/kcl-std/types/std-types-TaggedFace) - A tag which references a face of a solid, including the distinguished tags `START` and `END`. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -8,9 +8,13 @@ layout: manual | ||||
| Identifies the starting face of an extrusion. I.e., the face which is extruded. | ||||
|  | ||||
| ```kcl | ||||
| START: string = 'start' | ||||
| START: TaggedFace | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`TaggedFace`](/docs/kcl-std/types/std-types-TaggedFace) - A tag which references a face of a solid, including the distinguished tags `START` and `END`. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -8,9 +8,13 @@ layout: manual | ||||
| The X-axis (can be used in both 2d and 3d contexts). | ||||
|  | ||||
| ```kcl | ||||
| X | ||||
| X: Axis3d | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`Axis3d`](/docs/kcl-std/types/std-types-Axis3d) - An abstract and infinite line in 3d space. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -8,9 +8,13 @@ layout: manual | ||||
| An abstract 3d plane aligned with the X and Y axes. Its normal is the positive Z axis. | ||||
|  | ||||
| ```kcl | ||||
| XY | ||||
| XY: Plane | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`Plane`](/docs/kcl-std/types/std-types-Plane) - An abstract plane. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -8,9 +8,13 @@ layout: manual | ||||
| An abstract 3d plane aligned with the X and Z axes. Its normal is the negative Y axis. | ||||
|  | ||||
| ```kcl | ||||
| XZ | ||||
| XZ: Plane | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`Plane`](/docs/kcl-std/types/std-types-Plane) - An abstract plane. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -8,9 +8,13 @@ layout: manual | ||||
| The Y-axis (can be used in both 2d and 3d contexts). | ||||
|  | ||||
| ```kcl | ||||
| Y | ||||
| Y: Axis3d | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`Axis3d`](/docs/kcl-std/types/std-types-Axis3d) - An abstract and infinite line in 3d space. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -8,9 +8,13 @@ layout: manual | ||||
| An abstract 3d plane aligned with the Y and Z axes. Its normal is the positive X axis. | ||||
|  | ||||
| ```kcl | ||||
| YZ | ||||
| YZ: Plane | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`Plane`](/docs/kcl-std/types/std-types-Plane) - An abstract plane. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -8,9 +8,13 @@ layout: manual | ||||
| The 3D Z-axis. | ||||
|  | ||||
| ```kcl | ||||
| Z | ||||
| Z: Axis3d | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`Axis3d`](/docs/kcl-std/types/std-types-Axis3d) - An abstract and infinite line in 3d space. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -13,6 +13,10 @@ E: number = 2.71828182845904523536028747135266250_ | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`number`](/docs/kcl-std/types/std-types-number) - A number. | ||||
|  | ||||
| ### Examples | ||||
|  | ||||
| ```kcl | ||||
|  | ||||
| @ -11,7 +11,7 @@ The value of `pi`, Archimedes’ constant (π). | ||||
| PI: number(_?) = 3.14159265358979323846264338327950288_? | ||||
| ``` | ||||
|  | ||||
| `PI` is a number and is technically a ratio, so you might expect it to have type `number(_)`. | ||||
| `PI` is a number and is technically a ratio, so you might expect it to have type [`number(_)`](/docs/kcl-std/types/std-types-number). | ||||
| However, `PI` is nearly always used for converting between different units - usually degrees to or | ||||
| from radians. Therefore, `PI` is treated a bit specially by KCL and always has unknown units. This | ||||
| means that if you use `PI`, you will need to give KCL some extra information about the units of numbers. | ||||
| @ -19,6 +19,10 @@ Usually you should use type ascription on the result of calculations, e.g., `(2 | ||||
| It is better to use `units::toRadians` or `units::toDegrees` to convert between angles with | ||||
| different units where possible. | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`number(_?)`](/docs/kcl-std/types/std-types-number) - A number. | ||||
|  | ||||
| ### Examples | ||||
|  | ||||
| ```kcl | ||||
|  | ||||
| @ -13,6 +13,10 @@ TAU: number = 6.28318530717958647692528676655900577_ | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`number`](/docs/kcl-std/types/std-types-number) - A number. | ||||
|  | ||||
| ### Examples | ||||
|  | ||||
| ```kcl | ||||
|  | ||||
| @ -13,4 +13,8 @@ sweep::SKETCH_PLANE: string = 'sketchPlane' | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`string`](/docs/kcl-std/types/std-types-string) - A sequence of characters | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -13,4 +13,8 @@ sweep::TRAJECTORY: string = 'trajectoryCurve' | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`string`](/docs/kcl-std/types/std-types-string) - A sequence of characters | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -13,4 +13,8 @@ turns::HALF_TURN: number(deg) = 180deg | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`number(deg)`](/docs/kcl-std/types/std-types-number) - A number. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -13,4 +13,8 @@ turns::QUARTER_TURN: number(deg) = 90deg | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`number(deg)`](/docs/kcl-std/types/std-types-number) - A number. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -13,4 +13,8 @@ turns::THREE_QUARTER_TURN: number(deg) = 270deg | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`number(deg)`](/docs/kcl-std/types/std-types-number) - A number. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -8,9 +8,13 @@ layout: manual | ||||
| No turn, zero degrees/radians. | ||||
|  | ||||
| ```kcl | ||||
| turns::ZERO | ||||
| turns::ZERO: number(Angle) | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Type | ||||
|  | ||||
| [`number(Angle)`](/docs/kcl-std/types/std-types-number) - A number. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -16,7 +16,7 @@ angledLine( | ||||
|   lengthY?: number(Length), | ||||
|   endAbsoluteX?: number(Length), | ||||
|   endAbsoluteY?: number(Length), | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -33,7 +33,7 @@ angledLine( | ||||
| | `lengthY` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | Draw the line this distance along the Y axis. Only one of `length`, `lengthX`, `lengthY`, `endAbsoluteX`, `endAbsoluteY` can be given. | No | | ||||
| | `endAbsoluteX` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | Draw the line along the given angle until it reaches this point along the X axis. Only one of `length`, `lengthX`, `lengthY`, `endAbsoluteX`, `endAbsoluteY` can be given. | No | | ||||
| | `endAbsoluteY` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | Draw the line along the given angle until it reaches this point along the Y axis. Only one of `length`, `lengthX`, `lengthY`, `endAbsoluteX`, `endAbsoluteY` can be given. | No | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this line. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this line. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -11,9 +11,9 @@ Draw an angled line from the current origin, constructing a line segment such th | ||||
| angledLineThatIntersects( | ||||
|   @sketch: Sketch, | ||||
|   angle: number(Angle), | ||||
|   intersectTag: tag, | ||||
|   intersectTag: TaggedEdge, | ||||
|   offset?: number(Length), | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -25,9 +25,9 @@ angledLineThatIntersects( | ||||
| |----------|------|-------------|----------| | ||||
| | `sketch` | [`Sketch`](/docs/kcl-std/types/std-types-Sketch) | Which sketch should this path be added to? | Yes | | ||||
| | `angle` | [`number(Angle)`](/docs/kcl-std/types/std-types-number) | Which angle should the line be drawn at? | Yes | | ||||
| | `intersectTag` | [`tag`](/docs/kcl-std/types/std-types-tag) | The tag of the line to intersect with. | Yes | | ||||
| | `intersectTag` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The tag of the line to intersect with. | Yes | | ||||
| | `offset` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | The offset from the intersecting line. | No | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this line. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this line. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -16,7 +16,7 @@ arc( | ||||
|   diameter?: number(Length), | ||||
|   interiorAbsolute?: Point2d, | ||||
|   endAbsolute?: Point2d, | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -40,7 +40,7 @@ for to construct your shape, you're likely looking for tangentialArc. | ||||
| | `diameter` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | How large should the circle be? Incompatible with `radius`. | No | | ||||
| | `interiorAbsolute` | [`Point2d`](/docs/kcl-std/types/std-types-Point2d) | Any point between the arc's start and end? Requires `endAbsolute`. Incompatible with `angleStart` or `angleEnd`. | No | | ||||
| | `endAbsolute` | [`Point2d`](/docs/kcl-std/types/std-types-Point2d) | Where should this arc end? Requires `interiorAbsolute`. Incompatible with `angleStart` or `angleEnd`. | No | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this arc. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this arc. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -16,7 +16,7 @@ bezierCurve( | ||||
|   control1Absolute?: Point2d, | ||||
|   control2Absolute?: Point2d, | ||||
|   endAbsolute?: Point2d, | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -33,7 +33,7 @@ bezierCurve( | ||||
| | `control1Absolute` | [`Point2d`](/docs/kcl-std/types/std-types-Point2d) | First control point for the cubic. Absolute point. | No | | ||||
| | `control2Absolute` | [`Point2d`](/docs/kcl-std/types/std-types-Point2d) | Second control point for the cubic. Absolute point. | No | | ||||
| | `endAbsolute` | [`Point2d`](/docs/kcl-std/types/std-types-Point2d) | Coordinate on the plane at which this line should end. | No | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this line. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this line. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -13,7 +13,7 @@ circle( | ||||
|   center: Point2d, | ||||
|   radius?: number(Length), | ||||
|   diameter?: number(Length), | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -27,7 +27,7 @@ circle( | ||||
| | `center` | [`Point2d`](/docs/kcl-std/types/std-types-Point2d) | The center of the circle. | Yes | | ||||
| | `radius` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | The radius of the circle. Incompatible with `diameter`. | No | | ||||
| | `diameter` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | The diameter of the circle. Incompatible with `radius`. | No | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this circle. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this circle. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -13,7 +13,7 @@ circleThreePoint( | ||||
|   p1: Point2d, | ||||
|   p2: Point2d, | ||||
|   p3: Point2d, | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -27,7 +27,7 @@ circleThreePoint( | ||||
| | `p1` | [`Point2d`](/docs/kcl-std/types/std-types-Point2d) | 1st point to derive the circle. | Yes | | ||||
| | `p2` | [`Point2d`](/docs/kcl-std/types/std-types-Point2d) | 2nd point to derive the circle. | Yes | | ||||
| | `p3` | [`Point2d`](/docs/kcl-std/types/std-types-Point2d) | 3rd point to derive the circle. | Yes | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Identifier for the circle to reference elsewhere. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Identifier for the circle to reference elsewhere. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -10,7 +10,7 @@ Construct a line segment from the current origin back to the profile's origin, e | ||||
| ```kcl | ||||
| close( | ||||
|   @sketch: Sketch, | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -24,7 +24,7 @@ starting point. | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `sketch` | [`Sketch`](/docs/kcl-std/types/std-types-Sketch) | The sketch you want to close. | Yes | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this line. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this line. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Get the shared edge between two faces. | ||||
|  | ||||
| ```kcl | ||||
| getCommonEdge(faces: [tag; 2]): Edge | ||||
| getCommonEdge(faces: [TaggedFace; 2]): Edge | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ getCommonEdge(faces: [tag; 2]): Edge | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `faces` | `[tag; 2]` | The tags of the faces you want to find the common edge between. | Yes | | ||||
| | `faces` | `[TaggedFace; 2]` | The tags of the faces you want to find the common edge between. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Get the next adjacent edge to the edge given. | ||||
|  | ||||
| ```kcl | ||||
| getNextAdjacentEdge(@edge: tag): Edge | ||||
| getNextAdjacentEdge(@edge: TaggedEdge): Edge | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ getNextAdjacentEdge(@edge: tag): Edge | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `edge` | [`tag`](/docs/kcl-std/types/std-types-tag) | The tag of the edge you want to find the next adjacent edge of. | Yes | | ||||
| | `edge` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The tag of the edge you want to find the next adjacent edge of. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Get the opposite edge to the edge given. | ||||
|  | ||||
| ```kcl | ||||
| getOppositeEdge(@edge: tag): Edge | ||||
| getOppositeEdge(@edge: TaggedEdge): Edge | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ getOppositeEdge(@edge: tag): Edge | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `edge` | [`tag`](/docs/kcl-std/types/std-types-tag) | The tag of the edge you want to find the opposite edge of. | Yes | | ||||
| | `edge` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The tag of the edge you want to find the opposite edge of. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Get the previous adjacent edge to the edge given. | ||||
|  | ||||
| ```kcl | ||||
| getPreviousAdjacentEdge(@edge: tag): Edge | ||||
| getPreviousAdjacentEdge(@edge: TaggedEdge): Edge | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ getPreviousAdjacentEdge(@edge: tag): Edge | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `edge` | [`tag`](/docs/kcl-std/types/std-types-tag) | The tag of the edge you want to find the previous adjacent edge of. | Yes | | ||||
| | `edge` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The tag of the edge you want to find the previous adjacent edge of. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -14,7 +14,7 @@ involuteCircular( | ||||
|   endRadius: number(Length), | ||||
|   angle: number(Angle), | ||||
|   reverse?: bool, | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -29,7 +29,7 @@ involuteCircular( | ||||
| | `endRadius` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | The involute is described between two circles, end_radius is the radius of the outer circle. | Yes | | ||||
| | `angle` | [`number(Angle)`](/docs/kcl-std/types/std-types-number) | The angle to rotate the involute by. A value of zero will produce a curve with a tangent along the x-axis at the start point of the curve. | Yes | | ||||
| | `reverse` | [`bool`](/docs/kcl-std/types/std-types-bool) | If reverse is true, the segment will start from the end of the involute, otherwise it will start from that start. | No | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this line. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this line. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -12,7 +12,7 @@ line( | ||||
|   @sketch: Sketch, | ||||
|   endAbsolute?: Point2d, | ||||
|   end?: Point2d, | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -25,7 +25,7 @@ line( | ||||
| | `sketch` | [`Sketch`](/docs/kcl-std/types/std-types-Sketch) | Which sketch should this path be added to? | Yes | | ||||
| | `endAbsolute` | [`Point2d`](/docs/kcl-std/types/std-types-Point2d) | Which absolute point should this line go to? Incompatible with `end`. | No | | ||||
| | `end` | [`Point2d`](/docs/kcl-std/types/std-types-Point2d) | How far away (along the X and Y axes) should this line go? Incompatible with `endAbsolute`. | No | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this line. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this line. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -14,8 +14,8 @@ loft( | ||||
|   bezApproximateRational?: bool, | ||||
|   baseCurveIndex?: number(_), | ||||
|   tolerance?: number(Length), | ||||
|   tagStart?: tag, | ||||
|   tagEnd?: tag, | ||||
|   tagStart?: TagDecl, | ||||
|   tagEnd?: TagDecl, | ||||
| ): Solid | ||||
| ``` | ||||
|  | ||||
| @ -30,8 +30,8 @@ The sketches need to be closed and on different planes that are parallel. | ||||
| | `bezApproximateRational` | [`bool`](/docs/kcl-std/types/std-types-bool) | Attempt to approximate rational curves (such as arcs) using a bezier. This will remove banding around interpolations between arcs and non-arcs. It may produce errors in other scenarios. Over time, this field won't be necessary. | No | | ||||
| | `baseCurveIndex` | [`number(_)`](/docs/kcl-std/types/std-types-number) | This can be set to override the automatically determined topological base curve, which is usually the first section encountered. | No | | ||||
| | `tolerance` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | Tolerance for the loft operation. | No | | ||||
| | `tagStart` | [`tag`](/docs/kcl-std/types/std-types-tag) | A named tag for the face at the start of the loft, i.e. the original sketch. | No | | ||||
| | `tagEnd` | [`tag`](/docs/kcl-std/types/std-types-tag) | A named tag for the face at the end of the loft. | No | | ||||
| | `tagStart` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | A named tag for the face at the start of the loft, i.e. the original sketch. | No | | ||||
| | `tagEnd` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | A named tag for the face at the end of the loft. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -15,8 +15,8 @@ revolve( | ||||
|   tolerance?: number(Length), | ||||
|   symmetric?: bool, | ||||
|   bidirectionalAngle?: number(Angle), | ||||
|   tagStart?: tag, | ||||
|   tagEnd?: tag, | ||||
|   tagStart?: TagDecl, | ||||
|   tagEnd?: TagDecl, | ||||
| ): [Solid; 1+] | ||||
| ``` | ||||
| 
 | ||||
| @ -41,8 +41,8 @@ revolved around the same axis. | ||||
| | `tolerance` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | Tolerance for the revolve operation. | No | | ||||
| | `symmetric` | [`bool`](/docs/kcl-std/types/std-types-bool) | If true, the extrusion will happen symmetrically around the sketch. Otherwise, the extrusion will happen on only one side of the sketch. | No | | ||||
| | `bidirectionalAngle` | [`number(Angle)`](/docs/kcl-std/types/std-types-number) | If specified, will also revolve in the opposite direction to 'angle' to the specified angle. If 'symmetric' is true, this value is ignored. | No | | ||||
| | `tagStart` | [`tag`](/docs/kcl-std/types/std-types-tag) | A named tag for the face at the start of the revolve, i.e. the original sketch. | No | | ||||
| | `tagEnd` | [`tag`](/docs/kcl-std/types/std-types-tag) | A named tag for the face at the end of the revolve. | No | | ||||
| | `tagStart` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | A named tag for the face at the start of the revolve, i.e. the original sketch. | No | | ||||
| | `tagEnd` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | A named tag for the face at the end of the revolve. | No | | ||||
| 
 | ||||
| ### Returns | ||||
| 
 | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Compute the angle (in degrees) of the provided line segment. | ||||
|  | ||||
| ```kcl | ||||
| segAng(@tag: tag): number(Angle) | ||||
| segAng(@tag: TaggedEdge): number(Angle) | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ segAng(@tag: tag): number(Angle) | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | The line segment being queried by its tag. | Yes | | ||||
| | `tag` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The line segment being queried by its tag. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Compute the ending point of the provided line segment. | ||||
|  | ||||
| ```kcl | ||||
| segEnd(@tag: tag): Point2d | ||||
| segEnd(@tag: TaggedEdge): Point2d | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ segEnd(@tag: tag): Point2d | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | The line segment being queried by its tag. | Yes | | ||||
| | `tag` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The line segment being queried by its tag. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Compute the ending point of the provided line segment along the 'x' axis. | ||||
|  | ||||
| ```kcl | ||||
| segEndX(@tag: tag): number(Length) | ||||
| segEndX(@tag: TaggedEdge): number(Length) | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ segEndX(@tag: tag): number(Length) | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | The line segment being queried by its tag. | Yes | | ||||
| | `tag` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The line segment being queried by its tag. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Compute the ending point of the provided line segment along the 'y' axis. | ||||
|  | ||||
| ```kcl | ||||
| segEndY(@tag: tag): number(Length) | ||||
| segEndY(@tag: TaggedEdge): number(Length) | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ segEndY(@tag: tag): number(Length) | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | The line segment being queried by its tag. | Yes | | ||||
| | `tag` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The line segment being queried by its tag. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Compute the length of the provided line segment. | ||||
|  | ||||
| ```kcl | ||||
| segLen(@tag: tag): number(Length) | ||||
| segLen(@tag: TaggedEdge): number(Length) | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ segLen(@tag: tag): number(Length) | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | The line segment being queried by its tag. | Yes | | ||||
| | `tag` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The line segment being queried by its tag. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Compute the starting point of the provided line segment. | ||||
|  | ||||
| ```kcl | ||||
| segStart(@tag: tag): Point2d | ||||
| segStart(@tag: TaggedEdge): Point2d | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ segStart(@tag: tag): Point2d | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | The line segment being queried by its tag. | Yes | | ||||
| | `tag` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The line segment being queried by its tag. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Compute the starting point of the provided line segment along the 'x' axis. | ||||
|  | ||||
| ```kcl | ||||
| segStartX(@tag: tag): number(Length) | ||||
| segStartX(@tag: TaggedEdge): number(Length) | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ segStartX(@tag: tag): number(Length) | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | The line segment being queried by its tag. | Yes | | ||||
| | `tag` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The line segment being queried by its tag. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Compute the starting point of the provided line segment along the 'y' axis. | ||||
|  | ||||
| ```kcl | ||||
| segStartY(@tag: tag): number(Length) | ||||
| segStartY(@tag: TaggedEdge): number(Length) | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ segStartY(@tag: tag): number(Length) | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | The line segment being queried by its tag. | Yes | | ||||
| | `tag` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The line segment being queried by its tag. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -11,7 +11,7 @@ Start a new profile at a given point. | ||||
| startProfile( | ||||
|   @startProfileOn: Plane | Face, | ||||
|   at: Point2d, | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -23,7 +23,7 @@ startProfile( | ||||
| |----------|------|-------------|----------| | ||||
| | `startProfileOn` | [`Plane`](/docs/kcl-std/types/std-types-Plane) or [`Face`](/docs/kcl-std/types/std-types-Face) | What to start the profile on. | Yes | | ||||
| | `at` | [`Point2d`](/docs/kcl-std/types/std-types-Point2d) | Where to start the profile. An absolute point. | Yes | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Tag this first starting point. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Tag this first starting point. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -10,7 +10,7 @@ Start a new 2-dimensional sketch on a specific plane or face. | ||||
| ```kcl | ||||
| startSketchOn( | ||||
|   @planeOrSolid: Solid | Plane, | ||||
|   face?: tag, | ||||
|   face?: TaggedFace, | ||||
| ): Plane | Face | ||||
| ``` | ||||
|  | ||||
| @ -36,7 +36,7 @@ face, since it will include all the parent faces and Solids. | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `planeOrSolid` | [`Solid`](/docs/kcl-std/types/std-types-Solid) or [`Plane`](/docs/kcl-std/types/std-types-Plane) | Profile whose start is being used. | Yes | | ||||
| | `face` | [`tag`](/docs/kcl-std/types/std-types-tag) | Identify a face of a solid if a solid is specified as the input argument (`planeOrSolid`). | No | | ||||
| | `face` | [`TaggedFace`](/docs/kcl-std/types/std-types-TaggedFace) | Identify a face of a solid if a solid is specified as the input argument (`planeOrSolid`). | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -14,8 +14,8 @@ sweep( | ||||
|   sectional?: bool, | ||||
|   tolerance?: number(Length), | ||||
|   relativeTo?: string, | ||||
|   tagStart?: tag, | ||||
|   tagEnd?: tag, | ||||
|   tagStart?: TagDecl, | ||||
|   tagEnd?: TagDecl, | ||||
| ): [Solid; 1+] | ||||
| ``` | ||||
|  | ||||
| @ -37,8 +37,8 @@ swept along the same path. | ||||
| | `sectional` | [`bool`](/docs/kcl-std/types/std-types-bool) | If true, the sweep will be broken up into sub-sweeps (extrusions, revolves, sweeps) based on the trajectory path components. | No | | ||||
| | `tolerance` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | Tolerance for this operation. | No | | ||||
| | `relativeTo` | [`string`](/docs/kcl-std/types/std-types-string) | What is the sweep relative to? Can be either 'sketchPlane' or 'trajectoryCurve'. | No | | ||||
| | `tagStart` | [`tag`](/docs/kcl-std/types/std-types-tag) | A named tag for the face at the start of the sweep, i.e. the original sketch. | No | | ||||
| | `tagEnd` | [`tag`](/docs/kcl-std/types/std-types-tag) | A named tag for the face at the end of the sweep. | No | | ||||
| | `tagStart` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | A named tag for the face at the start of the sweep, i.e. the original sketch. | No | | ||||
| | `tagEnd` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | A named tag for the face at the end of the sweep. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -8,7 +8,7 @@ layout: manual | ||||
| Returns the angle coming out of the end of the segment in degrees. | ||||
|  | ||||
| ```kcl | ||||
| tangentToEnd(@tag: tag): number(Angle) | ||||
| tangentToEnd(@tag: TaggedEdge): number(Angle) | ||||
| ``` | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,7 @@ tangentToEnd(@tag: tag): number(Angle) | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | The line segment being queried by its tag. | Yes | | ||||
| | `tag` | [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | The line segment being queried by its tag. | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -15,7 +15,7 @@ tangentialArc( | ||||
|   radius?: number(Length), | ||||
|   diameter?: number(Length), | ||||
|   angle?: number(Angle), | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -35,7 +35,7 @@ for 'angle' degrees along the imaginary circle. | ||||
| | `radius` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | Radius of the imaginary circle. `angle` must be given. Incompatible with `end` and `endAbsolute` and `diameter`. | No | | ||||
| | `diameter` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | Diameter of the imaginary circle. `angle` must be given. Incompatible with `end` and `endAbsolute` and `radius`. | No | | ||||
| | `angle` | [`number(Angle)`](/docs/kcl-std/types/std-types-number) | Offset of the arc. `radius` must be given. Incompatible with `end` and `endAbsolute`. | No | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this arc. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this arc. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -12,7 +12,7 @@ xLine( | ||||
|   @sketch: Sketch, | ||||
|   length?: number(Length), | ||||
|   endAbsolute?: number(Length), | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -25,7 +25,7 @@ xLine( | ||||
| | `sketch` | [`Sketch`](/docs/kcl-std/types/std-types-Sketch) | Which sketch should this path be added to? | Yes | | ||||
| | `length` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | How far away along the X axis should this line go? Incompatible with `endAbsolute`. | No | | ||||
| | `endAbsolute` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | Which absolute X value should this line go to? Incompatible with `length`. | No | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this line. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this line. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -12,7 +12,7 @@ yLine( | ||||
|   @sketch: Sketch, | ||||
|   length?: number(Length), | ||||
|   endAbsolute?: number(Length), | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Sketch | ||||
| ``` | ||||
|  | ||||
| @ -25,7 +25,7 @@ yLine( | ||||
| | `sketch` | [`Sketch`](/docs/kcl-std/types/std-types-Sketch) | Which sketch should this path be added to? | Yes | | ||||
| | `length` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | How far away along the Y axis should this line go? Incompatible with `endAbsolute`. | No | | ||||
| | `endAbsolute` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | Which absolute Y value should this line go to? Incompatible with `length`. | No | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this line. | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this line. | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -12,7 +12,7 @@ chamfer( | ||||
|   @solid: Solid, | ||||
|   length: number(Length), | ||||
|   tags: [Edge; 1+], | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Solid | ||||
| ``` | ||||
|  | ||||
| @ -27,7 +27,7 @@ a sharp, straight transitional edge. | ||||
| | `solid` | [`Solid`](/docs/kcl-std/types/std-types-Solid) | The solid whose edges should be chamfered | Yes | | ||||
| | `length` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | The length of the chamfer | Yes | | ||||
| | `tags` | [`[Edge; 1+]`](/docs/kcl-std/types/std-types-Edge) | The paths you want to chamfer | Yes | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this chamfer | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this chamfer | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -13,7 +13,7 @@ fillet( | ||||
|   radius: number(Length), | ||||
|   tags: [Edge; 1+], | ||||
|   tolerance?: number(Length), | ||||
|   tag?: tag, | ||||
|   tag?: TagDecl, | ||||
| ): Solid | ||||
| ``` | ||||
|  | ||||
| @ -29,7 +29,7 @@ will smoothly blend the transition. | ||||
| | `radius` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | The radius of the fillet | Yes | | ||||
| | `tags` | [`[Edge; 1+]`](/docs/kcl-std/types/std-types-Edge) | The paths you want to fillet | Yes | | ||||
| | `tolerance` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | The tolerance for this fillet | No | | ||||
| | [`tag`](/docs/kcl-std/types/std-types-tag) | [`tag`](/docs/kcl-std/types/std-types-tag) | Create a new tag which refers to this fillet | No | | ||||
| | `tag` | [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | Create a new tag which refers to this fillet | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -11,7 +11,7 @@ Remove volume from a 3-dimensional shape such that a wall of the provided thickn | ||||
| shell( | ||||
|   @solids: [Solid; 1+], | ||||
|   thickness: number(Length), | ||||
|   faces: [tag; 1+], | ||||
|   faces: [TaggedFace; 1+], | ||||
| ): [Solid] | ||||
| ``` | ||||
|  | ||||
| @ -23,7 +23,7 @@ shell( | ||||
| |----------|------|-------------|----------| | ||||
| | `solids` | [`[Solid; 1+]`](/docs/kcl-std/types/std-types-Solid) | Which solid (or solids) to shell out | Yes | | ||||
| | `thickness` | [`number(Length)`](/docs/kcl-std/types/std-types-number) | The thickness of the shell | Yes | | ||||
| | `faces` | [`[tag; 1+]`](/docs/kcl-std/types/std-types-tag) | The faces you want removed | Yes | | ||||
| | `faces` | [`[TaggedFace; 1+]`](/docs/kcl-std/types/std-types-TaggedFace) | The faces you want removed | Yes | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
|  | ||||
| @ -145,12 +145,12 @@ See also the [types overview](/docs/kcl-lang/types) | ||||
|  | ||||
| * [**Primitive types**](/docs/kcl-lang/types) | ||||
|   * [`ImportedGeometry`](/docs/kcl-std/types/std-types-ImportedGeometry) | ||||
|   * [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | ||||
|   * [`any`](/docs/kcl-std/types/std-types-any) | ||||
|   * [`bool`](/docs/kcl-std/types/std-types-bool) | ||||
|   * [`fn`](/docs/kcl-std/types/std-types-fn) | ||||
|   * [`number`](/docs/kcl-std/types/std-types-number) | ||||
|   * [`string`](/docs/kcl-std/types/std-types-string) | ||||
|   * [`tag`](/docs/kcl-std/types/std-types-tag) | ||||
| * [**std::types**](/docs/kcl-std/modules/std-types) | ||||
|   * [`Axis2d`](/docs/kcl-std/types/std-types-Axis2d) | ||||
|   * [`Axis3d`](/docs/kcl-std/types/std-types-Axis3d) | ||||
| @ -162,3 +162,5 @@ See also the [types overview](/docs/kcl-lang/types) | ||||
|   * [`Point3d`](/docs/kcl-std/types/std-types-Point3d) | ||||
|   * [`Sketch`](/docs/kcl-std/types/std-types-Sketch) | ||||
|   * [`Solid`](/docs/kcl-std/types/std-types-Solid) | ||||
|   * [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | ||||
|   * [`TaggedFace`](/docs/kcl-std/types/std-types-TaggedFace) | ||||
|  | ||||
| @ -24,6 +24,9 @@ Types can (optionally) be used to describe a function's arguments and returned v | ||||
| * [`Point3d`](/docs/kcl-std/types/std-types-Point3d) | ||||
| * [`Sketch`](/docs/kcl-std/types/std-types-Sketch) | ||||
| * [`Solid`](/docs/kcl-std/types/std-types-Solid) | ||||
| * [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl) | ||||
| * [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) | ||||
| * [`TaggedFace`](/docs/kcl-std/types/std-types-TaggedFace) | ||||
| * [`any`](/docs/kcl-std/types/std-types-any) | ||||
| * [`bool`](/docs/kcl-std/types/std-types-bool) | ||||
| * [`fn`](/docs/kcl-std/types/std-types-fn) | ||||
|  | ||||
							
								
								
									
										102
									
								
								docs/kcl-std/types/std-types-TagDecl.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,102 @@ | ||||
| --- | ||||
| title: "TagDecl" | ||||
| subtitle: "Type in std::types" | ||||
| excerpt: "Tags are used to give a name (tag) to a specific path." | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
| Tags are used to give a name (tag) to a specific path. | ||||
|  | ||||
| ### Tag Declaration | ||||
|  | ||||
| The syntax for declaring a tag is `$myTag`. You would use it in the following | ||||
| way: | ||||
|  | ||||
| ```js | ||||
| startSketchOn(XZ) | ||||
|   |> startProfile(at = origin) | ||||
|   |> angledLine(angle = 0, length = 191.26, tag = $rectangleSegmentA001) | ||||
|   |> angledLine( | ||||
|        angle = segAng(rectangleSegmentA001) - 90deg, | ||||
|        length = 196.99, | ||||
|        tag = $rectangleSegmentB001, | ||||
|      ) | ||||
|   |> angledLine( | ||||
|        angle = segAng(rectangleSegmentA001), | ||||
|        length = -segLen(rectangleSegmentA001), | ||||
|        tag = $rectangleSegmentC001, | ||||
|      ) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| ``` | ||||
|  | ||||
| ### Tag Scope | ||||
|  | ||||
| Tags are scoped globally if in the root context meaning in this example you can | ||||
| use the tag `rectangleSegmentA001` in any function or expression in the file. | ||||
|  | ||||
| However if the code was written like this: | ||||
|  | ||||
| ```js | ||||
| fn rect(origin) { | ||||
|   return startSketchOn(XZ) | ||||
|     |> startProfile(at = origin) | ||||
|     |> angledLine(angle = 0, length = 191.26, tag = $rectangleSegmentA001) | ||||
|     |> angledLine( | ||||
|          angle = segAng(rectangleSegmentA001) - 90, | ||||
|          length = 196.99, | ||||
|          tag = $rectangleSegmentB001 | ||||
|        ) | ||||
|     |> angledLine( | ||||
|          angle = segAng(rectangleSegmentA001), | ||||
|          length = -segLen(rectangleSegmentA001), | ||||
|          tag = $rectangleSegmentC001 | ||||
|        ) | ||||
|     |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|     |> close() | ||||
| } | ||||
|  | ||||
| rect(origin = [0, 0]) | ||||
| rect(origin = [20, 0]) | ||||
| ``` | ||||
|  | ||||
| Those tags would only be available in the `rect` function and not globally. | ||||
|  | ||||
| However you likely want to use those tags somewhere outside the `rect` function. | ||||
|  | ||||
| Tags are accessible through the sketch group they are declared in. | ||||
| For example the following code works. | ||||
|  | ||||
| ```js | ||||
| fn rect(origin) { | ||||
|   return startSketchOn(XZ) | ||||
|     |> startProfile(at = origin) | ||||
|     |> angledLine(angle = 0, length = 191.26, tag = $rectangleSegmentA001) | ||||
|     |> angledLine( | ||||
|          angle = segAng(rectangleSegmentA001) - 90deg, | ||||
|          length = 196.99, | ||||
|          tag = $rectangleSegmentB001, | ||||
|        ) | ||||
|     |> angledLine( | ||||
|          angle = segAng(rectangleSegmentA001), | ||||
|          length = -segLen(rectangleSegmentA001), | ||||
|          tag = $rectangleSegmentC001, | ||||
|        ) | ||||
|     |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|     |> close() | ||||
| } | ||||
|  | ||||
| rect(origin = [0, 0]) | ||||
| myRect = rect(origin = [20, 0]) | ||||
|  | ||||
| myRect | ||||
|   |> extrude(length = 10) | ||||
|   |> fillet(radius = 0.5, tags = [myRect.tags.rectangleSegmentA001]) | ||||
| ``` | ||||
|  | ||||
| See how we use the tag `rectangleSegmentA001` in the `fillet` function outside | ||||
| the `rect` function. This is because the `rect` function is returning the | ||||
| sketch group that contains the tags. | ||||
|  | ||||
|  | ||||
|  | ||||
							
								
								
									
										17
									
								
								docs/kcl-std/types/std-types-TaggedEdge.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,17 @@ | ||||
| --- | ||||
| title: "TaggedEdge" | ||||
| subtitle: "Type in std::types" | ||||
| excerpt: "A tag which references a line, arc, or other edge in a sketch or an edge of a solid." | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
| A tag which references a line, arc, or other edge in a sketch or an edge of a solid. | ||||
|  | ||||
| Created by using a tag declarator (see the docs for [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl)). Can be used where an [`Edge`](/docs/kcl-std/types/std-types-Edge) is | ||||
| required. | ||||
|  | ||||
| If a line in a sketch is tagged and then the sketch is extruded, the tag is a [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) before | ||||
| extrusion and a [`TaggedFace`](/docs/kcl-std/types/std-types-TaggedFace) after extrusion. | ||||
|  | ||||
|  | ||||
|  | ||||
							
								
								
									
										16
									
								
								docs/kcl-std/types/std-types-TaggedFace.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,16 @@ | ||||
| --- | ||||
| title: "TaggedFace" | ||||
| subtitle: "Type in std::types" | ||||
| excerpt: "A tag which references a face of a solid, including the distinguished tags `START` and `END`." | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
| A tag which references a face of a solid, including the distinguished tags `START` and `END`. | ||||
|  | ||||
| Created by using a tag declarator (see the docs for [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl)). | ||||
|  | ||||
| If a line in a sketch is tagged and then the sketch is extruded, the tag is a [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) before | ||||
| extrusion and a [`TaggedFace`](/docs/kcl-std/types/std-types-TaggedFace) after extrusion. | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -1,109 +1,19 @@ | ||||
| --- | ||||
| title: "tag" | ||||
| subtitle: "Type in std::types" | ||||
| excerpt: "Tags are used to give a name (tag) to a specific path." | ||||
| excerpt: "Reference a previously created tag. Used much like a variable." | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
| Tags are used to give a name (tag) to a specific path. | ||||
| **WARNING:** This type is deprecated. | ||||
|  | ||||
| ### Tag Declaration | ||||
| Reference a previously created tag. Used much like a variable. | ||||
|  | ||||
| The syntax for declaring a tag is `$myTag` you would use it in the following | ||||
| way: | ||||
|  | ||||
| ```js | ||||
| startSketchOn(XZ) | ||||
|   |> startProfile(at = origin) | ||||
|   |> angledLine(angle = 0, length = 191.26, tag = $rectangleSegmentA001) | ||||
|   |> angledLine( | ||||
|        angle = segAng(rectangleSegmentA001) - 90deg, | ||||
|        length = 196.99, | ||||
|        tag = $rectangleSegmentB001, | ||||
|      ) | ||||
|   |> angledLine( | ||||
|        angle = segAng(rectangleSegmentA001), | ||||
|        length = -segLen(rectangleSegmentA001), | ||||
|        tag = $rectangleSegmentC001, | ||||
|      ) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| ```kcl | ||||
| type tag = TaggedEdge | ||||
| ``` | ||||
|  | ||||
| ### Tag Identifier | ||||
|  | ||||
| As per the example above you can use the tag identifier to get a reference to the | ||||
| tagged object. The syntax for this is `myTag`. | ||||
|  | ||||
| In the example above we use the tag identifier to get the angle of the segment | ||||
| `segAng(rectangleSegmentA001)`. | ||||
|  | ||||
| ### Tag Scope | ||||
|  | ||||
| Tags are scoped globally if in the root context meaning in this example you can | ||||
| use the tag `rectangleSegmentA001` in any function or expression in the file. | ||||
|  | ||||
| However if the code was written like this: | ||||
|  | ||||
| ```js | ||||
| fn rect(origin) { | ||||
|   return startSketchOn(XZ) | ||||
|     |> startProfile(at = origin) | ||||
|     |> angledLine(angle = 0, length = 191.26, tag = $rectangleSegmentA001) | ||||
|     |> angledLine( | ||||
|          angle = segAng(rectangleSegmentA001) - 90, | ||||
|          length = 196.99, | ||||
|          tag = $rectangleSegmentB001) | ||||
|     |> angledLine( | ||||
|          angle = segAng(rectangleSegmentA001), | ||||
|          length = -segLen(rectangleSegmentA001), | ||||
|          tag = $rectangleSegmentC001 | ||||
|        ) | ||||
|     |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|     |> close() | ||||
| } | ||||
|  | ||||
| rect(origin = [0, 0]) | ||||
| rect(origin = [20, 0]) | ||||
| ``` | ||||
|  | ||||
| Those tags would only be available in the `rect` function and not globally. | ||||
|  | ||||
| However you likely want to use those tags somewhere outside the `rect` function. | ||||
|  | ||||
| Tags are accessible through the sketch group they are declared in. | ||||
| For example the following code works. | ||||
|  | ||||
| ```js | ||||
| fn rect(origin) { | ||||
|   return startSketchOn(XZ) | ||||
|     |> startProfile(at = origin) | ||||
|     |> angledLine(angle = 0, length = 191.26, tag = $rectangleSegmentA001) | ||||
|     |> angledLine( | ||||
|          angle = segAng(rectangleSegmentA001) - 90deg, | ||||
|          length = 196.99 | ||||
|          tag = $rectangleSegmentB001, | ||||
|        ) | ||||
|     |> angledLine( | ||||
|          angle = segAng(rectangleSegmentA001), | ||||
|          length = -segLen(rectangleSegmentA001) | ||||
|          tag = $rectangleSegmentC001, | ||||
|        ) | ||||
|     |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|     |> close() | ||||
| } | ||||
|  | ||||
| rect(origin = [0, 0]) | ||||
| myRect = rect(origin = [20, 0]) | ||||
|  | ||||
| myRect | ||||
|   |> extrude(length = 10) | ||||
|   |> fillet(radius = 0.5, tags = [myRect.tags.rectangleSegmentA001]) | ||||
| ``` | ||||
|  | ||||
| See how we use the tag `rectangleSegmentA001` in the `fillet` function outside | ||||
| the `rect` function. This is because the `rect` function is returning the | ||||
| sketch group that contains the tags. | ||||
| Prefer to use [`TaggedEdge`](/docs/kcl-std/types/std-types-TaggedEdge) or [`TaggedFace`](/docs/kcl-std/types/std-types-TaggedFace). For more details on tags, see the docs for [`TagDecl`](/docs/kcl-std/types/std-types-TagDecl). | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -4,7 +4,6 @@ import * as fsp from 'fs/promises' | ||||
|  | ||||
| import { executorInputPath, getUtils } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
| import { expectPixelColor } from '@e2e/playwright/fixtures/sceneFixture' | ||||
|  | ||||
| test.describe('Command bar tests', () => { | ||||
|   test('Extrude from command bar selects extrude line after', async ({ | ||||
| @ -515,47 +514,6 @@ test.describe('Command bar tests', () => { | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   test( | ||||
|     `Zoom to fit to shared model on web`, | ||||
|     { tag: ['@web'] }, | ||||
|     async ({ page, scene }) => { | ||||
|       if (process.env.TARGET !== 'web') { | ||||
|         // This test is web-only | ||||
|         // TODO: re-enable on CI as part of a new @web test suite | ||||
|         return | ||||
|       } | ||||
|       await test.step(`Prepare and navigate to home page with query params`, async () => { | ||||
|         // a quad in the top left corner of the XZ plane (which is out of the current view) | ||||
|         const code = `sketch001 = startSketchOn(XZ) | ||||
| profile001 = startProfile(sketch001, at = [-484.34, 484.95]) | ||||
|   |> yLine(length = -69.1) | ||||
|   |> xLine(length = 66.84) | ||||
|   |> yLine(length = 71.37) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| ` | ||||
|         const targetURL = `?create-file=true&name=test&units=mm&code=${encodeURIComponent(btoa(code))}&ask-open-desktop=true` | ||||
|         await page.goto(page.url() + targetURL) | ||||
|         expect(page.url()).toContain(targetURL) | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Submit the command`, async () => { | ||||
|         await page.getByTestId('continue-to-web-app-button').click() | ||||
|  | ||||
|         await scene.connectionEstablished() | ||||
|  | ||||
|         // This makes SystemIOMachineActors.createKCLFile run after EngineStream/firstPlay | ||||
|         await page.waitForTimeout(3000) | ||||
|  | ||||
|         await page.getByTestId('command-bar-submit').click() | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Ensure we created the project and are in the modeling scene`, async () => { | ||||
|         await expectPixelColor(page, [252, 252, 252], { x: 600, y: 260 }, 8) | ||||
|       }) | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test(`Can add and edit a named parameter or constant`, async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|  | ||||
| @ -170,7 +170,7 @@ test( | ||||
|       // error text on hover | ||||
|       await page.hover('.cm-lint-marker-error') | ||||
|       const crypticErrorText = | ||||
|         'tag requires a value with type `tag`, but found a value with type `string`.' | ||||
|         'tag requires a value with type `TagDecl`, but found a value with type `string`.' | ||||
|       await expect(page.getByText(crypticErrorText).first()).toBeVisible() | ||||
|  | ||||
|       // black pixel means the scene has been cleared. | ||||
| @ -369,7 +369,7 @@ test( | ||||
|       // error text on hover | ||||
|       await page.hover('.cm-lint-marker-error') | ||||
|       const crypticErrorText = | ||||
|         'tag requires a value with type `tag`, but found a value with type `string`.' | ||||
|         'tag requires a value with type `TagDecl`, but found a value with type `string`.' | ||||
|       await expect(page.getByText(crypticErrorText).first()).toBeVisible() | ||||
|  | ||||
|       // black pixel means the scene has been cleared. | ||||
| @ -408,7 +408,7 @@ test( | ||||
|     // error text on hover | ||||
|     await page.hover('.cm-lint-marker-error') | ||||
|     const crypticErrorText = | ||||
|       'tag requires a value with type `tag`, but found a value with type `string`.' | ||||
|       'tag requires a value with type `TagDecl`, but found a value with type `string`.' | ||||
|     await expect(page.getByText(crypticErrorText).first()).toBeVisible() | ||||
|   } | ||||
| ) | ||||
|  | ||||
| @ -575,7 +575,7 @@ extrude002 = extrude(profile002, length = 150) | ||||
|         name: 'Projects', | ||||
|       }) | ||||
|       const projectLink = page.getByRole('link', { name: 'bracket' }) | ||||
|       const networkHealthIndicator = page.getByTestId('network-toggle') | ||||
|       const networkHealthIndicator = page.getByTestId(/network-toggle/) | ||||
|  | ||||
|       await test.step('Check the home page', async () => { | ||||
|         await expect(projectsHeading).toBeVisible() | ||||
|  | ||||
| @ -1445,6 +1445,48 @@ solid001 = subtract([extrude001], tools = [extrude002]) | ||||
|     await u.closeDebugPanel() | ||||
|   }) | ||||
|  | ||||
|   test('Can edit a tangentialArc defined by angle and radius', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|     editor, | ||||
|     toolbar, | ||||
|     scene, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const viewportSize = { width: 1500, height: 750 } | ||||
|     await page.setBodyDimensions(viewportSize) | ||||
|  | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `@settings(defaultLengthUnit=in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfile(at = [-10, -10]) | ||||
|   |> line(end = [20.0, 10.0]) | ||||
|   |> tangentialArc(angle = 60deg, radius=10.0)` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await homePage.goToModelingScene() | ||||
|     await toolbar.waitForFeatureTreeToBeBuilt() | ||||
|     await scene.settled(cmdBar) | ||||
|  | ||||
|     await (await toolbar.getFeatureTreeOperation('Sketch', 0)).dblclick() | ||||
|  | ||||
|     await page.waitForTimeout(1000) | ||||
|  | ||||
|     await page.mouse.move(1200, 139) | ||||
|     await page.mouse.down() | ||||
|     await page.mouse.move(870, 250) | ||||
|  | ||||
|     await page.waitForTimeout(200) | ||||
|  | ||||
|     await editor.expectEditor.toContain( | ||||
|       `tangentialArc(angle = 234.01deg, radius = 4.08)`, | ||||
|       { shouldNormalise: true } | ||||
|     ) | ||||
|   }) | ||||
|  | ||||
|   test('Can delete a single segment line with keyboard', async ({ | ||||
|     page, | ||||
|     scene, | ||||
|  | ||||
| @ -798,7 +798,7 @@ test('theme persists', async ({ page, context, homePage }) => { | ||||
|  | ||||
|   await page.getByTestId('settings-close-button').click() | ||||
|  | ||||
|   const networkToggle = page.getByTestId('network-toggle') | ||||
|   const networkToggle = page.getByTestId(/network-toggle/) | ||||
|  | ||||
|   // simulate network down | ||||
|   await u.emulateNetworkConditions({ | ||||
|  | ||||
| Before Width: | Height: | Size: 56 KiB After Width: | Height: | Size: 55 KiB | 
| Before Width: | Height: | Size: 49 KiB After Width: | Height: | Size: 50 KiB | 
| Before Width: | Height: | Size: 56 KiB After Width: | Height: | Size: 57 KiB | 
| Before Width: | Height: | Size: 49 KiB After Width: | Height: | Size: 50 KiB | 
| Before Width: | Height: | Size: 50 KiB After Width: | Height: | Size: 48 KiB | 
| Before Width: | Height: | Size: 48 KiB After Width: | Height: | Size: 47 KiB | 
| Before Width: | Height: | Size: 44 KiB After Width: | Height: | Size: 43 KiB | 
| Before Width: | Height: | Size: 48 KiB After Width: | Height: | Size: 46 KiB | 
| Before Width: | Height: | Size: 52 KiB After Width: | Height: | Size: 51 KiB | 
| Before Width: | Height: | Size: 58 KiB After Width: | Height: | Size: 57 KiB | 
| Before Width: | Height: | Size: 31 KiB After Width: | Height: | Size: 32 KiB | 
| Before Width: | Height: | Size: 51 KiB After Width: | Height: | Size: 52 KiB | 
| Before Width: | Height: | Size: 49 KiB After Width: | Height: | Size: 50 KiB | 
| Before Width: | Height: | Size: 75 KiB After Width: | Height: | Size: 74 KiB | 
| Before Width: | Height: | Size: 47 KiB After Width: | Height: | Size: 47 KiB | 
| Before Width: | Height: | Size: 59 KiB After Width: | Height: | Size: 60 KiB | 
| Before Width: | Height: | Size: 133 KiB After Width: | Height: | Size: 134 KiB | 
| Before Width: | Height: | Size: 117 KiB After Width: | Height: | Size: 117 KiB | 
| Before Width: | Height: | Size: 133 KiB After Width: | Height: | Size: 134 KiB | 
| Before Width: | Height: | Size: 66 KiB After Width: | Height: | Size: 67 KiB | 
| Before Width: | Height: | Size: 69 KiB After Width: | Height: | Size: 70 KiB | 
| Before Width: | Height: | Size: 65 KiB After Width: | Height: | Size: 66 KiB | 
| Before Width: | Height: | Size: 68 KiB After Width: | Height: | Size: 68 KiB | 
| Before Width: | Height: | Size: 68 KiB After Width: | Height: | Size: 68 KiB | 
| Before Width: | Height: | Size: 64 KiB After Width: | Height: | Size: 64 KiB | 
							
								
								
									
										117
									
								
								e2e/playwright/temporary-workspace.spec.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,117 @@ | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
| import { stringToBase64 } from '@src/lib/base64' | ||||
|  | ||||
| test.describe('Temporary workspace', () => { | ||||
|   test( | ||||
|     'Opening a share link creates a temporary environment that is not saved', | ||||
|     { tag: ['@web'] }, | ||||
|     async ({ page, editor, scene, cmdBar, homePage }) => { | ||||
|       await test.step('Pre-condition: editor is empty', async () => { | ||||
|         await homePage.goToModelingScene() | ||||
|         await scene.settled(cmdBar) | ||||
|         await editor.expectEditor.toContain('') | ||||
|       }) | ||||
|  | ||||
|       await test.step('Go to share link, check new content present, make a change', async () => { | ||||
|         const code = `sketch001 = startSketchOn(XY) | ||||
|   profile001 = startProfile(sketch001, at = [-124.89, -186.4]) | ||||
|     |> line(end = [391.31, 444.04]) | ||||
|     |> line(end = [96.21, -493.07]) | ||||
|     |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|     |> close() | ||||
|   extrude001 = extrude(profile001, length = 5) | ||||
| ` | ||||
|  | ||||
|         const codeQueryParam = encodeURIComponent(stringToBase64(code)) | ||||
|         const targetURL = `?create-file=true&browser=test&code=${codeQueryParam}&ask-open-desktop=true` | ||||
|         await page.goto(page.url() + targetURL) | ||||
|         await expect.poll(() => page.url()).toContain(targetURL) | ||||
|         const button = page.getByRole('button', { name: 'Continue to web app' }) | ||||
|         await button.click() | ||||
|  | ||||
|         await editor.expectEditor.toContain(code, { shouldNormalise: true }) | ||||
|         await editor.scrollToText('-124.89', true) | ||||
|         await page.keyboard.press('9') | ||||
|         await page.keyboard.press('9') | ||||
|       }) | ||||
|  | ||||
|       await test.step('Post-condition: empty editor once again (original state)', async () => { | ||||
|         await homePage.goToModelingScene() | ||||
|         await scene.settled(cmdBar) | ||||
|         const code = await page.evaluate(() => | ||||
|           window.localStorage.getItem('persistCode') | ||||
|         ) | ||||
|         await expect(code).toContain('') | ||||
|       }) | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test( | ||||
|     'Opening a sample link creates a temporary environment that is not saved', | ||||
|     { tag: ['@web'] }, | ||||
|     async ({ page, editor, scene, cmdBar, homePage }) => { | ||||
|       await test.step('Pre-condition: editor is empty', async () => { | ||||
|         await homePage.goToModelingScene() | ||||
|         await scene.settled(cmdBar) | ||||
|         await editor.expectEditor.toContain('') | ||||
|       }) | ||||
|  | ||||
|       await test.step('Load sample, make an edit', async () => { | ||||
|         await page.goto( | ||||
|           `${page.url()}/?cmd=add-kcl-file-to-project&groupId=application&projectName=browser&source=kcl-samples&sample=brake-rotor/main.kcl` | ||||
|         ) | ||||
|  | ||||
|         await editor.scrollToText('114.3', true) | ||||
|         await page.keyboard.press('9') | ||||
|         await page.keyboard.press('9') | ||||
|       }) | ||||
|  | ||||
|       await test.step('Post-condition: empty editor once again (original state)', async () => { | ||||
|         await homePage.goToModelingScene() | ||||
|         await scene.settled(cmdBar) | ||||
|         const code = await page.evaluate(() => | ||||
|           window.localStorage.getItem('persistCode') | ||||
|         ) | ||||
|         await expect(code).toContain('') | ||||
|       }) | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test( | ||||
|     'Hitting save will save the temporary workspace', | ||||
|     { tag: ['@web'] }, | ||||
|     async ({ page, editor, scene, cmdBar, homePage }) => { | ||||
|       const buttonSaveTemporaryWorkspace = page.getByTestId('tws-save') | ||||
|  | ||||
|       await test.step('Pre-condition: editor is empty', async () => { | ||||
|         await homePage.goToModelingScene() | ||||
|         await scene.settled(cmdBar) | ||||
|         await editor.expectEditor.toContain('') | ||||
|       }) | ||||
|  | ||||
|       await test.step('Load sample, make an edit, *save*', async () => { | ||||
|         await page.goto( | ||||
|           `${page.url()}/?cmd=add-kcl-file-to-project&groupId=application&projectName=browser&source=kcl-samples&sample=brake-rotor/main.kcl` | ||||
|         ) | ||||
|         await homePage.goToModelingScene() | ||||
|         await scene.settled(cmdBar) | ||||
|  | ||||
|         await editor.scrollToText('114.3') | ||||
|         await editor.replaceCode('114.3', '999.9133') | ||||
|         await editor.expectEditor.toContain('999.9133') | ||||
|  | ||||
|         await buttonSaveTemporaryWorkspace.click() | ||||
|         await expect(buttonSaveTemporaryWorkspace).not.toBeVisible() | ||||
|  | ||||
|         await editor.expectEditor.toContain('999.9133') | ||||
|       }) | ||||
|  | ||||
|       await test.step('Post-condition: has the edits in localStorage', async () => { | ||||
|         const code = await page.evaluate(() => | ||||
|           window.localStorage.getItem('persistCode') | ||||
|         ) | ||||
|         await expect(code).toContain('999.9133') | ||||
|       }) | ||||
|     } | ||||
|   ) | ||||
| }) | ||||
| @ -24,16 +24,15 @@ test.describe('Test network related behaviors', () => { | ||||
|  | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|       const networkToggle = page.getByTestId('network-toggle') | ||||
|       const networkToggle = page.getByTestId(/network-toggle/) | ||||
|  | ||||
|       // This is how we wait until the stream is online | ||||
|       await expect( | ||||
|         page.getByRole('button', { name: 'Start Sketch' }) | ||||
|       ).not.toBeDisabled({ timeout: 15000 }) | ||||
|  | ||||
|       const networkWidget = page.locator('[data-testid="network-toggle"]') | ||||
|       await expect(networkWidget).toBeVisible() | ||||
|       await networkWidget.hover() | ||||
|       await expect(networkToggle).toBeVisible() | ||||
|       await networkToggle.hover() | ||||
|  | ||||
|       const networkPopover = page.locator('[data-testid="network-popover"]') | ||||
|       await expect(networkPopover).not.toBeVisible() | ||||
| @ -44,7 +43,7 @@ test.describe('Test network related behaviors', () => { | ||||
|       ).toBeVisible() | ||||
|  | ||||
|       // Click the network widget | ||||
|       await networkWidget.click() | ||||
|       await networkToggle.click() | ||||
|  | ||||
|       // Check the modal opened. | ||||
|       await expect(networkPopover).toBeVisible() | ||||
| @ -65,8 +64,8 @@ test.describe('Test network related behaviors', () => { | ||||
|       // Expect the network to be down | ||||
|       await expect(networkToggle).toContainText('Network health (Offline)') | ||||
|  | ||||
|       // Click the network widget | ||||
|       await networkWidget.click() | ||||
|       // Click the network toggle | ||||
|       await networkToggle.click() | ||||
|  | ||||
|       // Check the modal opened. | ||||
|       await expect(networkPopover).toBeVisible() | ||||
| @ -99,7 +98,7 @@ test.describe('Test network related behaviors', () => { | ||||
|     'Engine disconnect & reconnect in sketch mode', | ||||
|     { tag: '@skipLocalEngine' }, | ||||
|     async ({ page, homePage, toolbar, scene, cmdBar }) => { | ||||
|       const networkToggle = page.getByTestId('network-toggle') | ||||
|       const networkToggle = page.getByTestId(/network-toggle/) | ||||
|       const networkToggleConnectedText = page.getByText( | ||||
|         'Network health (Strong)' | ||||
|       ) | ||||
| @ -286,7 +285,7 @@ profile001 = startProfile(sketch001, at = [12.34, -12.34]) | ||||
|     'Paused stream freezes view frame, unpause reconnect is seamless to user', | ||||
|     { tag: ['@desktop', '@skipLocalEngine'] }, | ||||
|     async ({ page, homePage, scene, cmdBar, toolbar, tronApp }) => { | ||||
|       const networkToggle = page.getByTestId('network-toggle') | ||||
|       const networkToggle = page.getByTestId(/network-toggle/) | ||||
|       const networkToggleConnectedText = page.getByText( | ||||
|         'Network health (Strong)' | ||||
|       ) | ||||
|  | ||||
| @ -37,7 +37,7 @@ export const headerMasks = (page: Page) => [ | ||||
| ] | ||||
|  | ||||
| export const lowerRightMasks = (page: Page) => [ | ||||
|   page.getByTestId('network-toggle'), | ||||
|   page.getByTestId(/network-toggle/), | ||||
|   page.getByTestId('billing-remaining-bar'), | ||||
| ] | ||||
|  | ||||
|  | ||||
| @ -8,37 +8,37 @@ test.describe('Testing Gizmo', () => { | ||||
|   const cases = [ | ||||
|     { | ||||
|       testDescription: 'top view', | ||||
|       clickPosition: { x: 951, y: 385 }, | ||||
|       clickPosition: { x: 951, y: 402 }, | ||||
|       expectedCameraPosition: { x: 800, y: -152, z: 4886.02 }, | ||||
|       expectedCameraTarget: { x: 800, y: -152, z: 26 }, | ||||
|     }, | ||||
|     { | ||||
|       testDescription: 'bottom view', | ||||
|       clickPosition: { x: 951, y: 429 }, | ||||
|       clickPosition: { x: 951, y: 449 }, | ||||
|       expectedCameraPosition: { x: 800, y: -152, z: -4834.02 }, | ||||
|       expectedCameraTarget: { x: 800, y: -152, z: 26 }, | ||||
|     }, | ||||
|     { | ||||
|       testDescription: 'right view', | ||||
|       clickPosition: { x: 929, y: 417 }, | ||||
|       clickPosition: { x: 929, y: 435 }, | ||||
|       expectedCameraPosition: { x: 5660.02, y: -152, z: 26 }, | ||||
|       expectedCameraTarget: { x: 800, y: -152, z: 26 }, | ||||
|     }, | ||||
|     { | ||||
|       testDescription: 'left view', | ||||
|       clickPosition: { x: 974, y: 397 }, | ||||
|       clickPosition: { x: 974, y: 417 }, | ||||
|       expectedCameraPosition: { x: -4060.02, y: -152, z: 26 }, | ||||
|       expectedCameraTarget: { x: 800, y: -152, z: 26 }, | ||||
|     }, | ||||
|     { | ||||
|       testDescription: 'back view', | ||||
|       clickPosition: { x: 967, y: 421 }, | ||||
|       clickPosition: { x: 967, y: 441 }, | ||||
|       expectedCameraPosition: { x: 800, y: 4708.02, z: 26 }, | ||||
|       expectedCameraTarget: { x: 800, y: -152, z: 26 }, | ||||
|     }, | ||||
|     { | ||||
|       testDescription: 'front view', | ||||
|       clickPosition: { x: 935, y: 393 }, | ||||
|       clickPosition: { x: 935, y: 413 }, | ||||
|       expectedCameraPosition: { x: 800, y: -5012.02, z: 26 }, | ||||
|       expectedCameraTarget: { x: 800, y: -152, z: 26 }, | ||||
|     }, | ||||
|  | ||||
| @ -109,6 +109,8 @@ When you submit a PR to add or modify KCL samples, images will be generated and | ||||
| [](makeup-mirror/main.kcl) | ||||
| #### [mounting-plate](mounting-plate/main.kcl) ([screenshot](screenshots/mounting-plate.png)) | ||||
| [](mounting-plate/main.kcl) | ||||
| #### [mug](mug/main.kcl) ([screenshot](screenshots/mug.png)) | ||||
| [](mug/main.kcl) | ||||
| #### [multi-axis-robot](multi-axis-robot/main.kcl) ([screenshot](screenshots/multi-axis-robot.png)) | ||||
| [](multi-axis-robot/main.kcl) | ||||
| #### [pdu-faceplate](pdu-faceplate/main.kcl) ([screenshot](screenshots/pdu-faceplate.png)) | ||||
| @ -139,6 +141,8 @@ When you submit a PR to add or modify KCL samples, images will be generated and | ||||
| [](socket-head-cap-screw/main.kcl) | ||||
| #### [spinning-highrise-tower](spinning-highrise-tower/main.kcl) ([screenshot](screenshots/spinning-highrise-tower.png)) | ||||
| [](spinning-highrise-tower/main.kcl) | ||||
| #### [spool](spool/main.kcl) ([screenshot](screenshots/spool.png)) | ||||
| [](spool/main.kcl) | ||||
| #### [spur-gear](spur-gear/main.kcl) ([screenshot](screenshots/spur-gear.png)) | ||||
| [](spur-gear/main.kcl) | ||||
| #### [spur-reduction-gearset](spur-reduction-gearset/main.kcl) ([screenshot](screenshots/spur-reduction-gearset.png)) | ||||
| @ -147,6 +151,8 @@ When you submit a PR to add or modify KCL samples, images will be generated and | ||||
| [](surgical-drill-guide/main.kcl) | ||||
| #### [t-slot-rail](t-slot-rail/main.kcl) ([screenshot](screenshots/t-slot-rail.png)) | ||||
| [](t-slot-rail/main.kcl) | ||||
| #### [teapot](teapot/main.kcl) ([screenshot](screenshots/teapot.png)) | ||||
| [](teapot/main.kcl) | ||||
| #### [telemetry-antenna](telemetry-antenna/main.kcl) ([screenshot](screenshots/telemetry-antenna.png)) | ||||
| [](telemetry-antenna/main.kcl) | ||||
| #### [thermal-block-insert](thermal-block-insert/main.kcl) ([screenshot](screenshots/thermal-block-insert.png)) | ||||
|  | ||||
| @ -447,6 +447,16 @@ | ||||
|       "main.kcl" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "file": "main.kcl", | ||||
|     "pathFromProjectDirectoryToFirstFile": "mug/main.kcl", | ||||
|     "multipleFiles": false, | ||||
|     "title": "Mug", | ||||
|     "description": "A cup with straight sides and a handle, typically used for hot drinks", | ||||
|     "files": [ | ||||
|       "main.kcl" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "file": "main.kcl", | ||||
|     "pathFromProjectDirectoryToFirstFile": "multi-axis-robot/main.kcl", | ||||
| @ -612,6 +622,16 @@ | ||||
|       "main.kcl" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "file": "main.kcl", | ||||
|     "pathFromProjectDirectoryToFirstFile": "spool/main.kcl", | ||||
|     "multipleFiles": false, | ||||
|     "title": "Spool", | ||||
|     "description": "A spool is a winding device for storing large batches of filament, wire, tape, etc", | ||||
|     "files": [ | ||||
|       "main.kcl" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "file": "main.kcl", | ||||
|     "pathFromProjectDirectoryToFirstFile": "spur-gear/main.kcl", | ||||
| @ -652,6 +672,16 @@ | ||||
|       "main.kcl" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "file": "main.kcl", | ||||
|     "pathFromProjectDirectoryToFirstFile": "teapot/main.kcl", | ||||
|     "multipleFiles": false, | ||||
|     "title": "Teapot", | ||||
|     "description": "A vessel with a spout and a handle used for serving tea and boiling water on a stovetop", | ||||
|     "files": [ | ||||
|       "main.kcl" | ||||
|     ] | ||||
|   }, | ||||
|   { | ||||
|     "file": "main.kcl", | ||||
|     "pathFromProjectDirectoryToFirstFile": "telemetry-antenna/main.kcl", | ||||
|  | ||||
							
								
								
									
										81
									
								
								public/kcl-samples/mug/main.kcl
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,81 @@ | ||||
| // Mug | ||||
| // A cup with straight sides and a handle, typically used for hot drinks | ||||
|  | ||||
| // Set units | ||||
| @settings(defaultLengthUnit = mm) | ||||
|  | ||||
| // Define parameters | ||||
| mugDiameter = 80 | ||||
| mugHeight = 95 | ||||
| wallThickness = 6 | ||||
| handleWidth = 18 | ||||
| handleThickness = 5 | ||||
|  | ||||
| // Revolve the outer perimeter of the mug body. This will be used to subtract material from the handle, so that it fits snug to the body | ||||
| tools = startSketchOn(XZ) | ||||
|   |> startProfile(at = [0, mugDiameter * 0.04]) | ||||
|   |> xLine(length = mugDiameter * 0.1) | ||||
|   |> tangentialArc(endAbsolute = [mugDiameter * 0.3, mugDiameter * 0.01]) | ||||
|   |> arc(angleStart = -180, angleEnd = -90, radius = mugDiameter * 0.01) | ||||
|   |> xLine(length = mugDiameter * 0.025) | ||||
|   |> arc(angleStart = -90, angleEnd = 0, radius = mugDiameter * 0.01) | ||||
|   |> xLine(endAbsolute = mugDiameter * 0.35) | ||||
|   |> tangentialArc(angle = 90, radius = mugDiameter * 0.15) | ||||
|   |> yLine(endAbsolute = mugHeight - (wallThickness / 2)) | ||||
|   |> tangentialArc(angle = 90, diameter = wallThickness) | ||||
|   |> xLine(endAbsolute = profileStartX()) | ||||
|   |> yLine(endAbsolute = profileStartY()) | ||||
|   |> close() | ||||
|   |> revolve(axis = Y) | ||||
|  | ||||
| // Draw the path of the mug handle | ||||
| handlePath = startSketchOn(XZ) | ||||
|   |> startProfile(at = [mugDiameter / 3, mugHeight * 0.25]) | ||||
|   |> angledLine(angle = -5, length = mugDiameter / 3) | ||||
|   |> tangentialArc(angle = 90, radius = mugHeight * 0.15) | ||||
|   |> tangentialArc(end = [0, mugHeight * 0.425]) | ||||
|   |> tangentialArc(angle = 90, radius = mugHeight * 0.15, tag = $seg02) | ||||
|   |> angledLine(angle = tangentToEnd(seg02), endAbsoluteX = profileStartX()) | ||||
|  | ||||
| // Sweep the handle profile along the path | ||||
| handle = startSketchOn(offsetPlane(YZ, offset = mugDiameter / 3)) | ||||
|   |> startProfile(at = [ | ||||
|        -handleWidth / 2, | ||||
|        mugHeight * 0.25 + handleThickness / 2 | ||||
|      ]) | ||||
|   |> arc( | ||||
|        interiorAbsolute = [ | ||||
|          0, | ||||
|          profileStartY() + handleThickness / 4 | ||||
|        ], | ||||
|        endAbsolute = [handleWidth / 2, profileStartY()], | ||||
|      ) | ||||
|   |> tangentialArc(end = [0, -handleThickness]) | ||||
|   |> tangentialArc(end = [-handleWidth, 0]) | ||||
|   |> tangentialArc(endAbsolute = profileStart()) | ||||
|   |> close() | ||||
|   |> sweep(path = handlePath) | ||||
|   |> subtract(tools) | ||||
|  | ||||
| // Create the body of the mug | ||||
| mug = startSketchOn(XZ) | ||||
|   |> startProfile(at = [0, mugDiameter * 0.04]) | ||||
|   |> xLine(length = mugDiameter * 0.1) | ||||
|   |> tangentialArc(endAbsolute = [mugDiameter * 0.3, mugDiameter * 0.01]) | ||||
|   |> arc(angleStart = -180, angleEnd = -90, radius = mugDiameter * 0.01) | ||||
|   |> xLine(length = mugDiameter * 0.025) | ||||
|   |> arc(angleStart = -90, angleEnd = 0, radius = mugDiameter * 0.01) | ||||
|   |> xLine(endAbsolute = mugDiameter * 0.35) | ||||
|   |> tangentialArc(angle = 90, radius = mugDiameter * 0.15) | ||||
|   |> yLine(endAbsolute = mugHeight - (wallThickness / 2)) | ||||
|   |> tangentialArc(angle = 180, diameter = wallThickness) | ||||
|   |> yLine(endAbsolute = mugDiameter * 0.15) | ||||
|   |> tangentialArc(angle = -90, radius = lastSegX() * 0.2) | ||||
|   |> xLine(endAbsolute = profileStartX()) | ||||
|   |> yLine(endAbsolute = profileStartY()) | ||||
|   |> close() | ||||
|   |> revolve(axis = Y) | ||||
|  | ||||
| // Join the mug and handle, then add an appearance | ||||
| [handle, mug] | ||||
|   |> appearance(color = "#9b1212", metalness = 40, roughness = 30) | ||||
							
								
								
									
										
											BIN
										
									
								
								public/kcl-samples/screenshots/mug.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 114 KiB | 
