Compare commits
	
		
			2 Commits
		
	
	
		
			franknoiro
			...
			nrc-conver
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 865c129c60 | |||
| 3e225d78c2 | 
| @ -1,3 +1,3 @@ | ||||
| [codespell] | ||||
| ignore-words-list: crate,everytime,inout,co-ordinate,ot,nwo,atleast,ue,afterall,ser,fromM,FromM | ||||
| ignore-words-list: crate,everytime,inout,co-ordinate,ot,nwo,atleast,ue,afterall,ser | ||||
| skip: **/target,node_modules,build,dist,./out,**/Cargo.lock,./docs/kcl/*.md,.yarn.lock,**/yarn.lock,./openapi/*.json,./packages/codemirror-lang-kcl/test/all.test.ts,./public/kcl-samples,./rust/kcl-lib/tests/kcl_samples,tsconfig.tsbuildinfo | ||||
|  | ||||
							
								
								
									
										24
									
								
								.github/ci-cd-scripts/start-vector-macos.sh
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										24
									
								
								.github/ci-cd-scripts/start-vector-macos.sh
									
									
									
									
										vendored
									
									
								
							| @ -1,24 +0,0 @@ | ||||
| #!/bin/bash | ||||
| set -euo pipefail | ||||
|  | ||||
| # Install vector | ||||
| brew tap vectordotdev/brew && brew install vector | ||||
|  | ||||
| # Configure vector | ||||
| mkdir -p /tmp/vector | ||||
| cp .github/workflows/vector.toml /tmp/vector.toml | ||||
| sed -i '' "s#OS_NAME#${OS_NAME}#g" /tmp/vector.toml | ||||
| sed -i '' "s#GITHUB_WORKFLOW#${GITHUB_WORKFLOW}#g" /tmp/vector.toml | ||||
| sed -i '' "s#GITHUB_REPOSITORY#${GITHUB_REPOSITORY}#g" /tmp/vector.toml | ||||
| sed -i '' "s#GITHUB_SHA#${GITHUB_SHA}#g" /tmp/vector.toml | ||||
| sed -i '' "s#GITHUB_REF_NAME#${GITHUB_REF_NAME}#g" /tmp/vector.toml | ||||
| sed -i '' "s#GH_ACTIONS_AXIOM_TOKEN#${GH_ACTIONS_AXIOM_TOKEN}#g" /tmp/vector.toml | ||||
|  | ||||
| # Display settings | ||||
| echo | ||||
| echo 'Vector config:' | ||||
| cat /tmp/vector.toml | ||||
| echo | ||||
|  | ||||
| # Start in the background | ||||
| $(brew --prefix)/opt/vector/bin/vector --config /tmp/vector.toml & | ||||
							
								
								
									
										24
									
								
								.github/ci-cd-scripts/start-vector-ubuntu.sh
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										24
									
								
								.github/ci-cd-scripts/start-vector-ubuntu.sh
									
									
									
									
										vendored
									
									
								
							| @ -1,24 +0,0 @@ | ||||
| #!/bin/bash | ||||
| set -euo pipefail | ||||
|  | ||||
| # Install vector | ||||
| curl --proto '=https' --tlsv1.2 -sSfL https://sh.vector.dev | bash -s -- -y | ||||
|  | ||||
| # Configure vector | ||||
| mkdir -p /tmp/vector | ||||
| cp .github/workflows/vector.toml /tmp/vector.toml | ||||
| sed -i "s#OS_NAME#${OS_NAME}#g" /tmp/vector.toml | ||||
| sed -i "s#GITHUB_WORKFLOW#${GITHUB_WORKFLOW}#g" /tmp/vector.toml | ||||
| sed -i "s#GITHUB_REPOSITORY#${GITHUB_REPOSITORY}#g" /tmp/vector.toml | ||||
| sed -i "s#GITHUB_SHA#${GITHUB_SHA}#g" /tmp/vector.toml | ||||
| sed -i "s#GITHUB_REF_NAME#${GITHUB_REF_NAME}#g" /tmp/vector.toml | ||||
| sed -i "s#GH_ACTIONS_AXIOM_TOKEN#${GH_ACTIONS_AXIOM_TOKEN}#g" /tmp/vector.toml | ||||
|  | ||||
| # Display settings | ||||
| echo | ||||
| echo 'Vector config:' | ||||
| cat /tmp/vector.toml | ||||
| echo | ||||
|  | ||||
| # Start in background | ||||
| ${HOME}/.vector/bin/vector --config /tmp/vector.toml & | ||||
							
								
								
									
										2
									
								
								.github/workflows/build-and-store-wasm.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/build-and-store-wasm.yml
									
									
									
									
										vendored
									
									
								
							| @ -24,7 +24,7 @@ jobs: | ||||
|         uses: actions-rust-lang/setup-rust-toolchain@v1 | ||||
|         with: | ||||
|           cache: false # Configured below. | ||||
|       - uses: taiki-e/install-action@d4635f2de61c8b8104d59cd4aede2060638378cc | ||||
|       - uses: taiki-e/install-action@37bdc826eaedac215f638a96472df572feab0f9b | ||||
|         with: | ||||
|           tool: wasm-pack | ||||
|       - name: Rust Cache | ||||
|  | ||||
							
								
								
									
										2
									
								
								.github/workflows/build-apps.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/build-apps.yml
									
									
									
									
										vendored
									
									
								
							| @ -77,7 +77,7 @@ jobs: | ||||
|         with: | ||||
|           cache: false # Configured below. | ||||
|  | ||||
|       - uses: taiki-e/install-action@d4635f2de61c8b8104d59cd4aede2060638378cc | ||||
|       - uses: taiki-e/install-action@37bdc826eaedac215f638a96472df572feab0f9b | ||||
|         if: ${{ steps.wasm.outputs.should-build-wasm == 'true' }} | ||||
|         with: | ||||
|           tool: wasm-pack | ||||
|  | ||||
							
								
								
									
										19
									
								
								.github/workflows/cargo-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										19
									
								
								.github/workflows/cargo-test.yml
									
									
									
									
										vendored
									
									
								
							| @ -34,11 +34,20 @@ jobs: | ||||
|         uses: actions-rust-lang/setup-rust-toolchain@v1 | ||||
|         with: | ||||
|           cache: false # Configured below. | ||||
|       - name: Start Vector | ||||
|         run: .github/ci-cd-scripts/start-vector-ubuntu.sh | ||||
|         env: | ||||
|           GH_ACTIONS_AXIOM_TOKEN: ${{ secrets.GH_ACTIONS_AXIOM_TOKEN }} | ||||
|           OS_NAME: ${{ env.OS_NAME }} | ||||
|       - name: Install vector | ||||
|         run: | | ||||
|           curl --proto '=https' --tlsv1.2 -sSfL https://sh.vector.dev > /tmp/vector.sh | ||||
|           chmod +x /tmp/vector.sh | ||||
|           /tmp/vector.sh -y -no-modify-path | ||||
|           mkdir -p /tmp/vector | ||||
|           cp .github/workflows/vector.toml /tmp/vector.toml | ||||
|           sed -i "s#GITHUB_WORKFLOW#${GITHUB_WORKFLOW}#g" /tmp/vector.toml | ||||
|           sed -i "s#GITHUB_REPOSITORY#${GITHUB_REPOSITORY}#g" /tmp/vector.toml | ||||
|           sed -i "s#GITHUB_SHA#${GITHUB_SHA}#g" /tmp/vector.toml | ||||
|           sed -i "s#GITHUB_REF_NAME#${GITHUB_REF_NAME}#g" /tmp/vector.toml | ||||
|           sed -i "s#GH_ACTIONS_AXIOM_TOKEN#${{secrets.GH_ACTIONS_AXIOM_TOKEN}}#g" /tmp/vector.toml | ||||
|           cat /tmp/vector.toml | ||||
|           ${HOME}/.vector/bin/vector --config /tmp/vector.toml & | ||||
|       - uses: taiki-e/install-action@cargo-llvm-cov | ||||
|       - uses: taiki-e/install-action@nextest | ||||
|       - name: Install just | ||||
|  | ||||
							
								
								
									
										24
									
								
								.github/workflows/e2e-tests.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										24
									
								
								.github/workflows/e2e-tests.yml
									
									
									
									
										vendored
									
									
								
							| @ -140,7 +140,7 @@ jobs: | ||||
|         with: | ||||
|           cache: false # Configured below. | ||||
|  | ||||
|       - uses: taiki-e/install-action@d4635f2de61c8b8104d59cd4aede2060638378cc | ||||
|       - uses: taiki-e/install-action@37bdc826eaedac215f638a96472df572feab0f9b | ||||
|         if: ${{ needs.conditions.outputs.should-run == 'true' && steps.wasm.outputs.should-build-wasm == 'true' }} | ||||
|         with: | ||||
|           tool: wasm-pack | ||||
| @ -339,12 +339,22 @@ jobs: | ||||
|         if: needs.conditions.outputs.should-run == 'true' | ||||
|         run: yarn tronb:vite:dev | ||||
|  | ||||
|       - name: Start Vector | ||||
|         if: ${{ needs.conditions.outputs.should-run == 'true' && !contains(matrix.os, 'windows') }} | ||||
|         run: .github/ci-cd-scripts/start-vector-${{ env.OS_NAME }}.sh | ||||
|         env: | ||||
|           GH_ACTIONS_AXIOM_TOKEN: ${{ secrets.GH_ACTIONS_AXIOM_TOKEN }} | ||||
|           OS_NAME: ${{ env.OS_NAME }} | ||||
|       - name: Install vector | ||||
|         if: ${{ needs.conditions.outputs.should-run == 'true' && contains(matrix.os, 'ubuntu') }} | ||||
|         shell: bash | ||||
|         run: | | ||||
|           curl --proto '=https' --tlsv1.2 -sSfL https://sh.vector.dev > /tmp/vector.sh | ||||
|           chmod +x /tmp/vector.sh | ||||
|           /tmp/vector.sh -y -no-modify-path | ||||
|           mkdir -p /tmp/vector | ||||
|           cp .github/workflows/vector.toml /tmp/vector.toml | ||||
|           sed -i "s#GITHUB_WORKFLOW#${GITHUB_WORKFLOW}#g" /tmp/vector.toml | ||||
|           sed -i "s#GITHUB_REPOSITORY#${GITHUB_REPOSITORY}#g" /tmp/vector.toml | ||||
|           sed -i "s#GITHUB_SHA#${GITHUB_SHA}#g" /tmp/vector.toml | ||||
|           sed -i "s#GITHUB_REF_NAME#${GITHUB_REF_NAME}#g" /tmp/vector.toml | ||||
|           sed -i "s#GH_ACTIONS_AXIOM_TOKEN#${{secrets.GH_ACTIONS_AXIOM_TOKEN}}#g" /tmp/vector.toml | ||||
|           cat /tmp/vector.toml | ||||
|           ${HOME}/.vector/bin/vector --config /tmp/vector.toml & | ||||
|  | ||||
|       - uses: actions/download-artifact@v4 | ||||
|         if: ${{ needs.conditions.outputs.should-run == 'true' && !cancelled() && (success() || failure()) }} | ||||
|  | ||||
							
								
								
									
										6
									
								
								.github/workflows/static-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/static-analysis.yml
									
									
									
									
										vendored
									
									
								
							| @ -51,7 +51,7 @@ jobs: | ||||
|         with: | ||||
|           cache: false # Configured below. | ||||
|  | ||||
|       - uses: taiki-e/install-action@d4635f2de61c8b8104d59cd4aede2060638378cc | ||||
|       - uses: taiki-e/install-action@37bdc826eaedac215f638a96472df572feab0f9b | ||||
|         with: | ||||
|           tool: wasm-pack | ||||
|  | ||||
| @ -191,7 +191,7 @@ jobs: | ||||
|           cache: 'yarn' | ||||
|  | ||||
|       - run: yarn install | ||||
|       - uses: taiki-e/install-action@d4635f2de61c8b8104d59cd4aede2060638378cc | ||||
|       - uses: taiki-e/install-action@37bdc826eaedac215f638a96472df572feab0f9b | ||||
|         with: | ||||
|           tool: wasm-pack | ||||
|  | ||||
| @ -236,7 +236,7 @@ jobs: | ||||
|           cache: 'yarn' | ||||
|  | ||||
|       - run: yarn install | ||||
|       - uses: taiki-e/install-action@d4635f2de61c8b8104d59cd4aede2060638378cc | ||||
|       - uses: taiki-e/install-action@37bdc826eaedac215f638a96472df572feab0f9b | ||||
|         with: | ||||
|           tool: wasm-pack | ||||
|  | ||||
|  | ||||
							
								
								
									
										1
									
								
								.github/workflows/vector.toml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/vector.toml
									
									
									
									
										vendored
									
									
								
							| @ -8,7 +8,6 @@ include = ["/tmp/github-actions.log"] | ||||
| type = "remap" | ||||
| inputs = [ "github-actions-file" ] | ||||
| source = ''' | ||||
| .platform = "OS_NAME" | ||||
| .action = "GITHUB_WORKFLOW" | ||||
| .repo = "GITHUB_REPOSITORY" | ||||
| .sha = "GITHUB_SHA" | ||||
|  | ||||
							
								
								
									
										31
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										31
									
								
								Makefile
									
									
									
									
									
								
							| @ -5,40 +5,33 @@ all: install build check | ||||
| # INSTALL | ||||
|  | ||||
| ifeq ($(OS),Windows_NT) | ||||
| export WINDOWS := true | ||||
| ifndef MSYSTEM | ||||
| export POWERSHELL := true | ||||
| endif | ||||
| endif | ||||
|  | ||||
| ifdef WINDOWS | ||||
| CARGO ?= $(USERPROFILE)/.cargo/bin/cargo.exe | ||||
| WASM_PACK ?= $(USERPROFILE)/.cargo/bin/wasm-pack.exe | ||||
| 	CARGO ?= ~/.cargo/bin/cargo.exe | ||||
| 	WASM_PACK ?= ~/.cargo/bin/wasm-pack.exe | ||||
| else | ||||
| CARGO ?= ~/.cargo/bin/cargo | ||||
| WASM_PACK ?= ~/.cargo/bin/wasm-pack | ||||
| endif  | ||||
| 	CARGO ?= ~/.cargo/bin/cargo | ||||
| 	WASM_PACK ?= ~/.cargo/bin/wasm-pack | ||||
| endif | ||||
|  | ||||
| .PHONY: install | ||||
| install: node_modules/.yarn-integrity $(CARGO) $(WASM_PACK) ## Install dependencies | ||||
|  | ||||
| node_modules/.yarn-integrity: package.json yarn.lock | ||||
| 	yarn install | ||||
| ifdef POWERSHELL | ||||
| ifeq ($(OS),Windows_NT) | ||||
| 	@ type nul > $@ | ||||
| else | ||||
| 	@ touch $@ | ||||
| endif | ||||
|  | ||||
| $(CARGO): | ||||
| ifdef WINDOWS | ||||
| ifeq ($(OS),Windows_NT) | ||||
| 	yarn install:rust:windows | ||||
| else | ||||
| 	yarn install:rust | ||||
| endif | ||||
|  | ||||
| $(WASM_PACK): | ||||
| ifdef WINDOWS | ||||
| ifeq ($(OS),Windows_NT) | ||||
| 	yarn install:wasm-pack:cargo | ||||
| else | ||||
| 	yarn install:wasm-pack:sh | ||||
| @ -64,7 +57,7 @@ build-web: install public/kcl_wasm_lib_bg.wasm build/index.html | ||||
| build-desktop: install public/kcl_wasm_lib_bg.wasm .vite/build/main.js | ||||
|  | ||||
| public/kcl_wasm_lib_bg.wasm: $(CARGO_SOURCES) $(RUST_SOURCES) | ||||
| ifdef WINDOWS | ||||
| ifeq ($(OS),Windows_NT) | ||||
| 	yarn build:wasm:dev:windows | ||||
| else | ||||
| 	yarn build:wasm:dev | ||||
| @ -147,8 +140,8 @@ endif | ||||
|  | ||||
| .PHONY: clean | ||||
| clean: ## Delete all artifacts | ||||
| ifdef POWERSHELL | ||||
| 	git clean --force -d -x --exclude=.env* --exclude=**/*.env | ||||
| ifeq ($(OS),Windows_NT) | ||||
| 	git clean --force -d -X | ||||
| else | ||||
| 	rm -rf .vite/ build/ | ||||
| 	rm -rf trace.zip playwright-report/ test-results/ | ||||
| @ -159,7 +152,7 @@ endif | ||||
|  | ||||
| .PHONY: help | ||||
| help: install | ||||
| ifdef POWERSHELL | ||||
| ifeq ($(OS),Windows_NT) | ||||
| 	@ powershell -Command "Get-Content $(MAKEFILE_LIST) | Select-String -Pattern '^[^\s]+:.*##\s.*$$' | ForEach-Object { $$line = $$_.Line -split ':.*?##\s+'; Write-Host -NoNewline $$line[0].PadRight(30) -ForegroundColor Cyan; Write-Host $$line[1] }" | ||||
| else | ||||
| 	@ grep -E '^[^[:space:]]+:.*## .*$$' $(MAKEFILE_LIST) | awk 'BEGIN {FS = ":.*?## "}; {printf "\033[36m%-30s\033[0m %s\n", $$1, $$2}' | ||||
|  | ||||
| @ -36,9 +36,9 @@ myAngle = -120 | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line(end = [8, 0]) | ||||
|   |> angledLine(angle = abs(myAngle), length = 5) | ||||
|   |> angledLine({ angle = abs(myAngle), length = 5 }, %) | ||||
|   |> line(end = [-5, 0]) | ||||
|   |> angledLine(angle = myAngle, length = 5) | ||||
|   |> angledLine({ angle = myAngle, length = 5 }, %) | ||||
|   |> close() | ||||
|  | ||||
| baseExtrusion = extrude(sketch001, length = 5) | ||||
|  | ||||
| @ -33,7 +33,10 @@ acos(num: number): number | ||||
| ```js | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = toDegrees(acos(0.5)), length = 10) | ||||
|   |> angledLine({ | ||||
|        angle = toDegrees(acos(0.5)), | ||||
|        length = 10 | ||||
|      }, %) | ||||
|   |> line(end = [5, 0]) | ||||
|   |> line(endAbsolute = [12, 0]) | ||||
|   |> close() | ||||
|  | ||||
| @ -36,7 +36,7 @@ angleToMatchLengthX( | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line(end = [2, 5], tag = $seg01) | ||||
|   |> angledLine(angle = -angleToMatchLengthX(seg01, 7, %), endAbsoluteX = 10) | ||||
|   |> angledLineToX([-angleToMatchLengthX(seg01, 7, %), 10], %) | ||||
|   |> close() | ||||
|  | ||||
| extrusion = extrude(sketch001, length = 5) | ||||
|  | ||||
| @ -36,7 +36,10 @@ angleToMatchLengthY( | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line(end = [1, 2], tag = $seg01) | ||||
|   |> angledLine(angle = angleToMatchLengthY(seg01, 15, %), length = 5) | ||||
|   |> angledLine({ | ||||
|        angle = angleToMatchLengthY(seg01, 15, %), | ||||
|        length = 5 | ||||
|      }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -10,13 +10,8 @@ Draw a line segment relative to the current origin using the polar measure of so | ||||
|  | ||||
| ```js | ||||
| angledLine( | ||||
|   data: AngledLineData, | ||||
|   sketch: Sketch, | ||||
|   angle: number, | ||||
|   length?: number, | ||||
|   lengthX?: number, | ||||
|   lengthY?: number, | ||||
|   endAbsoluteX?: number, | ||||
|   endAbsoluteY?: number, | ||||
|   tag?: TagDeclarator, | ||||
| ): Sketch | ||||
| ``` | ||||
| @ -26,14 +21,9 @@ angledLine( | ||||
|  | ||||
| | Name | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `sketch` | [`Sketch`](/docs/kcl/types/Sketch) | Which sketch should this path be added to? | Yes | | ||||
| | `angle` | [`number`](/docs/kcl/types/number) | Which angle should the line be drawn at? | Yes | | ||||
| | `length` | [`number`](/docs/kcl/types/number) | Draw the line this distance along the given angle. Only one of `length`, `lengthX`, `lengthY`, `lengthAbsoluteEndX`, `lengthAbsoluteEndY` can be given. | No | | ||||
| | `lengthX` | [`number`](/docs/kcl/types/number) | Draw the line this distance along the X axis. Only one of `length`, `lengthX`, `lengthY`, `lengthAbsoluteEndX`, `lengthAbsoluteEndY` can be given. | No | | ||||
| | `lengthY` | [`number`](/docs/kcl/types/number) | Draw the line this distance along the Y axis. Only one of `length`, `lengthX`, `lengthY`, `lengthAbsoluteEndX`, `lengthAbsoluteEndY` can be given. | No | | ||||
| | `endAbsoluteX` | [`number`](/docs/kcl/types/number) | Draw the line along the given angle until it reaches this point along the X axis. Only one of `length`, `lengthX`, `lengthY`, `lengthAbsoluteEndX`, `lengthAbsoluteEndY` can be given. | No | | ||||
| | `endAbsoluteY` | [`number`](/docs/kcl/types/number) | Draw the line along the given angle until it reaches this point along the Y axis. Only one of `length`, `lengthX`, `lengthY`, `lengthAbsoluteEndX`, `lengthAbsoluteEndY` can be given. | No | | ||||
| | [`tag`](/docs/kcl/types/tag) | [`TagDeclarator`](/docs/kcl/types#tag-declaration) | Create a new tag which refers to this line | No | | ||||
| | `data` | [`AngledLineData`](/docs/kcl/types/AngledLineData) | Data to draw an angled line. | Yes | | ||||
| | `sketch` | [`Sketch`](/docs/kcl/types/Sketch) |  | Yes | | ||||
| | [`tag`](/docs/kcl/types/tag) | [`TagDeclarator`](/docs/kcl/types#tag-declaration) |  | No | | ||||
|  | ||||
| ### Returns | ||||
|  | ||||
| @ -46,7 +36,7 @@ angledLine( | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> yLine(endAbsolute = 15) | ||||
|   |> angledLine(angle = 30, length = 15) | ||||
|   |> angledLine({ angle = 30, length = 15 }, %) | ||||
|   |> line(end = [8, -10]) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
							
								
								
									
										48
									
								
								docs/kcl/angledLineOfXLength.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								docs/kcl/angledLineOfXLength.md
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										50
									
								
								docs/kcl/angledLineOfYLength.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								docs/kcl/angledLineOfYLength.md
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										48
									
								
								docs/kcl/angledLineToX.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								docs/kcl/angledLineToX.md
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										48
									
								
								docs/kcl/angledLineToY.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								docs/kcl/angledLineToY.md
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -33,7 +33,10 @@ asin(num: number): number | ||||
| ```js | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = toDegrees(asin(0.5)), length = 20) | ||||
|   |> angledLine({ | ||||
|        angle = toDegrees(asin(0.5)), | ||||
|        length = 20 | ||||
|      }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -33,7 +33,10 @@ atan(num: number): number | ||||
| ```js | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = toDegrees(atan(1.25)), length = 20) | ||||
|   |> angledLine({ | ||||
|        angle = toDegrees(atan(1.25)), | ||||
|        length = 20 | ||||
|      }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -37,7 +37,10 @@ atan2( | ||||
| ```js | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = toDegrees(atan2(1.25, 2)), length = 20) | ||||
|   |> angledLine({ | ||||
|        angle = toDegrees(atan2(1.25, 2)), | ||||
|        length = 20 | ||||
|      }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -17,10 +17,10 @@ std::math::E: number = 2.71828182845904523536028747135266250_ | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine( | ||||
|   |> angledLine({ | ||||
|     angle = 30, | ||||
|     length = 2 * E ^ 2, | ||||
|   ) | ||||
|   }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -17,10 +17,10 @@ std::math::TAU: number = 6.28318530717958647692528676655900577_ | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine( | ||||
|   |> angledLine({ | ||||
|     angle = 50, | ||||
|     length = 10 * TAU, | ||||
|   ) | ||||
|   }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -30,7 +30,7 @@ e(): number | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 30, length = 2 * e() ^ 2) | ||||
|   |> angledLine({ angle = 30, length = 2 * e() ^ 2 }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -8,11 +8,11 @@ Converts a number from centimeters to the current default unit. | ||||
|  | ||||
| No matter what units the current file uses, this function will always return a number equivalent to the input in centimeters. | ||||
|  | ||||
| For example, if the current file uses inches, `fromCm(1)` will return `0.393701`. If the current file uses millimeters, `fromCm(1)` will return `10`. If the current file uses centimeters, `fromCm(1)` will return `1`. | ||||
| For example, if the current project uses inches, `fromCm` will return `0.393701`. If the current project uses millimeters, `fromCm` will return `10`. If the current project uses centimeters, `fromCm` will return `1`. | ||||
|  | ||||
| **Caution**: This function is only intended to be used when you absolutely MUST have different units in your code than the file settings. Otherwise, it is a bad pattern to use this function. | ||||
| **Caution**: This function is only intended to be used when you absolutely MUST have different units in your code than the project settings. Otherwise, it is a bad pattern to use this function. | ||||
|  | ||||
| We merely provide these functions for convenience and readability, as `fromCm(10)` is more readable that your intent is "I want 10 centimeters" than `10 * 10`, if the file settings are in millimeters. | ||||
| We merely provide these functions for convenience and readability, as `fromCm(10)` is more readable that your intent is "I want 10 centimeters" than `10 * 10`, if the project settings are in millimeters. | ||||
|  | ||||
| ```js | ||||
| fromCm(input: number): number | ||||
|  | ||||
| @ -8,11 +8,11 @@ Converts a number from feet to the current default unit. | ||||
|  | ||||
| No matter what units the current file uses, this function will always return a number equivalent to the input in feet. | ||||
|  | ||||
| For example, if the current file uses inches, `fromFt(1)` will return `12`. If the current file uses millimeters, `fromFt(1)` will return `304.8`. If the current file uses feet, `fromFt(1)` will return `1`. | ||||
| For example, if the current project uses inches, `fromFt(1)` will return `12`. If the current project uses millimeters, `fromFt(1)` will return `304.8`. If the current project uses feet, `fromFt(1)` will return `1`. | ||||
|  | ||||
| **Caution**: This function is only intended to be used when you absolutely MUST have different units in your code than the file settings. Otherwise, it is a bad pattern to use this function. | ||||
| **Caution**: This function is only intended to be used when you absolutely MUST have different units in your code than the project settings. Otherwise, it is a bad pattern to use this function. | ||||
|  | ||||
| We merely provide these functions for convenience and readability, as `fromFt(10)` is more readable that your intent is "I want 10 feet" than `10 * 304.8`, if the file settings are in millimeters. | ||||
| We merely provide these functions for convenience and readability, as `fromFt(10)` is more readable that your intent is "I want 10 feet" than `10 * 304.8`, if the project settings are in millimeters. | ||||
|  | ||||
| ```js | ||||
| fromFt(input: number): number | ||||
|  | ||||
| @ -8,11 +8,11 @@ Converts a number from inches to the current default unit. | ||||
|  | ||||
| No matter what units the current file uses, this function will always return a number equivalent to the input in inches. | ||||
|  | ||||
| For example, if the current file uses inches, `fromInches(1)` will return `1`. If the current file uses millimeters, `fromInches(1)` will return `25.4`. | ||||
| For example, if the current project uses inches, `fromInches(1)` will return `1`. If the current project uses millimeters, `fromInches(1)` will return `25.4`. | ||||
|  | ||||
| **Caution**: This function is only intended to be used when you absolutely MUST have different units in your code than the file settings. Otherwise, it is a bad pattern to use this function. | ||||
| **Caution**: This function is only intended to be used when you absolutely MUST have different units in your code than the project settings. Otherwise, it is a bad pattern to use this function. | ||||
|  | ||||
| We merely provide these functions for convenience and readability, as `fromInches(10)` is more readable that your intent is "I want 10 inches" than `10 * 25.4`, if the file settings are in millimeters. | ||||
| We merely provide these functions for convenience and readability, as `fromInches(10)` is more readable that your intent is "I want 10 inches" than `10 * 25.4`, if the project settings are in millimeters. | ||||
|  | ||||
| ```js | ||||
| fromInches(input: number): number | ||||
|  | ||||
| @ -8,11 +8,11 @@ Converts a number from meters to the current default unit. | ||||
|  | ||||
| No matter what units the current file uses, this function will always return a number equivalent to the input in meters. | ||||
|  | ||||
| For example, if the current file uses inches, `fromM(1)` will return `39.3701`. If the current file uses millimeters, `fromM(1)` will return `1000`. If the current file uses meters, `fromM(1)` will return `1`. | ||||
| For example, if the current project uses inches, `fromM` will return `39.3701`. If the current project uses millimeters, `fromM` will return `1000`. If the current project uses meters, `fromM` will return `1`. | ||||
|  | ||||
| **Caution**: This function is only intended to be used when you absolutely MUST have different units in your code than the file settings. Otherwise, it is a bad pattern to use this function. | ||||
| **Caution**: This function is only intended to be used when you absolutely MUST have different units in your code than the project settings. Otherwise, it is a bad pattern to use this function. | ||||
|  | ||||
| We merely provide these functions for convenience and readability, as `fromM(10)` is more readable that your intent is "I want 10 meters" than `10 * 1000`, if the file settings are in millimeters. | ||||
| We merely provide these functions for convenience and readability, as `fromM(10)` is more readable that your intent is "I want 10 meters" than `10 * 1000`, if the project settings are in millimeters. | ||||
|  | ||||
| ```js | ||||
| fromM(input: number): number | ||||
|  | ||||
| @ -8,11 +8,11 @@ Converts a number from mm to the current default unit. | ||||
|  | ||||
| No matter what units the current file uses, this function will always return a number equivalent to the input in millimeters. | ||||
|  | ||||
| For example, if the current file uses inches, `fromMm(1)` will return `1/25.4`. If the current file uses millimeters, `fromMm(1)` will return `1`. | ||||
| For example, if the current project uses inches, `fromMm(1)` will return `1/25.4`. If the current project uses millimeters, `fromMm(1)` will return `1`. | ||||
|  | ||||
| **Caution**: This function is only intended to be used when you absolutely MUST have different units in your code than the file settings. Otherwise, it is a bad pattern to use this function. | ||||
| **Caution**: This function is only intended to be used when you absolutely MUST have different units in your code than the project settings. Otherwise, it is a bad pattern to use this function. | ||||
|  | ||||
| We merely provide these functions for convenience and readability, as `fromMm(10)` is more readable that your intent is "I want 10 millimeters" than `10 * (1/25.4)`, if the file settings are in inches. | ||||
| We merely provide these functions for convenience and readability, as `fromMm(10)` is more readable that your intent is "I want 10 millimeters" than `10 * (1/25.4)`, if the project settings are in inches. | ||||
|  | ||||
| ```js | ||||
| fromMm(input: number): number | ||||
|  | ||||
| @ -8,11 +8,11 @@ Converts a number from yards to the current default unit. | ||||
|  | ||||
| No matter what units the current file uses, this function will always return a number equivalent to the input in yards. | ||||
|  | ||||
| For example, if the current file uses inches, `fromYd(1)` will return `36`. If the current file uses millimeters, `fromYd(1)` will return `914.4`. If the current file uses yards, `fromYd(1)` will return `1`. | ||||
| For example, if the current project uses inches, `fromYd` will return `36`. If the current project uses millimeters, `fromYd` will return `914.4`. If the current project uses yards, `fromYd` will return `1`. | ||||
|  | ||||
| **Caution**: This function is only intended to be used when you absolutely MUST have different units in your code than the file settings. Otherwise, it is a bad pattern to use this function. | ||||
| **Caution**: This function is only intended to be used when you absolutely MUST have different units in your code than the project settings. Otherwise, it is a bad pattern to use this function. | ||||
|  | ||||
| We merely provide these functions for convenience and readability, as `fromYd(10)` is more readable that your intent is "I want 10 yards" than `10 * 914.4`, if the file settings are in millimeters. | ||||
| We merely provide these functions for convenience and readability, as `fromYd(10)` is more readable that your intent is "I want 10 yards" than `10 * 914.4`, if the project settings are in millimeters. | ||||
|  | ||||
| ```js | ||||
| fromYd(input: number): number | ||||
|  | ||||
| @ -30,10 +30,10 @@ getNextAdjacentEdge(tag: TagIdentifier): Uuid | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line(end = [10, 0]) | ||||
|   |> angledLine(angle = 60, length = 10) | ||||
|   |> angledLine(angle = 120, length = 10) | ||||
|   |> angledLine({ angle = 60, length = 10 }, %) | ||||
|   |> angledLine({ angle = 120, length = 10 }, %) | ||||
|   |> line(end = [-10, 0]) | ||||
|   |> angledLine(angle = 240, length = 10, tag = $referenceEdge) | ||||
|   |> angledLine({ angle = 240, length = 10 }, %, $referenceEdge) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(exampleSketch, length = 5) | ||||
|  | ||||
| @ -30,10 +30,10 @@ getOppositeEdge(tag: TagIdentifier): Uuid | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line(end = [10, 0]) | ||||
|   |> angledLine(angle = 60, length = 10) | ||||
|   |> angledLine(angle = 120, length = 10) | ||||
|   |> angledLine({ angle = 60, length = 10 }, %) | ||||
|   |> angledLine({ angle = 120, length = 10 }, %) | ||||
|   |> line(end = [-10, 0]) | ||||
|   |> angledLine(angle = 240, length = 10, tag = $referenceEdge) | ||||
|   |> angledLine({ angle = 240, length = 10 }, %, $referenceEdge) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(exampleSketch, length = 5) | ||||
|  | ||||
| @ -30,10 +30,10 @@ getPreviousAdjacentEdge(tag: TagIdentifier): Uuid | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line(end = [10, 0]) | ||||
|   |> angledLine(angle = 60, length = 10) | ||||
|   |> angledLine(angle = 120, length = 10) | ||||
|   |> angledLine({ angle = 60, length = 10 }, %) | ||||
|   |> angledLine({ angle = 120, length = 10 }, %) | ||||
|   |> line(end = [-10, 0]) | ||||
|   |> angledLine(angle = 240, length = 10, tag = $referenceEdge) | ||||
|   |> angledLine({ angle = 240, length = 10 }, %, $referenceEdge) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(exampleSketch, length = 5) | ||||
|  | ||||
| @ -43,7 +43,11 @@ layout: manual | ||||
|   * [`angleToMatchLengthX`](kcl/angleToMatchLengthX) | ||||
|   * [`angleToMatchLengthY`](kcl/angleToMatchLengthY) | ||||
|   * [`angledLine`](kcl/angledLine) | ||||
|   * [`angledLineOfXLength`](kcl/angledLineOfXLength) | ||||
|   * [`angledLineOfYLength`](kcl/angledLineOfYLength) | ||||
|   * [`angledLineThatIntersects`](kcl/angledLineThatIntersects) | ||||
|   * [`angledLineToX`](kcl/angledLineToX) | ||||
|   * [`angledLineToY`](kcl/angledLineToY) | ||||
|   * [`appearance`](kcl/appearance) | ||||
|   * [`arc`](kcl/arc) | ||||
|   * [`arcTo`](kcl/arcTo) | ||||
| @ -98,6 +102,7 @@ layout: manual | ||||
|   * [`patternLinear3d`](kcl/patternLinear3d) | ||||
|   * [`patternTransform`](kcl/patternTransform) | ||||
|   * [`patternTransform2d`](kcl/patternTransform2d) | ||||
|   * [`polar`](kcl/polar) | ||||
|   * [`polygon`](kcl/polygon) | ||||
|   * [`pop`](kcl/pop) | ||||
|   * [`pow`](kcl/pow) | ||||
| @ -138,7 +143,6 @@ layout: manual | ||||
|   * [`PI`](kcl/consts/std-math-PI) | ||||
|   * [`TAU`](kcl/consts/std-math-TAU) | ||||
|   * [`cos`](kcl/std-math-cos) | ||||
|   * [`polar`](kcl/std-math-polar) | ||||
|   * [`sin`](kcl/std-math-sin) | ||||
|   * [`tan`](kcl/std-math-tan) | ||||
| * **std::sketch** | ||||
|  | ||||
| @ -33,7 +33,10 @@ max(args: [number]): number | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 70, length = max(15, 31, 4, 13, 22)) | ||||
|   |> angledLine({ | ||||
|        angle = 70, | ||||
|        length = max(15, 31, 4, 13, 22) | ||||
|      }, %) | ||||
|   |> line(end = [20, 0]) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -33,7 +33,10 @@ min(args: [number]): number | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 70, length = min(15, 31, 4, 13, 22)) | ||||
|   |> angledLine({ | ||||
|        angle = 70, | ||||
|        length = min(15, 31, 4, 13, 22) | ||||
|      }, %) | ||||
|   |> line(end = [20, 0]) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
							
								
								
									
										43
									
								
								docs/kcl/polar.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								docs/kcl/polar.md
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -37,7 +37,7 @@ pow( | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 50, length = pow(5, 2)) | ||||
|   |> angledLine({ angle = 50, length = pow(5, 2) }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -29,8 +29,11 @@ profileStart(sketch: Sketch): [number] | ||||
| ```js | ||||
| sketch001 = startSketchOn(XY) | ||||
|   |> startProfileAt([5, 2], %) | ||||
|   |> angledLine(angle = 120, length = 50, tag = $seg01) | ||||
|   |> angledLine(angle = segAng(seg01) + 120, length = 50) | ||||
|   |> angledLine({ angle = 120, length = 50 }, %, $seg01) | ||||
|   |> angledLine({ | ||||
|        angle = segAng(seg01) + 120, | ||||
|        length = 50 | ||||
|      }, %) | ||||
|   |> line(end = profileStart(%)) | ||||
|   |> close() | ||||
|   |> extrude(length = 20) | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -78,7 +78,7 @@ assertEqual(sum, 6, 0.00001, "1 + 2 + 3 summed is 6") | ||||
| ```js | ||||
| // Declare a function that sketches a decagon. | ||||
| fn decagon(radius) { | ||||
|   // Each side of the decagon is turned this many radians from the previous angle. | ||||
|   // Each side of the decagon is turned this many degrees from the previous angle. | ||||
|   stepAngle = 1 / 10 * TAU | ||||
|  | ||||
|   // Start the decagon sketch at this point. | ||||
|  | ||||
| @ -151,9 +151,15 @@ cube | ||||
|  | ||||
| sketch001 = startSketchOn(XY) | ||||
| rectangleSketch = startProfileAt([-200, 23.86], sketch001) | ||||
|   |> angledLine(angle = 0, length = 73.47, tag = $rectangleSegmentA001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 50.61) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001)) | ||||
|   |> angledLine([0, 73.47], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        50.61 | ||||
|      ], %) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -85,9 +85,15 @@ cube | ||||
|  | ||||
| sketch001 = startSketchOn(XY) | ||||
| rectangleSketch = startProfileAt([-200, 23.86], sketch001) | ||||
|   |> angledLine(angle = 0, length = 73.47, tag = $rectangleSegmentA001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 50.61) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001)) | ||||
|   |> angledLine([0, 73.47], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        50.61 | ||||
|      ], %) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -32,9 +32,9 @@ exampleSketch = startSketchOn(XZ) | ||||
|   |> line(end = [10, 0]) | ||||
|   |> line(end = [5, 10], tag = $seg01) | ||||
|   |> line(end = [-10, 0]) | ||||
|   |> angledLine(angle = segAng(seg01), length = 10) | ||||
|   |> angledLine([segAng(seg01), 10], %) | ||||
|   |> line(end = [-10, 0]) | ||||
|   |> angledLine(angle = segAng(seg01), length = -15) | ||||
|   |> angledLine([segAng(seg01), -15], %) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(exampleSketch, length = 4) | ||||
|  | ||||
| @ -29,9 +29,9 @@ segLen(tag: TagIdentifier): number | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 60, length = 10, tag = $thing) | ||||
|   |> angledLine({ angle = 60, length = 10 }, %, $thing) | ||||
|   |> tangentialArc({ offset = -120, radius = 5 }, %) | ||||
|   |> angledLine(angle = -60, length = segLen(thing)) | ||||
|   |> angledLine({ angle = -60, length = segLen(thing) }, %) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(exampleSketch, length = 5) | ||||
|  | ||||
| @ -6,7 +6,7 @@ layout: manual | ||||
|  | ||||
| # KCL Settings | ||||
|  | ||||
| There are three levels of settings available in Zoo Design Studio: | ||||
| There are three levels of settings available in the KittyCAD Design Studiolication: | ||||
|  | ||||
| 1. [User Settings](/docs/kcl/settings/user): Global settings that apply to all projects, stored in `user.toml` | ||||
| 2. [Project Settings](/docs/kcl/settings/project): Settings specific to a project, stored in `project.toml` | ||||
| @ -14,7 +14,7 @@ There are three levels of settings available in Zoo Design Studio: | ||||
|  | ||||
| ## Configuration Files | ||||
|  | ||||
| Zoo Design Studio uses TOML files for configuration: | ||||
| The KittyCAD Design Studio uses TOML files for configuration: | ||||
|  | ||||
| * **User Settings**: `user.toml` - See [complete documentation](/docs/kcl/settings/user) | ||||
| * **Project Settings**: `project.toml` - See [complete documentation](/docs/kcl/settings/project) | ||||
|  | ||||
| @ -96,7 +96,7 @@ Permanently dismiss the banner warning to download the desktop app. This setting | ||||
|  | ||||
| ##### stream_idle_mode | ||||
|  | ||||
| When the user is idle, teardown the stream after some time. | ||||
| When the user is idle, and this is true, the stream will be torn down. | ||||
|  | ||||
|  | ||||
| **Default:** None | ||||
|  | ||||
| @ -33,7 +33,7 @@ sqrt(num: number): number | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 50, length = sqrt(2500)) | ||||
|   |> angledLine({ angle = 50, length = sqrt(2500) }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -9,15 +9,7 @@ Create a helix. | ||||
|  | ||||
|  | ||||
| ```js | ||||
| helix( | ||||
|   revolutions: number(_), | ||||
|   angleStart: number(deg), | ||||
|   ccw?: bool, | ||||
|   radius?: number(mm), | ||||
|   axis?: Axis3d | Edge, | ||||
|   length?: number(mm), | ||||
|   cylinder?: Solid, | ||||
| ): Helix | ||||
| helix(revolutions: number(_), angleStart: number(deg), ccw?: bool, radius?: number(mm), axis?: Axis3d | Edge, length?: number(mm), cylinder?: Solid): Helix | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -29,10 +29,10 @@ cos(@num: number(rad)): number(_) | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine( | ||||
|   |> angledLine({ | ||||
|     angle = 30, | ||||
|     length = 3 / cos(toRadians(30)), | ||||
|   ) | ||||
|   }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -29,10 +29,10 @@ tan(@num: number(rad)): number(_) | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine( | ||||
|   |> angledLine({ | ||||
|     angle = 50, | ||||
|     length = 50 * tan(1/2), | ||||
|   ) | ||||
|   }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -6,16 +6,10 @@ layout: manual | ||||
|  | ||||
|  | ||||
|  | ||||
| Construct a 2-dimensional circle, of the specified radius, centered at | ||||
| the provided (x, y) origin point. | ||||
| Construct a 2-dimensional circle, of the specified radius, centered atthe provided (x, y) origin point. | ||||
|  | ||||
| ```js | ||||
| circle( | ||||
|   @sketch_or_surface: Sketch | Plane | Face, | ||||
|   center: Point2d, | ||||
|   radius: number, | ||||
|   tag?: tag, | ||||
| ): Sketch | ||||
| circle(@sketch_or_surface: Sketch | Plane | Face, center: Point2d, radius: number, tag?: tag): Sketch | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -11,10 +11,7 @@ Only works on unclosed sketches for now. | ||||
| Mirror occurs around a local sketch axis rather than a global axis. | ||||
|  | ||||
| ```js | ||||
| mirror2d( | ||||
|   @sketches: [Sketch; 1+], | ||||
|   axis: Axis2d | Edge, | ||||
| ): Sketch | ||||
| mirror2d(@sketches: [Sketch; 1+], axis: Axis2d | Edge): Sketch | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
							
								
								
									
										32864
									
								
								docs/kcl/std.json
									
									
									
									
									
								
							
							
						
						
									
										32864
									
								
								docs/kcl/std.json
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -91,9 +91,15 @@ springSketch = startSketchOn(YZ) | ||||
|  | ||||
| sketch001 = startSketchOn(XY) | ||||
| rectangleSketch = startProfileAt([-200, 23.86], sketch001) | ||||
|   |> angledLine(angle = 0, length = 73.47, tag = $rectangleSegmentA001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 50.61) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001)) | ||||
|   |> angledLine([0, 73.47], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        50.61 | ||||
|      ], %) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -32,7 +32,10 @@ pillSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line(end = [20, 0]) | ||||
|   |> tangentialArcToRelative([0, 10], %, $arc1) | ||||
|   |> angledLine(angle = tangentToEnd(arc1), length = 20) | ||||
|   |> angledLine({ | ||||
|        angle = tangentToEnd(arc1), | ||||
|        length = 20 | ||||
|      }, %) | ||||
|   |> tangentialArcToRelative([0, -10], %) | ||||
|   |> close() | ||||
|  | ||||
| @ -47,7 +50,10 @@ pillSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line(end = [0, 20]) | ||||
|   |> tangentialArcTo([10, 20], %, $arc1) | ||||
|   |> angledLine(angle = tangentToEnd(arc1), length = 20) | ||||
|   |> angledLine({ | ||||
|        angle = tangentToEnd(arc1), | ||||
|        length = 20 | ||||
|      }, %) | ||||
|   |> tangentialArcToRelative([-10, 0], %) | ||||
|   |> close() | ||||
|  | ||||
| @ -60,7 +66,10 @@ pillExtrude = extrude(pillSketch, length = 10) | ||||
| rectangleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line(end = [10, 0], tag = $seg1) | ||||
|   |> angledLine(angle = tangentToEnd(seg1), length = 10) | ||||
|   |> angledLine({ | ||||
|        angle = tangentToEnd(seg1), | ||||
|        length = 10 | ||||
|      }, %) | ||||
|   |> line(end = [0, 10]) | ||||
|   |> line(end = [-20, 0]) | ||||
|   |> close() | ||||
| @ -74,7 +83,7 @@ rectangleExtrude = extrude(rectangleSketch, length = 10) | ||||
| bottom = startSketchOn(XY) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> arcTo({ end = [10, 10], interior = [5, 1] }, %, $arc1) | ||||
|   |> angledLine(angle = tangentToEnd(arc1), length = 20) | ||||
|   |> angledLine([tangentToEnd(arc1), 20], %) | ||||
|   |> close() | ||||
| ``` | ||||
|  | ||||
| @ -86,7 +95,7 @@ circSketch = startSketchOn(XY) | ||||
|  | ||||
| triangleSketch = startSketchOn(XY) | ||||
|   |> startProfileAt([-5, 0], %) | ||||
|   |> angledLine(angle = tangentToEnd(circ), length = 10) | ||||
|   |> angledLine([tangentToEnd(circ), 10], %) | ||||
|   |> line(end = [-15, 0]) | ||||
|   |> close() | ||||
| ``` | ||||
|  | ||||
| @ -35,9 +35,9 @@ tangentialArc( | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 60, length = 10) | ||||
|   |> angledLine({ angle = 60, length = 10 }, %) | ||||
|   |> tangentialArc({ radius = 10, offset = -120 }, %) | ||||
|   |> angledLine(angle = -60, length = 10) | ||||
|   |> angledLine({ angle = -60, length = 10 }, %) | ||||
|   |> close() | ||||
|  | ||||
| example = extrude(exampleSketch, length = 10) | ||||
|  | ||||
| @ -35,7 +35,7 @@ tangentialArcTo( | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 60, length = 10) | ||||
|   |> angledLine({ angle = 60, length = 10 }, %) | ||||
|   |> tangentialArcTo([15, 15], %) | ||||
|   |> line(end = [10, -15]) | ||||
|   |> close() | ||||
|  | ||||
| @ -35,7 +35,7 @@ tangentialArcToRelative( | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 45, length = 10) | ||||
|   |> angledLine({ angle = 45, length = 10 }, %) | ||||
|   |> tangentialArcToRelative([0, -10], %) | ||||
|   |> line(end = [-10, 0]) | ||||
|   |> close() | ||||
|  | ||||
| @ -30,7 +30,7 @@ tau(): number | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 50, length = 10 * tau()) | ||||
|   |> angledLine({ angle = 50, length = 10 * tau() }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -33,7 +33,10 @@ toDegrees(num: number): number | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 50, length = 70 * cos(toDegrees(pi() / 4))) | ||||
|   |> angledLine({ | ||||
|        angle = 50, | ||||
|        length = 70 * cos(toDegrees(pi() / 4)) | ||||
|      }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -33,7 +33,10 @@ toRadians(num: number): number | ||||
| ```js | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 50, length = 70 * cos(toRadians(45))) | ||||
|   |> angledLine({ | ||||
|        angle = 50, | ||||
|        length = 70 * cos(toRadians(45)) | ||||
|      }, %) | ||||
|   |> yLine(endAbsolute = 0) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -86,9 +86,15 @@ cube | ||||
|  | ||||
| sketch001 = startSketchOn(XY) | ||||
| rectangleSketch = startProfileAt([-200, 23.86], sketch001) | ||||
|   |> angledLine(angle = 0, length = 73.47, tag = $rectangleSegmentA001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 50.61) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001)) | ||||
|   |> angledLine([0, 73.47], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        50.61 | ||||
|      ], %) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -184,17 +184,15 @@ way: | ||||
| ```norun | ||||
| startSketchOn('XZ') | ||||
|   |> startProfileAt(origin, %) | ||||
|   |> angledLine(angle = 0, length = 191.26, tag = $rectangleSegmentA001) | ||||
|   |> angledLine( | ||||
|   |> angledLine({angle = 0, length = 191.26}, %, $rectangleSegmentA001) | ||||
|   |> angledLine({ | ||||
|        angle = segAng(rectangleSegmentA001) - 90, | ||||
|        length = 196.99, | ||||
|        tag = $rectangleSegmentB001, | ||||
|      ) | ||||
|   |> angledLine( | ||||
|      }, %, $rectangleSegmentB001) | ||||
|   |> angledLine({ | ||||
|        angle = segAng(rectangleSegmentA001), | ||||
|        length = -segLen(rectangleSegmentA001), | ||||
|        tag = $rectangleSegmentC001, | ||||
|      ) | ||||
|      }, %, $rectangleSegmentC001) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| ``` | ||||
| @ -219,17 +217,15 @@ However if the code was written like this: | ||||
| fn rect(origin) { | ||||
|   return startSketchOn('XZ') | ||||
|     |> startProfileAt(origin, %) | ||||
|     |> angledLine(angle = 0, length = 191.26, tag = $rectangleSegmentA001) | ||||
|     |> angledLine( | ||||
|     |> angledLine({angle = 0, length = 191.26}, %, $rectangleSegmentA001) | ||||
|     |> angledLine({ | ||||
|          angle = segAng(rectangleSegmentA001) - 90, | ||||
|          length = 196.99, | ||||
|          tag = $rectangleSegmentB001, | ||||
|        ) | ||||
|     |> angledLine( | ||||
|          length = 196.99 | ||||
|        }, %, $rectangleSegmentB001) | ||||
|     |> angledLine({ | ||||
|          angle = segAng(rectangleSegmentA001), | ||||
|          length = -segLen(rectangleSegmentA001) | ||||
|          tag = $rectangleSegmentC001, | ||||
|        ) | ||||
|        }, %, $rectangleSegmentC001) | ||||
|     |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|     |> close() | ||||
| } | ||||
| @ -249,17 +245,15 @@ For example the following code works. | ||||
| fn rect(origin) { | ||||
|   return startSketchOn('XZ') | ||||
|     |> startProfileAt(origin, %) | ||||
|     |> angledLine(angle = 0, length = 191.26, tag = $rectangleSegmentA001) | ||||
|     |> angledLine( | ||||
|     |> angledLine({angle = 0, length = 191.26}, %, $rectangleSegmentA001) | ||||
|     |> angledLine({ | ||||
|          angle = segAng(rectangleSegmentA001) - 90, | ||||
|          length = 196.99, | ||||
|          tag = $rectangleSegmentB001, | ||||
|        ) | ||||
|     |> angledLine( | ||||
|          length = 196.99 | ||||
|        }, %, $rectangleSegmentB001) | ||||
|     |> angledLine({ | ||||
|          angle = segAng(rectangleSegmentA001), | ||||
|          length = -segLen(rectangleSegmentA001) | ||||
|          tag = $rectangleSegmentC001, | ||||
|        ) | ||||
|        }, %, $rectangleSegmentC001) | ||||
|     |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|     |> close() | ||||
| } | ||||
|  | ||||
							
								
								
									
										22
									
								
								docs/kcl/types/PolarCoordsData.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								docs/kcl/types/PolarCoordsData.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,22 @@ | ||||
| --- | ||||
| title: "PolarCoordsData" | ||||
| excerpt: "Data for polar coordinates." | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
| Data for polar coordinates. | ||||
|  | ||||
| **Type:** `object` | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Properties | ||||
|  | ||||
| | Property | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `angle` |[`number`](/docs/kcl/types/number)| The angle of the line (in degrees). | No | | ||||
| | `length` |[`number`](/docs/kcl/types/number)| The length of the line. | No | | ||||
|  | ||||
|  | ||||
| @ -14,17 +14,15 @@ way: | ||||
| ```js | ||||
| startSketchOn('XZ') | ||||
|   |> startProfileAt(origin, %) | ||||
|   |> angledLine(angle = 0, length = 191.26, tag = $rectangleSegmentA001) | ||||
|   |> angledLine( | ||||
|   |> angledLine({angle = 0, length = 191.26}, %, $rectangleSegmentA001) | ||||
|   |> angledLine({ | ||||
|        angle = segAng(rectangleSegmentA001) - 90, | ||||
|        length = 196.99, | ||||
|        tag = $rectangleSegmentB001, | ||||
|      ) | ||||
|   |> angledLine( | ||||
|      }, %, $rectangleSegmentB001) | ||||
|   |> angledLine({ | ||||
|        angle = segAng(rectangleSegmentA001), | ||||
|        length = -segLen(rectangleSegmentA001), | ||||
|        tag = $rectangleSegmentC001, | ||||
|      ) | ||||
|      }, %, $rectangleSegmentC001) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| ``` | ||||
| @ -48,16 +46,15 @@ However if the code was written like this: | ||||
| fn rect(origin) { | ||||
|   return startSketchOn('XZ') | ||||
|     |> startProfileAt(origin, %) | ||||
|     |> angledLine(angle = 0, length = 191.26, tag = $rectangleSegmentA001) | ||||
|     |> angledLine( | ||||
|     |> angledLine({angle = 0, length = 191.26}, %, $rectangleSegmentA001) | ||||
|     |> angledLine({ | ||||
|          angle = segAng(rectangleSegmentA001) - 90, | ||||
|          length = 196.99, | ||||
|          tag = $rectangleSegmentB001) | ||||
|     |> angledLine( | ||||
|          length = 196.99 | ||||
|        }, %, $rectangleSegmentB001) | ||||
|     |> angledLine({ | ||||
|          angle = segAng(rectangleSegmentA001), | ||||
|          length = -segLen(rectangleSegmentA001), | ||||
|          tag = $rectangleSegmentC001 | ||||
|        ) | ||||
|          length = -segLen(rectangleSegmentA001) | ||||
|        }, %, $rectangleSegmentC001) | ||||
|     |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|     |> close() | ||||
| } | ||||
| @ -77,15 +74,15 @@ For example the following code works. | ||||
| fn rect(origin) { | ||||
|   return startSketchOn('XZ') | ||||
|     |> startProfileAt(origin, %) | ||||
|     |> angledLine(angle = 0, length = 191.26, tag = $rectangleSegmentA001) | ||||
|     |> angledLine( | ||||
|     |> angledLine({angle = 0, length = 191.26}, %, $rectangleSegmentA001) | ||||
|     |> angledLine({ | ||||
|          angle = segAng(rectangleSegmentA001) - 90, | ||||
|          length = 196.99 | ||||
|        , %, $rectangleSegmentB001) | ||||
|     |> angledLine( | ||||
|        }, %, $rectangleSegmentB001) | ||||
|     |> angledLine({ | ||||
|          angle = segAng(rectangleSegmentA001), | ||||
|          length = -segLen(rectangleSegmentA001) | ||||
|        , %, $rectangleSegmentC001) | ||||
|        }, %, $rectangleSegmentC001) | ||||
|     |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|     |> close() | ||||
| } | ||||
|  | ||||
| @ -38,10 +38,10 @@ xLine( | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> xLine(length = 15) | ||||
|   |> angledLine(angle = 80, length = 15) | ||||
|   |> angledLine({ angle = 80, length = 15 }, %) | ||||
|   |> line(end = [8, -10]) | ||||
|   |> xLine(length = 10) | ||||
|   |> angledLine(angle = 120, length = 30) | ||||
|   |> angledLine({ angle = 120, length = 30 }, %) | ||||
|   |> xLine(length = -15) | ||||
|   |> close() | ||||
|  | ||||
|  | ||||
| @ -38,7 +38,7 @@ yLine( | ||||
| exampleSketch = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> yLine(length = 15) | ||||
|   |> angledLine(angle = 30, length = 15) | ||||
|   |> angledLine({ angle = 30, length = 15 }, %) | ||||
|   |> line(end = [8, -10]) | ||||
|   |> yLine(length = -5) | ||||
|   |> close() | ||||
|  | ||||
| @ -137,7 +137,7 @@ async function doBasicSketch( | ||||
|     await page.waitForTimeout(100) | ||||
|   } | ||||
|  | ||||
|   await page.getByRole('button', { name: 'constraints: open menu' }).click() | ||||
|   await page.getByRole('button', { name: 'Length: open menu' }).click() | ||||
|   await page.getByRole('button', { name: 'Equal Length' }).click() | ||||
|  | ||||
|   // Open the code pane. | ||||
|  | ||||
| @ -38,7 +38,7 @@ test.describe('Point and click for boolean workflows', () => { | ||||
|         path.resolve( | ||||
|           __dirname, | ||||
|           '../../', | ||||
|           './rust/kcl-lib/e2e/executor/inputs/boolean-setup-with-sketch-on-faces.kcl' | ||||
|           './rust/kcl-lib/e2e/executor/inputs/boolean-setup-with' | ||||
|         ), | ||||
|         'utf-8' | ||||
|       ) | ||||
| @ -46,6 +46,8 @@ test.describe('Point and click for boolean workflows', () => { | ||||
|         localStorage.setItem('persistCode', file) | ||||
|       }, file) | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.waitForExecutionDone() | ||||
|  | ||||
|       await scene.settled(cmdBar) | ||||
|  | ||||
|       // Test coordinates for selection - these might need adjustment based on actual scene layout | ||||
| @ -75,7 +77,6 @@ test.describe('Point and click for boolean workflows', () => { | ||||
|  | ||||
|         // Select first object in the scene, expect there to be a pixel diff from the selection color change | ||||
|         await clickFirstObject({ pixelDiff: 50 }) | ||||
|         await page.waitForTimeout(1000) | ||||
|  | ||||
|         // For subtract, we need to proceed to the next step before selecting the second object | ||||
|         if (operationName !== 'subtract') { | ||||
| @ -86,8 +87,6 @@ test.describe('Point and click for boolean workflows', () => { | ||||
|         // Select second object | ||||
|         await clickSecondObject({ pixelDiff: 50 }) | ||||
|  | ||||
|         await page.waitForTimeout(1000) | ||||
|  | ||||
|         // Confirm the operation in the command bar | ||||
|         await cmdBar.progressCmdBar() | ||||
|  | ||||
|  | ||||
| @ -4,7 +4,6 @@ import { uuidv4 } from '@src/lib/utils' | ||||
|  | ||||
| import type { HomePageFixture } from '@e2e/playwright/fixtures/homePageFixture' | ||||
| import type { SceneFixture } from '@e2e/playwright/fixtures/sceneFixture' | ||||
| import type { ToolbarFixture } from '@e2e/playwright/fixtures/toolbarFixture' | ||||
| import { getUtils } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| @ -16,7 +15,6 @@ test.describe( | ||||
|       page: Page, | ||||
|       homePage: HomePageFixture, | ||||
|       scene: SceneFixture, | ||||
|       toolbar: ToolbarFixture, | ||||
|       plane: string, | ||||
|       clickCoords: { x: number; y: number } | ||||
|     ) => { | ||||
| @ -61,12 +59,9 @@ test.describe( | ||||
|       await u.sendCustomCmd(updateCamCommand) | ||||
|  | ||||
|       await u.closeDebugPanel() | ||||
|  | ||||
|       await page.mouse.click(clickCoords.x, clickCoords.y) | ||||
|       await page.waitForTimeout(600) // wait for animation | ||||
|  | ||||
|       await toolbar.waitUntilSketchingReady() | ||||
|  | ||||
|       await expect( | ||||
|         page.getByRole('button', { name: 'line Line', exact: true }) | ||||
|       ).toBeVisible() | ||||
| @ -122,12 +117,11 @@ test.describe( | ||||
|     ] | ||||
|  | ||||
|     for (const config of planeConfigs) { | ||||
|       test(config.plane, async ({ page, homePage, scene, toolbar }) => { | ||||
|       test(config.plane, async ({ page, homePage, scene }) => { | ||||
|         await sketchOnPlaneAndBackSideTest( | ||||
|           page, | ||||
|           homePage, | ||||
|           scene, | ||||
|           toolbar, | ||||
|           config.plane, | ||||
|           config.coords | ||||
|         ) | ||||
|  | ||||
| @ -15,7 +15,6 @@ test.describe('Code pane and errors', { tag: ['@skipWin'] }, () => { | ||||
|     page, | ||||
|     homePage, | ||||
|     scene, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
| @ -37,7 +36,7 @@ extrude001 = extrude(sketch001, length = 5)` | ||||
|  | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // Ensure no badge is present | ||||
|     const codePaneButtonHolder = page.locator('#code-button-holder') | ||||
| @ -172,8 +171,6 @@ extrude001 = extrude(sketch001, length = 5)` | ||||
|     context, | ||||
|     page, | ||||
|     homePage, | ||||
|     scene, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     // Load the app with the working starter code | ||||
|     await context.addInitScript((code) => { | ||||
| @ -183,7 +180,9 @@ extrude001 = extrude(sketch001, length = 5)` | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await scene.settled(cmdBar) | ||||
|     // FIXME: await scene.waitForExecutionDone() does not work. It still fails. | ||||
|     // I needed to increase this timeout to get this to pass. | ||||
|     await page.waitForTimeout(10000) | ||||
|  | ||||
|     // Ensure badge is present | ||||
|     const codePaneButtonHolder = page.locator('#code-button-holder') | ||||
|  | ||||
| @ -317,13 +317,9 @@ test.describe('Command bar tests', { tag: ['@skipWin'] }, () => { | ||||
|   test('Can switch between sketch tools via command bar', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|     scene, | ||||
|     cmdBar, | ||||
|     toolbar, | ||||
|   }) => { | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|  | ||||
|     const sketchButton = page.getByRole('button', { name: 'Start Sketch' }) | ||||
|     const cmdBarButton = page.getByRole('button', { name: 'Commands' }) | ||||
| @ -347,9 +343,7 @@ test.describe('Command bar tests', { tag: ['@skipWin'] }, () => { | ||||
|  | ||||
|     // Start a sketch | ||||
|     await sketchButton.click() | ||||
|  | ||||
|     await page.mouse.click(700, 200) | ||||
|     await toolbar.waitUntilSketchingReady() | ||||
|  | ||||
|     // Switch between sketch tools via the command bar | ||||
|     await expect(lineToolButton).toHaveAttribute('aria-pressed', 'true') | ||||
|  | ||||
| @ -11,7 +11,7 @@ import { expect, test } from '@e2e/playwright/zoo-test' | ||||
| test( | ||||
|   'export works on the first try', | ||||
|   { tag: ['@electron', '@skipLocalEngine'] }, | ||||
|   async ({ page, context, scene, tronApp, cmdBar }, testInfo) => { | ||||
|   async ({ page, context, scene, tronApp }, testInfo) => { | ||||
|     if (!tronApp) { | ||||
|       fail() | ||||
|     } | ||||
| @ -37,7 +37,8 @@ test( | ||||
|       const projectName = page.getByText(`bracket`) | ||||
|       await expect(projectName).toBeVisible() | ||||
|       await projectName.click() | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|       await page.waitForTimeout(1_000) // wait for panel buttons to be available | ||||
|  | ||||
|       // Expect zero errors in gutter | ||||
|       await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
| @ -46,9 +47,10 @@ test( | ||||
|       const exportButton = page.getByTestId('export-pane-button') | ||||
|       await expect(exportButton).toBeVisible() | ||||
|       await exportButton.click() | ||||
|       await page.waitForTimeout(1_000) // wait for export options to be available | ||||
|  | ||||
|       // Select the first format option | ||||
|       const gltfOption = cmdBar.selectOption({ name: 'glTF' }) | ||||
|       const gltfOption = page.getByText('glTF') | ||||
|       const exportFileName = `main.gltf` // source file is named `main.kcl` | ||||
|       await expect(gltfOption).toBeVisible() | ||||
|       await page.keyboard.press('Enter') | ||||
| @ -56,6 +58,7 @@ test( | ||||
|       // Click the checkbox | ||||
|       const submitButton = page.getByText('Confirm Export') | ||||
|       await expect(submitButton).toBeVisible() | ||||
|       await page.waitForTimeout(500) | ||||
|       await page.keyboard.press('Enter') | ||||
|  | ||||
|       // Look out for the toast message | ||||
| @ -107,7 +110,8 @@ test( | ||||
|  | ||||
|       // Close the file pane | ||||
|       await u.closeFilePanel() | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|       await page.waitForTimeout(1_000) // wait for panel buttons to be available | ||||
|  | ||||
|       // Expect zero errors in gutter | ||||
|       await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
| @ -116,9 +120,10 @@ test( | ||||
|       const exportButton = page.getByTestId('export-pane-button') | ||||
|       await expect(exportButton).toBeVisible() | ||||
|       await exportButton.click() | ||||
|       await page.waitForTimeout(1_000) // wait for export options to be available | ||||
|  | ||||
|       // Select the first format option | ||||
|       const gltfOption = cmdBar.selectOption({ name: 'glTF' }) | ||||
|       const gltfOption = page.getByText('glTF') | ||||
|       const exportFileName = `other.gltf` // source file is named `other.kcl` | ||||
|       await expect(gltfOption).toBeVisible() | ||||
|       await page.keyboard.press('Enter') | ||||
| @ -126,6 +131,7 @@ test( | ||||
|       // Click the checkbox | ||||
|       const submitButton = page.getByText('Confirm Export') | ||||
|       await expect(submitButton).toBeVisible() | ||||
|       await page.waitForTimeout(500) | ||||
|       await page.keyboard.press('Enter') | ||||
|  | ||||
|       // Look out for the toast message | ||||
|  | ||||
| @ -78,14 +78,12 @@ sketch001 = startSketchOn(XY) | ||||
|  | ||||
|     // Ensure we execute the first time. | ||||
|     await u.openDebugPanel() | ||||
|     await expect | ||||
|       .poll(() => | ||||
|         page.locator('[data-receive-command-type="scene_clear_all"]').count() | ||||
|       ) | ||||
|       .toBe(1) | ||||
|     await expect | ||||
|       .poll(() => page.locator('[data-message-type="execution-done"]').count()) | ||||
|       .toBe(2) | ||||
|     await expect( | ||||
|       page.locator('[data-receive-command-type="scene_clear_all"]') | ||||
|     ).toHaveCount(1) | ||||
|     await expect( | ||||
|       page.locator('[data-message-type="execution-done"]') | ||||
|     ).toHaveCount(2) | ||||
|  | ||||
|     // Add whitespace to the end of the code. | ||||
|     await u.codeLocator.click() | ||||
| @ -112,14 +110,12 @@ sketch001 = startSketchOn(XY) | ||||
|   test('ensure we use the cache, and do not clear on append', async ({ | ||||
|     homePage, | ||||
|     page, | ||||
|     scene, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await page.keyboard.type(`sketch001 = startSketchOn(XY) | ||||
| @ -503,7 +499,7 @@ sketch_001 = startSketchOn(XY) | ||||
|     await page.keyboard.press('ArrowLeft') | ||||
|     await page.keyboard.press('ArrowRight') | ||||
|  | ||||
|     await scene.connectionEstablished() | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // error in guter | ||||
|     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible() | ||||
| @ -1353,51 +1349,4 @@ sketch001 = startSketchOn(XZ) | ||||
|       15 | ||||
|     ) | ||||
|   }) | ||||
|  | ||||
|   test(`test-toolbar-buttons`, async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|     toolbar, | ||||
|     scene, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     await test.step('Load an empty file', async () => { | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem('persistCode', '') | ||||
|       }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|       // wait until scene is ready to be interacted with | ||||
|       await scene.connectionEstablished() | ||||
|       await scene.settled(cmdBar) | ||||
|     }) | ||||
|  | ||||
|     await test.step('Test toolbar button correct selection', async () => { | ||||
|       await toolbar.expectToolbarMode.toBe('modeling') | ||||
|  | ||||
|       await toolbar.startSketchPlaneSelection() | ||||
|  | ||||
|       // Click on a default plane | ||||
|       await page.mouse.click(700, 200) | ||||
|  | ||||
|       // tools cannot be selected immediately, couldn't find an event to await instead. | ||||
|       await page.waitForTimeout(1000) | ||||
|  | ||||
|       await toolbar.selectCenterRectangle() | ||||
|  | ||||
|       await expect(page.getByTestId('center-rectangle')).toHaveAttribute( | ||||
|         'aria-pressed', | ||||
|         'true' | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await test.step('Test Toolbar dropdown remembering last selection', async () => { | ||||
|       // Select another tool | ||||
|       await page.getByTestId('circle-center').click() | ||||
|  | ||||
|       // center-rectangle should still be the active option in the rectangle dropdown | ||||
|       await expect(page.getByTestId('center-rectangle')).toBeVisible() | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -19,7 +19,7 @@ length001 = timesFive(1) * 5 | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([20, 10], %) | ||||
|   |> line(end = [10, 10]) | ||||
|   |> angledLine(angle = -45, length = length001) | ||||
|   |> angledLine([-45, length001], %) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| revolve001 = revolve(sketch001, axis = X) | ||||
| @ -38,9 +38,15 @@ extrude001 = extrude(sketch002, length = 10) | ||||
|  | ||||
| const FEATURE_TREE_SKETCH_CODE = `sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> angledLine(angle = 0, length = 4, tag = $rectangleSegmentA001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 2, tag = $rectangleSegmentB001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001), tag = $rectangleSegmentC001) | ||||
|   |> angledLine([0, 4], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        2 | ||||
|      ], %, $rectangleSegmentB001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $rectangleSegmentC001) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close(%) | ||||
| extrude001 = extrude(sketch001, length = 10) | ||||
| @ -58,7 +64,7 @@ test.describe('Feature Tree pane', () => { | ||||
|   test( | ||||
|     'User can go to definition and go to function definition', | ||||
|     { tag: '@electron' }, | ||||
|     async ({ context, homePage, scene, editor, toolbar, cmdBar, page }) => { | ||||
|     async ({ context, homePage, scene, editor, toolbar }) => { | ||||
|       await context.folderSetupFn(async (dir) => { | ||||
|         const bracketDir = join(dir, 'test-sample') | ||||
|         await fsp.mkdir(bracketDir, { recursive: true }) | ||||
| @ -80,13 +86,9 @@ test.describe('Feature Tree pane', () => { | ||||
|           sortBy: 'last-modified-desc', | ||||
|         }) | ||||
|         await homePage.openProject('test-sample') | ||||
|         await scene.connectionEstablished() | ||||
|         await scene.settled(cmdBar) | ||||
|  | ||||
|         await scene.waitForExecutionDone() | ||||
|         await editor.closePane() | ||||
|         await toolbar.openFeatureTreePane() | ||||
|         await expect | ||||
|           .poll(() => page.getByText('Feature tree').count()) | ||||
|           .toBeGreaterThan(1) | ||||
|       }) | ||||
|  | ||||
|       async function testViewSource({ | ||||
| @ -252,7 +254,7 @@ test.describe('Feature Tree pane', () => { | ||||
|         sortBy: 'last-modified-desc', | ||||
|       }) | ||||
|       await homePage.openProject('test-sample') | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|       await toolbar.openFeatureTreePane() | ||||
|     }) | ||||
|  | ||||
| @ -337,7 +339,7 @@ test.describe('Feature Tree pane', () => { | ||||
|         sortBy: 'last-modified-desc', | ||||
|       }) | ||||
|       await homePage.openProject('test-sample') | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|       await toolbar.openFeatureTreePane() | ||||
|     }) | ||||
|  | ||||
| @ -412,7 +414,8 @@ profile003 = startProfileAt([0, -4.93], sketch001) | ||||
|     const planeColor: [number, number, number] = [74, 74, 74] | ||||
|  | ||||
|     await homePage.openProject('test-sample') | ||||
|     await scene.settled(cmdBar) | ||||
|     // FIXME: @lf94 has a better way to verify execution completion, in a PR rn | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     await test.step(`Verify we see the sketch`, async () => { | ||||
|       await scene.expectPixelColor(sketchColor, testPoint, 10) | ||||
|  | ||||
| @ -47,7 +47,6 @@ test.describe('integrations tests', () => { | ||||
|         await scene.connectionEstablished() | ||||
|         await scene.settled(cmdBar) | ||||
|         await clickObj() | ||||
|         await page.waitForTimeout(1000) | ||||
|         await scene.moveNoWhere() | ||||
|         await editor.expectState({ | ||||
|           activeLines: [ | ||||
| @ -73,11 +72,11 @@ test.describe('integrations tests', () => { | ||||
|       }) | ||||
|       await test.step('setup for next assertion', async () => { | ||||
|         await toolbar.openFile('main.kcl') | ||||
|         await page.waitForTimeout(1000) | ||||
|  | ||||
|         await scene.settled(cmdBar) | ||||
|  | ||||
|         await clickObj() | ||||
|         await page.waitForTimeout(1000) | ||||
|         await scene.moveNoWhere() | ||||
|         await page.waitForTimeout(1000) | ||||
|         await editor.expectState({ | ||||
|           activeLines: [ | ||||
|             '|>startProfileAt([75.8,317.2],%)//[$startCapTag,$EndCapTag]', | ||||
| @ -90,7 +89,7 @@ test.describe('integrations tests', () => { | ||||
|         await toolbar.expectFileTreeState(['main.kcl', fileName]) | ||||
|       }) | ||||
|       await test.step('check sketch mode is exited when opening a different file', async () => { | ||||
|         await toolbar.openFile(fileName) | ||||
|         await toolbar.openFile(fileName, { wait: false }) | ||||
|  | ||||
|         // check we're out of sketch mode | ||||
|         await expect(toolbar.exitSketchBtn).not.toBeVisible() | ||||
|  | ||||
| @ -112,16 +112,22 @@ export class CmdBarFixture { | ||||
|    * and assumes we are past the `pickCommand` step. | ||||
|    */ | ||||
|   progressCmdBar = async (shouldFuzzProgressMethod = true) => { | ||||
|     await this.page.waitForTimeout(2000) | ||||
|     const arrowButton = this.page.getByRole('button', { | ||||
|       name: 'arrow right Continue', | ||||
|     }) | ||||
|     if (await arrowButton.isVisible()) { | ||||
|       await arrowButton.click() | ||||
|     // FIXME: Progressing the command bar is a race condition. We have an async useEffect that reports the final state via useCalculateKclExpression. If this does not run quickly enough, it will not "fail" the continue because you can press continue if the state is not ready. E2E tests do not know this. | ||||
|     // Wait 1250ms to assume the await executeAst of the KCL input field is finished | ||||
|     await this.page.waitForTimeout(1250) | ||||
|     if (shouldFuzzProgressMethod || Math.random() > 0.5) { | ||||
|       const arrowButton = this.page.getByRole('button', { | ||||
|         name: 'arrow right Continue', | ||||
|       }) | ||||
|       if (await arrowButton.isVisible()) { | ||||
|         await arrowButton.click() | ||||
|       } else { | ||||
|         await this.page | ||||
|           .getByRole('button', { name: 'checkmark Submit command' }) | ||||
|           .click() | ||||
|       } | ||||
|     } else { | ||||
|       await this.page | ||||
|         .getByRole('button', { name: 'checkmark Submit command' }) | ||||
|         .click() | ||||
|       await this.page.keyboard.press('Enter') | ||||
|     } | ||||
|   } | ||||
|  | ||||
|  | ||||
| @ -39,8 +39,7 @@ export class AuthenticatedApp { | ||||
|   } | ||||
|  | ||||
|   async initialise(code = '') { | ||||
|     const testDir = this.testInfo.outputPath('electron-test-projects-dir') | ||||
|     await setup(this.context, this.page, testDir, this.testInfo) | ||||
|     await setup(this.context, this.page, this.testInfo) | ||||
|     const u = await getUtils(this.page) | ||||
|  | ||||
|     await this.page.addInitScript(async (code) => { | ||||
| @ -103,11 +102,11 @@ export class ElectronZoo { | ||||
|             return resolve(undefined) | ||||
|           } | ||||
|  | ||||
|           if (Date.now() - timeA > 3000) { | ||||
|           if (Date.now() - timeA > 10000) { | ||||
|             return resolve(undefined) | ||||
|           } | ||||
|  | ||||
|           setTimeout(checkDisconnected, 1) | ||||
|           setTimeout(checkDisconnected, 0) | ||||
|         } | ||||
|         checkDisconnected() | ||||
|       }) | ||||
| @ -129,9 +128,11 @@ export class ElectronZoo { | ||||
|     const that = this | ||||
|  | ||||
|     const options = { | ||||
|       timeout: 120000, | ||||
|       args: ['.', '--no-sandbox'], | ||||
|       env: { | ||||
|         ...process.env, | ||||
|         TEST_SETTINGS_FILE_KEY: this.projectDirName, | ||||
|         IS_PLAYWRIGHT: 'true', | ||||
|       }, | ||||
|       ...(process.env.ELECTRON_OVERRIDE_DIST_PATH | ||||
| @ -199,14 +200,7 @@ export class ElectronZoo { | ||||
|  | ||||
|     await this.context.tracing.startChunk() | ||||
|  | ||||
|     // THIS IS ABSOLUTELY NECESSARY TO CHANGE THE PROJECT DIRECTORY BETWEEN | ||||
|     // TESTS BECAUSE OF THE ELECTRON INSTANCE REUSE. | ||||
|     await this.electron?.evaluate(({ app }, projectDirName) => { | ||||
|       // @ts-ignore can't declaration merge see main.ts | ||||
|       app.testProperty['TEST_SETTINGS_FILE_KEY'] = projectDirName | ||||
|     }, this.projectDirName) | ||||
|  | ||||
|     await setup(this.context, this.page, this.projectDirName, testInfo) | ||||
|     await setup(this.context, this.page, testInfo) | ||||
|  | ||||
|     await this.cleanProjectDir() | ||||
|  | ||||
| @ -256,6 +250,11 @@ export class ElectronZoo { | ||||
|     //   return app.reuseWindowForTest(); | ||||
|     // }); | ||||
|  | ||||
|     await this.electron?.evaluate(({ app }, projectDirName) => { | ||||
|       // @ts-ignore can't declaration merge see main.ts | ||||
|       app.testProperty['TEST_SETTINGS_FILE_KEY'] = projectDirName | ||||
|     }, this.projectDirName) | ||||
|  | ||||
|     // Always start at the root view | ||||
|     await this.page.goto(this.firstUrl) | ||||
|  | ||||
| @ -279,9 +278,8 @@ export class ElectronZoo { | ||||
|       // Not a problem if it already exists. | ||||
|     } | ||||
|  | ||||
|     const tempSettingsFilePath = path.resolve( | ||||
|     const tempSettingsFilePath = path.join( | ||||
|       this.projectDirName, | ||||
|       '..', | ||||
|       SETTINGS_FILE_NAME | ||||
|     ) | ||||
|  | ||||
|  | ||||
| @ -43,15 +43,21 @@ type DragFromHandler = ( | ||||
| export class SceneFixture { | ||||
|   public page: Page | ||||
|   public streamWrapper!: Locator | ||||
|   public loadingIndicator!: Locator | ||||
|   public networkToggleConnected!: Locator | ||||
|   public startEditSketchBtn!: Locator | ||||
|  | ||||
|   get exeIndicator() { | ||||
|     return this.page | ||||
|       .getByTestId('model-state-indicator-execution-done') | ||||
|       .or(this.page.getByTestId('model-state-indicator-receive-reliable')) | ||||
|   } | ||||
|  | ||||
|   constructor(page: Page) { | ||||
|     this.page = page | ||||
|     this.streamWrapper = page.getByTestId('stream') | ||||
|     this.networkToggleConnected = page | ||||
|       .getByTestId('network-toggle-ok') | ||||
|       .or(page.getByTestId('network-toggle-other')) | ||||
|     this.networkToggleConnected = page.getByTestId('network-toggle-ok') | ||||
|     this.loadingIndicator = this.streamWrapper.getByTestId('loading') | ||||
|     this.startEditSketchBtn = page | ||||
|       .getByRole('button', { name: 'Start Sketch' }) | ||||
|       .or(page.getByRole('button', { name: 'Edit Sketch' })) | ||||
| @ -225,6 +231,10 @@ export class SceneFixture { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   waitForExecutionDone = async () => { | ||||
|     await expect(this.exeIndicator).toBeVisible({ timeout: 30000 }) | ||||
|   } | ||||
|  | ||||
|   connectionEstablished = async () => { | ||||
|     const timeout = 30000 | ||||
|     await expect(this.networkToggleConnected).toBeVisible({ timeout }) | ||||
| @ -233,9 +243,6 @@ export class SceneFixture { | ||||
|   settled = async (cmdBar: CmdBarFixture) => { | ||||
|     const u = await getUtils(this.page) | ||||
|  | ||||
|     await expect(this.startEditSketchBtn).not.toBeDisabled() | ||||
|     await expect(this.startEditSketchBtn).toBeVisible() | ||||
|  | ||||
|     await cmdBar.openCmdBar() | ||||
|     await cmdBar.chooseCommand('Settings · app · show debug panel') | ||||
|     await cmdBar.selectOption({ name: 'on' }).click() | ||||
| @ -243,6 +250,10 @@ export class SceneFixture { | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
|     await this.waitForExecutionDone() | ||||
|     await expect(this.startEditSketchBtn).not.toBeDisabled() | ||||
|     await expect(this.startEditSketchBtn).toBeVisible() | ||||
|   } | ||||
|  | ||||
|   expectPixelColor = async ( | ||||
|  | ||||
| @ -44,7 +44,6 @@ export class ToolbarFixture { | ||||
|   featureTreePane!: Locator | ||||
|   gizmo!: Locator | ||||
|   gizmoDisabled!: Locator | ||||
|   insertButton!: Locator | ||||
|  | ||||
|   constructor(page: Page) { | ||||
|     this.page = page | ||||
| @ -79,14 +78,18 @@ export class ToolbarFixture { | ||||
|     // element or two different elements can represent these states. | ||||
|     this.gizmo = page.getByTestId('gizmo') | ||||
|     this.gizmoDisabled = page.getByTestId('gizmo-disabled') | ||||
|  | ||||
|     this.insertButton = page.getByTestId('insert-pane-button') | ||||
|   } | ||||
|  | ||||
|   get logoLink() { | ||||
|     return this.page.getByTestId('app-logo') | ||||
|   } | ||||
|  | ||||
|   get exeIndicator() { | ||||
|     return this.page | ||||
|       .getByTestId('model-state-indicator-receive-reliable') | ||||
|       .or(this.page.getByTestId('model-state-indicator-execution-done')) | ||||
|   } | ||||
|  | ||||
|   startSketchPlaneSelection = async () => | ||||
|     doAndWaitForImageDiff(this.page, () => this.startSketchBtn.click(), 500) | ||||
|  | ||||
| @ -162,14 +165,20 @@ export class ToolbarFixture { | ||||
|     } | ||||
|   } | ||||
|   /** | ||||
|    * Opens file by it's name | ||||
|    * Opens file by it's name and waits for execution to finish | ||||
|    */ | ||||
|   openFile = async (fileName: string) => { | ||||
|   openFile = async ( | ||||
|     fileName: string, | ||||
|     { wait }: { wait?: boolean } = { wait: true } | ||||
|   ) => { | ||||
|     await this.filePane.getByText(fileName).click() | ||||
|     if (wait) { | ||||
|       await expect(this.exeIndicator).toBeVisible({ timeout: 15_000 }) | ||||
|     } | ||||
|   } | ||||
|   selectCenterRectangle = async () => { | ||||
|     await this.page | ||||
|       .getByRole('button', { name: 'caret down rectangles:' }) | ||||
|       .getByRole('button', { name: 'caret down Corner rectangle:' }) | ||||
|       .click() | ||||
|     await expect( | ||||
|       this.page.getByTestId('dropdown-center-rectangle') | ||||
| @ -178,7 +187,7 @@ export class ToolbarFixture { | ||||
|   } | ||||
|   selectBoolean = async (operation: 'union' | 'subtract' | 'intersect') => { | ||||
|     await this.page | ||||
|       .getByRole('button', { name: 'caret down booleans: open menu' }) | ||||
|       .getByRole('button', { name: 'caret down Union: open menu' }) | ||||
|       .click() | ||||
|     const operationTestId = `dropdown-boolean-${operation}` | ||||
|     await expect(this.page.getByTestId(operationTestId)).toBeVisible() | ||||
| @ -186,19 +195,25 @@ export class ToolbarFixture { | ||||
|   } | ||||
|  | ||||
|   selectCircleThreePoint = async () => { | ||||
|     await this.page.getByRole('button', { name: 'caret down circles:' }).click() | ||||
|     await this.page | ||||
|       .getByRole('button', { name: 'caret down Center circle:' }) | ||||
|       .click() | ||||
|     await expect( | ||||
|       this.page.getByTestId('dropdown-circle-three-points') | ||||
|     ).toBeVisible() | ||||
|     await this.page.getByTestId('dropdown-circle-three-points').click() | ||||
|   } | ||||
|   selectArc = async () => { | ||||
|     await this.page.getByRole('button', { name: 'caret down arcs:' }).click() | ||||
|     await this.page | ||||
|       .getByRole('button', { name: 'caret down Tangential Arc:' }) | ||||
|       .click() | ||||
|     await expect(this.page.getByTestId('dropdown-arc')).toBeVisible() | ||||
|     await this.page.getByTestId('dropdown-arc').click() | ||||
|   } | ||||
|   selectThreePointArc = async () => { | ||||
|     await this.page.getByRole('button', { name: 'caret down arcs:' }).click() | ||||
|     await this.page | ||||
|       .getByRole('button', { name: 'caret down Tangential Arc:' }) | ||||
|       .click() | ||||
|     await expect( | ||||
|       this.page.getByTestId('dropdown-three-point-arc') | ||||
|     ).toBeVisible() | ||||
|  | ||||
| @ -10,7 +10,6 @@ test.describe('Import UI tests', () => { | ||||
|     toolbar, | ||||
|     scene, | ||||
|     editor, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const projectDir = path.join(dir, 'import-test') | ||||
| @ -21,9 +20,15 @@ test.describe('Import UI tests', () => { | ||||
|         path.join(projectDir, 'toBeImported.kcl'), | ||||
|         `sketch001 = startSketchOn(XZ) | ||||
| profile001 = startProfileAt([281.54, 305.81], sketch001) | ||||
|   |> angledLine(angle = 0, length = 123.43, tag = $rectangleSegmentA001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 85.99) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001)) | ||||
|   |> angledLine([0, 123.43], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        85.99 | ||||
|      ], %) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| extrude(profile001, length = 100)` | ||||
| @ -38,9 +43,15 @@ importedCube | ||||
|  | ||||
| sketch001 = startSketchOn(XZ) | ||||
| profile001 = startProfileAt([-134.53, -56.17], sketch001) | ||||
|   |> angledLine(angle = 0, length = 79.05, tag = $rectangleSegmentA001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 76.28) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001), tag = $seg01) | ||||
|   |> angledLine([0, 79.05], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        76.28 | ||||
|      ], %) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $seg01) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)], tag = $seg02) | ||||
|   |> close() | ||||
| extrude001 = extrude(profile001, length = 100) | ||||
| @ -50,7 +61,7 @@ sketch002 = startSketchOn(extrude001, seg01)` | ||||
|     }) | ||||
|  | ||||
|     await homePage.openProject('import-test') | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     await scene.moveCameraTo( | ||||
|       { | ||||
|  | ||||
| @ -7,7 +7,7 @@ import { expect, test } from '@e2e/playwright/zoo-test' | ||||
| test( | ||||
|   'When machine-api server not found butt is disabled and shows the reason', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ context, page, scene, cmdBar }, testInfo) => { | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const bracketDir = join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
| @ -23,7 +23,10 @@ test( | ||||
|  | ||||
|     await page.getByText('bracket').click() | ||||
|  | ||||
|     await scene.settled(cmdBar) | ||||
|     await expect(page.getByTestId('loading')).toBeAttached() | ||||
|     await expect(page.getByTestId('loading')).not.toBeAttached({ | ||||
|       timeout: 20_000, | ||||
|     }) | ||||
|  | ||||
|     const notFoundText = 'Machine API server was not discovered' | ||||
|     await expect(page.getByText(notFoundText).first()).not.toBeVisible() | ||||
| @ -44,7 +47,7 @@ test( | ||||
| test( | ||||
|   'When machine-api server not found home screen & project status shows the reason', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ context, page, scene, cmdBar }, testInfo) => { | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const bracketDir = join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
| @ -68,7 +71,10 @@ test( | ||||
|  | ||||
|     await page.getByText('bracket').click() | ||||
|  | ||||
|     await scene.settled(cmdBar) | ||||
|     await expect(page.getByTestId('loading')).toBeAttached() | ||||
|     await expect(page.getByTestId('loading')).not.toBeAttached({ | ||||
|       timeout: 20_000, | ||||
|     }) | ||||
|  | ||||
|     await expect(page.getByText(notFoundText).nth(1)).not.toBeVisible() | ||||
|  | ||||
|  | ||||
| @ -61,7 +61,6 @@ function tomlStringOverWriteNamedViewUuids(toml: string): string { | ||||
| } | ||||
|  | ||||
| test.describe('Named view tests', () => { | ||||
|   test.skip() // TODO: Jace is working on these | ||||
|   test('Verify project.toml is not created', async ({ page }, testInfo) => { | ||||
|     // Create project and load it | ||||
|     const projectName = 'named-views' | ||||
| @ -89,7 +88,7 @@ test.describe('Named view tests', () => { | ||||
|  | ||||
|     // Create and load project | ||||
|     await createProject({ name: projectName, page }) | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // Create named view | ||||
|     const projectDirName = testInfo.outputPath('electron-test-projects-dir') | ||||
| @ -111,17 +110,14 @@ test.describe('Named view tests', () => { | ||||
|       expect(exists).toBe(true) | ||||
|     }).toPass() | ||||
|  | ||||
|     await expect(async () => { | ||||
|       // Read project.toml into memory | ||||
|       let tomlString = await fsp.readFile(tempProjectSettingsFilePath, 'utf-8') | ||||
|     // Read project.toml into memory | ||||
|     let tomlString = await fsp.readFile(tempProjectSettingsFilePath, 'utf-8') | ||||
|     // Rewrite the uuids in the named views to match snapshot otherwise they will be randomly generated from rust and break | ||||
|     tomlString = tomlStringOverWriteNamedViewUuids(tomlString) | ||||
|  | ||||
|       // Rewrite the uuids in the named views to match snapshot otherwise they will be randomly generated from rust and break | ||||
|       tomlString = tomlStringOverWriteNamedViewUuids(tomlString) | ||||
|  | ||||
|       // Write the entire tomlString to a snapshot. | ||||
|       // There are many key/value pairs to check this is a safer match. | ||||
|       expect(tomlString).toMatchSnapshot('verify-named-view-gets-created') | ||||
|     }).toPass() | ||||
|     // Write the entire tomlString to a snapshot. | ||||
|     // There are many key/value pairs to check this is a safer match. | ||||
|     expect(tomlString).toMatchSnapshot('verify-named-view-gets-created') | ||||
|   }) | ||||
|   test('Verify named view gets deleted', async ({ | ||||
|     cmdBar, | ||||
| @ -134,7 +130,7 @@ test.describe('Named view tests', () => { | ||||
|  | ||||
|     // Create project and go into the project | ||||
|     await createProject({ name: projectName, page }) | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // Create a new named view | ||||
|     await cmdBar.openCmdBar() | ||||
| @ -156,16 +152,14 @@ test.describe('Named view tests', () => { | ||||
|       expect(exists).toBe(true) | ||||
|     }).toPass() | ||||
|  | ||||
|     await expect(async () => { | ||||
|       // Read project.toml into memory | ||||
|       let tomlString = await fsp.readFile(tempProjectSettingsFilePath, 'utf-8') | ||||
|       // Rewrite the uuids in the named views to match snapshot otherwise they will be randomly generated from rust and break | ||||
|       tomlString = tomlStringOverWriteNamedViewUuids(tomlString) | ||||
|     // Read project.toml into memory | ||||
|     let tomlString = await fsp.readFile(tempProjectSettingsFilePath, 'utf-8') | ||||
|     // Rewrite the uuids in the named views to match snapshot otherwise they will be randomly generated from rust and break | ||||
|     tomlString = tomlStringOverWriteNamedViewUuids(tomlString) | ||||
|  | ||||
|       // Write the entire tomlString to a snapshot. | ||||
|       // There are many key/value pairs to check this is a safer match. | ||||
|       expect(tomlString).toMatchSnapshot('verify-named-view-gets-created') | ||||
|     }).toPass() | ||||
|     // Write the entire tomlString to a snapshot. | ||||
|     // There are many key/value pairs to check this is a safer match. | ||||
|     expect(tomlString).toMatchSnapshot('verify-named-view-gets-created') | ||||
|  | ||||
|     // Delete a named view | ||||
|     await cmdBar.openCmdBar() | ||||
| @ -173,16 +167,14 @@ test.describe('Named view tests', () => { | ||||
|     cmdBar.selectOption({ name: myNamedView2 }) | ||||
|     await cmdBar.progressCmdBar(false) | ||||
|  | ||||
|     await expect(async () => { | ||||
|       // Read project.toml into memory again since we deleted a named view | ||||
|       let tomlString = await fsp.readFile(tempProjectSettingsFilePath, 'utf-8') | ||||
|       // Rewrite the uuids in the named views to match snapshot otherwise they will be randomly generated from rust and break | ||||
|       tomlString = tomlStringOverWriteNamedViewUuids(tomlString) | ||||
|     // Read project.toml into memory again since we deleted a named view | ||||
|     tomlString = await fsp.readFile(tempProjectSettingsFilePath, 'utf-8') | ||||
|     // Rewrite the uuids in the named views to match snapshot otherwise they will be randomly generated from rust and break | ||||
|     tomlString = tomlStringOverWriteNamedViewUuids(tomlString) | ||||
|  | ||||
|       // // Write the entire tomlString to a snapshot. | ||||
|       // // There are many key/value pairs to check this is a safer match. | ||||
|       expect(tomlString).toMatchSnapshot('verify-named-view-gets-deleted') | ||||
|     }).toPass() | ||||
|     // // Write the entire tomlString to a snapshot. | ||||
|     // // There are many key/value pairs to check this is a safer match. | ||||
|     expect(tomlString).toMatchSnapshot('verify-named-view-gets-deleted') | ||||
|   }) | ||||
|   test('Verify named view gets loaded', async ({ | ||||
|     cmdBar, | ||||
| @ -194,7 +186,7 @@ test.describe('Named view tests', () => { | ||||
|  | ||||
|     // Create project and go into the project | ||||
|     await createProject({ name: projectName, page }) | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // Create a new named view | ||||
|     await cmdBar.openCmdBar() | ||||
| @ -216,16 +208,14 @@ test.describe('Named view tests', () => { | ||||
|       expect(exists).toBe(true) | ||||
|     }).toPass() | ||||
|  | ||||
|     await expect(async () => { | ||||
|       // Read project.toml into memory | ||||
|       let tomlString = await fsp.readFile(tempProjectSettingsFilePath, 'utf-8') | ||||
|       // Rewrite the uuids in the named views to match snapshot otherwise they will be randomly generated from rust and break | ||||
|       tomlString = tomlStringOverWriteNamedViewUuids(tomlString) | ||||
|     // Read project.toml into memory | ||||
|     let tomlString = await fsp.readFile(tempProjectSettingsFilePath, 'utf-8') | ||||
|     // Rewrite the uuids in the named views to match snapshot otherwise they will be randomly generated from rust and break | ||||
|     tomlString = tomlStringOverWriteNamedViewUuids(tomlString) | ||||
|  | ||||
|       // Write the entire tomlString to a snapshot. | ||||
|       // There are many key/value pairs to check this is a safer match. | ||||
|       expect(tomlString).toMatchSnapshot('verify-named-view-gets-created') | ||||
|     }).toPass() | ||||
|     // Write the entire tomlString to a snapshot. | ||||
|     // There are many key/value pairs to check this is a safer match. | ||||
|     expect(tomlString).toMatchSnapshot('verify-named-view-gets-created') | ||||
|  | ||||
|     // Create a load a named view | ||||
|     await cmdBar.openCmdBar() | ||||
| @ -249,7 +239,7 @@ test.describe('Named view tests', () => { | ||||
|  | ||||
|     // Create and load project | ||||
|     await createProject({ name: projectName, page }) | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // Create named view | ||||
|     const projectDirName = testInfo.outputPath('electron-test-projects-dir') | ||||
| @ -292,15 +282,13 @@ test.describe('Named view tests', () => { | ||||
|       expect(exists).toBe(true) | ||||
|     }).toPass() | ||||
|  | ||||
|     await expect(async () => { | ||||
|       // Read project.toml into memory | ||||
|       let tomlString = await fsp.readFile(tempProjectSettingsFilePath, 'utf-8') | ||||
|       // Rewrite the uuids in the named views to match snapshot otherwise they will be randomly generated from rust and break | ||||
|       tomlString = tomlStringOverWriteNamedViewUuids(tomlString) | ||||
|     // Read project.toml into memory | ||||
|     let tomlString = await fsp.readFile(tempProjectSettingsFilePath, 'utf-8') | ||||
|     // Rewrite the uuids in the named views to match snapshot otherwise they will be randomly generated from rust and break | ||||
|     tomlString = tomlStringOverWriteNamedViewUuids(tomlString) | ||||
|  | ||||
|       // Write the entire tomlString to a snapshot. | ||||
|       // There are many key/value pairs to check this is a safer match. | ||||
|       expect(tomlString).toMatchSnapshot('verify-two-named-view-gets-created') | ||||
|     }).toPass() | ||||
|     // Write the entire tomlString to a snapshot. | ||||
|     // There are many key/value pairs to check this is a safer match. | ||||
|     expect(tomlString).toMatchSnapshot('verify-two-named-view-gets-created') | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -1,16 +0,0 @@ | ||||
| [settings] | ||||
| modeling = { } | ||||
| text_editor = { } | ||||
| command_bar = { } | ||||
|  | ||||
| [settings.app.named_views.0656fb1a-9640-473e-b334-591dc70c0138] | ||||
| name = "uuid1" | ||||
| eye_offset = 1_378.0059 | ||||
| fov_y = 45 | ||||
| is_ortho = false | ||||
| ortho_scale_enabled = true | ||||
| ortho_scale_factor = 1.6 | ||||
| pivot_position = [ 0, 0, 0 ] | ||||
| pivot_rotation = [ 0.5380994, 0.0, 0.0, 0.8428814 ] | ||||
| world_coord_system = "right_handed_up_z" | ||||
| version = 1 | ||||
| @ -1,28 +0,0 @@ | ||||
| [settings] | ||||
| modeling = { } | ||||
| text_editor = { } | ||||
| command_bar = { } | ||||
|  | ||||
| [settings.app.named_views.0656fb1a-9640-473e-b334-591dc70c0138] | ||||
| name = "uuid1" | ||||
| eye_offset = 1_378.0059 | ||||
| fov_y = 45 | ||||
| is_ortho = false | ||||
| ortho_scale_enabled = true | ||||
| ortho_scale_factor = 1.6 | ||||
| pivot_position = [ 0, 0, 0 ] | ||||
| pivot_rotation = [ 0.5380994, 0.0, 0.0, 0.8428814 ] | ||||
| world_coord_system = "right_handed_up_z" | ||||
| version = 1 | ||||
|  | ||||
| [settings.app.named_views.c810cf04-c6cc-4a4a-8b11-17bf445dcab7] | ||||
| name = "uuid2" | ||||
| eye_offset = 1_378.0059 | ||||
| fov_y = 45 | ||||
| is_ortho = false | ||||
| ortho_scale_enabled = true | ||||
| ortho_scale_factor = 1.6 | ||||
| pivot_position = [ 1_826.5239, 0.0, 0.0 ] | ||||
| pivot_rotation = [ 0.5380994, 0.0, 0.0, 0.8428814 ] | ||||
| world_coord_system = "right_handed_up_z" | ||||
| version = 1 | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -48,10 +48,7 @@ test.describe('Onboarding tests', () => { | ||||
|     await expect(page.getByText('Welcome to Design Studio! This')).toBeVisible() | ||||
|  | ||||
|     // *and* that the code is shown in the editor | ||||
|     await expect(page.locator('.cm-content')).toContainText( | ||||
|       '// Shelf Bracket', | ||||
|       { timeout: 10_000 } | ||||
|     ) | ||||
|     await expect(page.locator('.cm-content')).toContainText('// Shelf Bracket') | ||||
|  | ||||
|     // Make sure the model loaded | ||||
|     const XYPlanePoint = { x: 774, y: 116 } as const | ||||
| @ -94,8 +91,7 @@ test.describe('Onboarding tests', () => { | ||||
|  | ||||
|         // *and* that the code is shown in the editor | ||||
|         await expect(page.locator('.cm-content')).toContainText( | ||||
|           '// Shelf Bracket', | ||||
|           { timeout: 10_000 } | ||||
|           '// Shelf Bracket' | ||||
|         ) | ||||
|  | ||||
|         // TODO: jess make less shit | ||||
|  | ||||
| @ -1,292 +0,0 @@ | ||||
| import * as fsp from 'fs/promises' | ||||
| import path from 'path' | ||||
|  | ||||
| import type { CmdBarFixture } from '@e2e/playwright/fixtures/cmdBarFixture' | ||||
| import type { ToolbarFixture } from '@e2e/playwright/fixtures/toolbarFixture' | ||||
| import { | ||||
|   executorInputPath, | ||||
|   getUtils, | ||||
|   testsInputPath, | ||||
| } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
| import type { Page } from '@playwright/test' | ||||
|  | ||||
| async function insertPartIntoAssembly( | ||||
|   path: string, | ||||
|   alias: string, | ||||
|   toolbar: ToolbarFixture, | ||||
|   cmdBar: CmdBarFixture, | ||||
|   page: Page | ||||
| ) { | ||||
|   await toolbar.insertButton.click() | ||||
|   await cmdBar.selectOption({ name: path }).click() | ||||
|   await cmdBar.expectState({ | ||||
|     stage: 'arguments', | ||||
|     currentArgKey: 'localName', | ||||
|     currentArgValue: '', | ||||
|     headerArguments: { Path: path, LocalName: '' }, | ||||
|     highlightedHeaderArg: 'localName', | ||||
|     commandName: 'Insert', | ||||
|   }) | ||||
|   await page.keyboard.insertText(alias) | ||||
|   await cmdBar.progressCmdBar() | ||||
|   await cmdBar.expectState({ | ||||
|     stage: 'review', | ||||
|     headerArguments: { Path: path, LocalName: alias }, | ||||
|     commandName: 'Insert', | ||||
|   }) | ||||
|   await cmdBar.progressCmdBar() | ||||
| } | ||||
|  | ||||
| // test file is for testing point an click code gen functionality that's assemblies related | ||||
| test.describe('Point-and-click assemblies tests', () => { | ||||
|   test( | ||||
|     `Insert kcl parts into assembly as whole module import`, | ||||
|     { tag: ['@electron'] }, | ||||
|     async ({ | ||||
|       context, | ||||
|       page, | ||||
|       homePage, | ||||
|       scene, | ||||
|       editor, | ||||
|       toolbar, | ||||
|       cmdBar, | ||||
|       tronApp, | ||||
|     }) => { | ||||
|       if (!tronApp) { | ||||
|         fail() | ||||
|       } | ||||
|  | ||||
|       const midPoint = { x: 500, y: 250 } | ||||
|       const partPoint = { x: midPoint.x + 30, y: midPoint.y - 30 } // mid point, just off top right | ||||
|       const defaultPlanesColor: [number, number, number] = [180, 220, 180] | ||||
|       const partColor: [number, number, number] = [100, 100, 100] | ||||
|       const tolerance = 50 | ||||
|       const u = await getUtils(page) | ||||
|       const gizmo = page.locator('[aria-label*=gizmo]') | ||||
|       const resetCameraButton = page.getByRole('button', { name: 'Reset view' }) | ||||
|  | ||||
|       await test.step('Setup parts and expect empty assembly scene', async () => { | ||||
|         const projectName = 'assembly' | ||||
|         await context.folderSetupFn(async (dir) => { | ||||
|           const bracketDir = path.join(dir, projectName) | ||||
|           await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|           await Promise.all([ | ||||
|             fsp.copyFile( | ||||
|               executorInputPath('cylinder.kcl'), | ||||
|               path.join(bracketDir, 'cylinder.kcl') | ||||
|             ), | ||||
|             fsp.copyFile( | ||||
|               executorInputPath('e2e-can-sketch-on-chamfer.kcl'), | ||||
|               path.join(bracketDir, 'bracket.kcl') | ||||
|             ), | ||||
|             fsp.copyFile( | ||||
|               testsInputPath('cube.step'), | ||||
|               path.join(bracketDir, 'cube.step') | ||||
|             ), | ||||
|             fsp.writeFile(path.join(bracketDir, 'main.kcl'), ''), | ||||
|           ]) | ||||
|         }) | ||||
|         await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|         await homePage.openProject(projectName) | ||||
|         await scene.settled(cmdBar) | ||||
|         await toolbar.closePane('code') | ||||
|         await scene.expectPixelColor(defaultPlanesColor, midPoint, tolerance) | ||||
|       }) | ||||
|  | ||||
|       await test.step('Insert kcl as first part as module', async () => { | ||||
|         await insertPartIntoAssembly( | ||||
|           'cylinder.kcl', | ||||
|           'cylinder', | ||||
|           toolbar, | ||||
|           cmdBar, | ||||
|           page | ||||
|         ) | ||||
|         await toolbar.openPane('code') | ||||
|         await editor.expectEditor.toContain( | ||||
|           ` | ||||
|         import "cylinder.kcl" as cylinder | ||||
|         cylinder | ||||
|       `, | ||||
|           { shouldNormalise: true } | ||||
|         ) | ||||
|         await scene.settled(cmdBar) | ||||
|  | ||||
|         // Check scene for changes | ||||
|         await toolbar.closePane('code') | ||||
|         await u.doAndWaitForCmd(async () => { | ||||
|           await gizmo.click({ button: 'right' }) | ||||
|           await resetCameraButton.click() | ||||
|         }, 'zoom_to_fit') | ||||
|         await toolbar.closePane('debug') | ||||
|         await scene.expectPixelColor(partColor, partPoint, tolerance) | ||||
|         await toolbar.openPane('code') | ||||
|       }) | ||||
|  | ||||
|       await test.step('Insert kcl second part as module', async () => { | ||||
|         await insertPartIntoAssembly( | ||||
|           'bracket.kcl', | ||||
|           'bracket', | ||||
|           toolbar, | ||||
|           cmdBar, | ||||
|           page | ||||
|         ) | ||||
|         await editor.expectEditor.toContain( | ||||
|           ` | ||||
|         import "cylinder.kcl" as cylinder | ||||
|         import "bracket.kcl" as bracket | ||||
|         cylinder | ||||
|         bracket | ||||
|       `, | ||||
|           { shouldNormalise: true } | ||||
|         ) | ||||
|         await scene.settled(cmdBar) | ||||
|       }) | ||||
|  | ||||
|       await test.step('Insert a second time and expect error', async () => { | ||||
|         // TODO: revisit once we have clone with #6209 | ||||
|         await insertPartIntoAssembly( | ||||
|           'bracket.kcl', | ||||
|           'bracket', | ||||
|           toolbar, | ||||
|           cmdBar, | ||||
|           page | ||||
|         ) | ||||
|         await editor.expectEditor.toContain( | ||||
|           ` | ||||
|         import "cylinder.kcl" as cylinder | ||||
|         import "bracket.kcl" as bracket | ||||
|         import "bracket.kcl" as bracket | ||||
|         cylinder | ||||
|         bracket | ||||
|         bracket | ||||
|       `, | ||||
|           { shouldNormalise: true } | ||||
|         ) | ||||
|         await scene.settled(cmdBar) | ||||
|         await expect(page.locator('.cm-lint-marker-error')).toBeVisible() | ||||
|       }) | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test( | ||||
|     `Insert foreign parts into assembly as whole module import`, | ||||
|     { tag: ['@electron'] }, | ||||
|     async ({ | ||||
|       context, | ||||
|       page, | ||||
|       homePage, | ||||
|       scene, | ||||
|       editor, | ||||
|       toolbar, | ||||
|       cmdBar, | ||||
|       tronApp, | ||||
|     }) => { | ||||
|       if (!tronApp) { | ||||
|         fail() | ||||
|       } | ||||
|  | ||||
|       const midPoint = { x: 500, y: 250 } | ||||
|       const partPoint = { x: midPoint.x + 30, y: midPoint.y - 30 } // mid point, just off top right | ||||
|       const defaultPlanesColor: [number, number, number] = [180, 220, 180] | ||||
|       const partColor: [number, number, number] = [150, 150, 150] | ||||
|       const tolerance = 50 | ||||
|  | ||||
|       const complexPlmFileName = 'cube_Complex-PLM_Name_-001.sldprt' | ||||
|       const camelCasedSolidworksFileName = 'cubeComplexPLMName001' | ||||
|  | ||||
|       await test.step('Setup parts and expect empty assembly scene', async () => { | ||||
|         const projectName = 'assembly' | ||||
|         await context.folderSetupFn(async (dir) => { | ||||
|           const bracketDir = path.join(dir, projectName) | ||||
|           await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|           await Promise.all([ | ||||
|             fsp.copyFile( | ||||
|               testsInputPath('cube.step'), | ||||
|               path.join(bracketDir, 'cube.step') | ||||
|             ), | ||||
|             fsp.copyFile( | ||||
|               testsInputPath('cube.sldprt'), | ||||
|               path.join(bracketDir, complexPlmFileName) | ||||
|             ), | ||||
|             fsp.writeFile(path.join(bracketDir, 'main.kcl'), ''), | ||||
|           ]) | ||||
|         }) | ||||
|         await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|         await homePage.openProject(projectName) | ||||
|         await scene.settled(cmdBar) | ||||
|         await toolbar.closePane('code') | ||||
|         await scene.expectPixelColor(defaultPlanesColor, midPoint, tolerance) | ||||
|       }) | ||||
|  | ||||
|       await test.step('Insert step part as module', async () => { | ||||
|         await insertPartIntoAssembly('cube.step', 'cube', toolbar, cmdBar, page) | ||||
|         await toolbar.openPane('code') | ||||
|         await editor.expectEditor.toContain( | ||||
|           ` | ||||
|         import "cube.step" as cube | ||||
|         cube | ||||
|       `, | ||||
|           { shouldNormalise: true } | ||||
|         ) | ||||
|         await scene.settled(cmdBar) | ||||
|  | ||||
|         // TODO: remove this once #5780 is fixed | ||||
|         await page.reload() | ||||
|  | ||||
|         await scene.settled(cmdBar) | ||||
|         await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|         await toolbar.closePane('code') | ||||
|         await scene.expectPixelColor(partColor, partPoint, tolerance) | ||||
|       }) | ||||
|  | ||||
|       await test.step('Insert second step part by clicking', async () => { | ||||
|         await toolbar.openPane('files') | ||||
|         await toolbar.expectFileTreeState([ | ||||
|           complexPlmFileName, | ||||
|           'cube.step', | ||||
|           'main.kcl', | ||||
|         ]) | ||||
|         await toolbar.openFile(complexPlmFileName) | ||||
|  | ||||
|         // Go through the ToastInsert prompt | ||||
|         await page.getByText('Insert into my current file').click() | ||||
|  | ||||
|         // Check getPathFilenameInVariableCase output | ||||
|         const parsedValueFromFile = | ||||
|           await cmdBar.currentArgumentInput.inputValue() | ||||
|         expect(parsedValueFromFile).toEqual(camelCasedSolidworksFileName) | ||||
|  | ||||
|         // Continue on with the flow | ||||
|         await page.keyboard.insertText('cubeSw') | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await cmdBar.expectState({ | ||||
|           stage: 'review', | ||||
|           headerArguments: { Path: complexPlmFileName, LocalName: 'cubeSw' }, | ||||
|           commandName: 'Insert', | ||||
|         }) | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await toolbar.closePane('files') | ||||
|         await toolbar.openPane('code') | ||||
|         await editor.expectEditor.toContain( | ||||
|           ` | ||||
|         import "cube.step" as cube | ||||
|         import "${complexPlmFileName}" as cubeSw | ||||
|         cube | ||||
|         cubeSw | ||||
|       `, | ||||
|           { shouldNormalise: true } | ||||
|         ) | ||||
|         await scene.settled(cmdBar) | ||||
|  | ||||
|         // TODO: remove this once #5780 is fixed | ||||
|         await page.reload() | ||||
|         await scene.settled(cmdBar) | ||||
|  | ||||
|         await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|         await toolbar.closePane('code') | ||||
|         await scene.expectPixelColor(partColor, partPoint, tolerance) | ||||
|       }) | ||||
|     } | ||||
|   ) | ||||
| }) | ||||
| @ -5,14 +5,13 @@ import path from 'node:path' | ||||
| import type { EditorFixture } from '@e2e/playwright/fixtures/editorFixture' | ||||
| import type { SceneFixture } from '@e2e/playwright/fixtures/sceneFixture' | ||||
| import type { ToolbarFixture } from '@e2e/playwright/fixtures/toolbarFixture' | ||||
| import { orRunWhenFullSuiteEnabled } from '@e2e/playwright/test-utils' | ||||
| import { getUtils, orRunWhenFullSuiteEnabled } from '@e2e/playwright/test-utils' | ||||
| import { expect, test } from '@e2e/playwright/zoo-test' | ||||
|  | ||||
| // test file is for testing point an click code gen functionality that's not sketch mode related | ||||
|  | ||||
| test.describe('Point-and-click tests', () => { | ||||
|   test('verify extruding circle works', async ({ | ||||
|     page, | ||||
|     context, | ||||
|     homePage, | ||||
|     cmdBar, | ||||
| @ -31,9 +30,8 @@ test.describe('Point-and-click tests', () => { | ||||
|     await context.addInitScript((file) => { | ||||
|       localStorage.setItem('persistCode', file) | ||||
|     }, file) | ||||
|  | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.connectionEstablished() | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     const [clickCircle, moveToCircle] = scene.makeMouseHelpers(582, 217) | ||||
|  | ||||
| @ -74,6 +72,7 @@ test.describe('Point-and-click tests', () => { | ||||
|  | ||||
|     await test.step('do extrude flow and check extrude code is added to editor', async () => { | ||||
|       await toolbar.extrudeButton.click() | ||||
|  | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'arguments', | ||||
|         currentArgKey: 'distance', | ||||
| @ -187,7 +186,6 @@ test.describe('Point-and-click tests', () => { | ||||
|       editor, | ||||
|       toolbar, | ||||
|       scene, | ||||
|       cmdBar, | ||||
|     }) => { | ||||
|       const file = await fs.readFile( | ||||
|         path.resolve( | ||||
| @ -202,7 +200,9 @@ test.describe('Point-and-click tests', () => { | ||||
|       }, file) | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.settled(cmdBar) | ||||
|       await expect( | ||||
|         page.getByTestId('model-state-indicator-receive-reliable') | ||||
|       ).toBeVisible() | ||||
|  | ||||
|       const sketchOnAChamfer = _sketchOnAChamfer(page, editor, toolbar, scene) | ||||
|  | ||||
| @ -210,7 +210,7 @@ test.describe('Point-and-click tests', () => { | ||||
|         clickCoords: { x: 570, y: 220 }, | ||||
|         cameraPos: { x: 16020, y: -2000, z: 10500 }, | ||||
|         cameraTarget: { x: -150, y: -4500, z: -80 }, | ||||
|         beforeChamferSnippet: `angledLine(angle=segAng(rectangleSegmentA001)-90,length=217.26,tag=$seg01) | ||||
|         beforeChamferSnippet: `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01) | ||||
|       chamfer(length = 30,tags = [ | ||||
|       seg01, | ||||
|       getNextAdjacentEdge(yo), | ||||
| @ -223,9 +223,9 @@ test.describe('Point-and-click tests', () => { | ||||
|           'sketch002 = startSketchOn(extrude001, seg03)', | ||||
|         afterRectangle1stClickSnippet: | ||||
|           'startProfileAt([205.96, 254.59], sketch002)', | ||||
|         afterRectangle2ndClickSnippet: `angledLine(angle=0,length=11.39,tag=$rectangleSegmentA002) | ||||
|         |>angledLine(angle=segAng(rectangleSegmentA002)-90,length=105.26) | ||||
|         |>angledLine(angle=segAng(rectangleSegmentA002),length=-segLen(rectangleSegmentA002)) | ||||
|         afterRectangle2ndClickSnippet: `angledLine([0,11.39],%,$rectangleSegmentA002) | ||||
|         |>angledLine([segAng(rectangleSegmentA002)-90,105.26],%) | ||||
|         |>angledLine([segAng(rectangleSegmentA002),-segLen(rectangleSegmentA002)],%) | ||||
|         |>line(endAbsolute=[profileStartX(%),profileStartY(%)]) | ||||
|         |>close()`, | ||||
|       }) | ||||
| @ -234,7 +234,10 @@ test.describe('Point-and-click tests', () => { | ||||
|         clickCoords: { x: 690, y: 250 }, | ||||
|         cameraPos: { x: 16020, y: -2000, z: 10500 }, | ||||
|         cameraTarget: { x: -150, y: -4500, z: -80 }, | ||||
|         beforeChamferSnippet: `angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 217.26, tag = $seg01)chamfer( | ||||
|         beforeChamferSnippet: `angledLine([ | ||||
|          segAng(rectangleSegmentA001) - 90, | ||||
|          217.26 | ||||
|        ], %, $seg01)chamfer( | ||||
|          length = 30, | ||||
|          tags = [ | ||||
|            seg01, | ||||
| @ -247,9 +250,9 @@ test.describe('Point-and-click tests', () => { | ||||
|           'sketch003 = startSketchOn(extrude001, seg04)', | ||||
|         afterRectangle1stClickSnippet: | ||||
|           'startProfileAt([-209.64, 255.28], sketch003)', | ||||
|         afterRectangle2ndClickSnippet: `angledLine(angle=0,length=11.56,tag=$rectangleSegmentA003) | ||||
|         |>angledLine(angle=segAng(rectangleSegmentA003)-90,length=106.84) | ||||
|         |>angledLine(angle=segAng(rectangleSegmentA003),length=-segLen(rectangleSegmentA003)) | ||||
|         afterRectangle2ndClickSnippet: `angledLine([0,11.56],%,$rectangleSegmentA003) | ||||
|         |>angledLine([segAng(rectangleSegmentA003)-90,106.84],%) | ||||
|         |>angledLine([segAng(rectangleSegmentA003),-segLen(rectangleSegmentA003)],%) | ||||
|         |>line(endAbsolute=[profileStartX(%),profileStartY(%)]) | ||||
|         |>close()`, | ||||
|       }) | ||||
| @ -258,7 +261,7 @@ test.describe('Point-and-click tests', () => { | ||||
|         clickCoords: { x: 677, y: 87 }, | ||||
|         cameraPos: { x: -6200, y: 1500, z: 6200 }, | ||||
|         cameraTarget: { x: 8300, y: 1100, z: 4800 }, | ||||
|         beforeChamferSnippet: `angledLine(angle = 0, length = 268.43, tag = $rectangleSegmentA001)chamfer( | ||||
|         beforeChamferSnippet: `angledLine([0, 268.43], %, $rectangleSegmentA001)chamfer( | ||||
|          length = 30, | ||||
|          tags = [ | ||||
|            getNextAdjacentEdge(yo), | ||||
| @ -269,9 +272,9 @@ test.describe('Point-and-click tests', () => { | ||||
|           'sketch004 = startSketchOn(extrude001, seg05)', | ||||
|         afterRectangle1stClickSnippet: | ||||
|           'startProfileAt([82.57, 322.96], sketch004)', | ||||
|         afterRectangle2ndClickSnippet: `angledLine(angle=0,length=11.16,tag=$rectangleSegmentA004) | ||||
|         |>angledLine(angle=segAng(rectangleSegmentA004)-90,length=103.07) | ||||
|         |>angledLine(angle=segAng(rectangleSegmentA004),length=-segLen(rectangleSegmentA004)) | ||||
|         afterRectangle2ndClickSnippet: `angledLine([0,11.16],%,$rectangleSegmentA004) | ||||
|         |>angledLine([segAng(rectangleSegmentA004)-90,103.07],%) | ||||
|         |>angledLine([segAng(rectangleSegmentA004),-segLen(rectangleSegmentA004)],%) | ||||
|         |>line(endAbsolute=[profileStartX(%),profileStartY(%)]) | ||||
|         |>close()`, | ||||
|       }) | ||||
| @ -287,9 +290,9 @@ test.describe('Point-and-click tests', () => { | ||||
|           'sketch005 = startSketchOn(extrude001, seg06)', | ||||
|         afterRectangle1stClickSnippet: | ||||
|           'startProfileAt([-23.43, 19.69], sketch005)', | ||||
|         afterRectangle2ndClickSnippet: `angledLine(angle=0,length=9.1,tag=$rectangleSegmentA005) | ||||
|         |>angledLine(angle=segAng(rectangleSegmentA005)-90,length=84.07) | ||||
|         |>angledLine(angle=segAng(rectangleSegmentA005),length=-segLen(rectangleSegmentA005)) | ||||
|         afterRectangle2ndClickSnippet: `angledLine([0,9.1],%,$rectangleSegmentA005) | ||||
|         |>angledLine([segAng(rectangleSegmentA005)-90,84.07],%) | ||||
|         |>angledLine([segAng(rectangleSegmentA005),-segLen(rectangleSegmentA005)],%) | ||||
|         |>line(endAbsolute=[profileStartX(%),profileStartY(%)]) | ||||
|         |>close()`, | ||||
|       }) | ||||
| @ -299,9 +302,15 @@ test.describe('Point-and-click tests', () => { | ||||
|           `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([75.8, 317.2], %) // [$startCapTag, $EndCapTag] | ||||
|   |> angledLine(angle = 0, length = 268.43, tag = $rectangleSegmentA001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 217.26, tag = $seg01) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001), tag = $yo) | ||||
|   |> angledLine([0, 268.43], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        217.26 | ||||
|      ], %, $seg01) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $yo) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)], tag = $seg02) | ||||
|   |> close() | ||||
| extrude001 = extrude(sketch001, length = 100) | ||||
| @ -311,30 +320,48 @@ extrude001 = extrude(sketch001, length = 100) | ||||
|   |> chamfer(length = 30, tags = [getNextAdjacentEdge(yo)], tag = $seg06) | ||||
| sketch005 = startSketchOn(extrude001, seg06) | ||||
| profile004=startProfileAt([-23.43,19.69], sketch005) | ||||
|   |> angledLine(angle = 0, length = 9.1, tag = $rectangleSegmentA005) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA005) - 90, length = 84.07) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA005), length = -segLen(rectangleSegmentA005)) | ||||
|   |> angledLine([0, 9.1], %, $rectangleSegmentA005) | ||||
|   |> angledLine([segAng(rectangleSegmentA005) - 90, 84.07], %) | ||||
|   |> angledLine([segAng(rectangleSegmentA005), -segLen(rectangleSegmentA005)], %) | ||||
|   |> line(endAbsolute=[profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| sketch004 = startSketchOn(extrude001, seg05) | ||||
| profile003 = startProfileAt([82.57, 322.96], sketch004) | ||||
|   |> angledLine(angle = 0, length = 11.16, tag = $rectangleSegmentA004) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA004) - 90, length = 103.07) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA004), length = -segLen(rectangleSegmentA004)) | ||||
|   |> angledLine([0, 11.16], %, $rectangleSegmentA004) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA004) - 90, | ||||
|        103.07 | ||||
|      ], %) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA004), | ||||
|        -segLen(rectangleSegmentA004) | ||||
|      ], %) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| sketch003 = startSketchOn(extrude001, seg04) | ||||
| profile002 = startProfileAt([-209.64, 255.28], sketch003) | ||||
|   |> angledLine(angle = 0, length = 11.56, tag = $rectangleSegmentA003) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA003) - 90, length = 106.84) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA003), length = -segLen(rectangleSegmentA003)) | ||||
|   |> angledLine([0, 11.56], %, $rectangleSegmentA003) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA003) - 90, | ||||
|        106.84 | ||||
|      ], %) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA003), | ||||
|        -segLen(rectangleSegmentA003) | ||||
|      ], %) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| sketch002 = startSketchOn(extrude001, seg03) | ||||
| profile001 = startProfileAt([205.96, 254.59], sketch002) | ||||
|   |> angledLine(angle = 0, length = 11.39, tag = $rectangleSegmentA002) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA002) - 90, length = 105.26) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA002), length = -segLen(rectangleSegmentA002)) | ||||
|   |> angledLine([0, 11.39], %, $rectangleSegmentA002) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA002) - 90, | ||||
|        105.26 | ||||
|      ], %) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA002), | ||||
|        -segLen(rectangleSegmentA002) | ||||
|      ], %) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| `, | ||||
| @ -350,7 +377,6 @@ profile001 = startProfileAt([205.96, 254.59], sketch002) | ||||
|       editor, | ||||
|       toolbar, | ||||
|       scene, | ||||
|       cmdBar, | ||||
|     }) => { | ||||
|       const file = await fs.readFile( | ||||
|         path.resolve( | ||||
| @ -366,7 +392,7 @@ profile001 = startProfileAt([205.96, 254.59], sketch002) | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|  | ||||
|       const sketchOnAChamfer = _sketchOnAChamfer(page, editor, toolbar, scene) | ||||
|  | ||||
| @ -374,7 +400,7 @@ profile001 = startProfileAt([205.96, 254.59], sketch002) | ||||
|         clickCoords: { x: 570, y: 220 }, | ||||
|         cameraPos: { x: 16020, y: -2000, z: 10500 }, | ||||
|         cameraTarget: { x: -150, y: -4500, z: -80 }, | ||||
|         beforeChamferSnippet: `angledLine(angle=segAng(rectangleSegmentA001)-90,length=217.26,tag=$seg01) | ||||
|         beforeChamferSnippet: `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01) | ||||
|       chamfer(extrude001,length=30,tags=[ | ||||
|       seg01, | ||||
|       getNextAdjacentEdge(yo), | ||||
| @ -386,9 +412,9 @@ profile001 = startProfileAt([205.96, 254.59], sketch002) | ||||
|           'sketch002 = startSketchOn(extrude001, seg03)', | ||||
|         afterRectangle1stClickSnippet: | ||||
|           'startProfileAt([205.96, 254.59], sketch002)', | ||||
|         afterRectangle2ndClickSnippet: `angledLine(angle=0,length=11.39,tag=$rectangleSegmentA002) | ||||
|         |>angledLine(angle=segAng(rectangleSegmentA002)-90,length=105.26) | ||||
|         |>angledLine(angle=segAng(rectangleSegmentA002),length=-segLen(rectangleSegmentA002)) | ||||
|         afterRectangle2ndClickSnippet: `angledLine([0,11.39],%,$rectangleSegmentA002) | ||||
|         |>angledLine([segAng(rectangleSegmentA002)-90,105.26],%) | ||||
|         |>angledLine([segAng(rectangleSegmentA002),-segLen(rectangleSegmentA002)],%) | ||||
|         |>line(endAbsolute=[profileStartX(%),profileStartY(%)]) | ||||
|         |>close()`, | ||||
|       }) | ||||
| @ -396,9 +422,15 @@ profile001 = startProfileAt([205.96, 254.59], sketch002) | ||||
|         `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([75.8, 317.2], %) | ||||
|   |> angledLine(angle = 0, length = 268.43, tag = $rectangleSegmentA001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 217.26, tag = $seg01) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001), tag = $yo) | ||||
|   |> angledLine([0, 268.43], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        217.26 | ||||
|      ], %, $seg01) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $yo) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)], tag = $seg02) | ||||
|   |> close() | ||||
| extrude001 = extrude(sketch001, length = 100) | ||||
| @ -418,9 +450,15 @@ chamf = chamfer( | ||||
|      ) | ||||
| sketch002 = startSketchOn(extrude001, seg03) | ||||
| profile001 = startProfileAt([205.96, 254.59], sketch002) | ||||
|   |> angledLine(angle = 0, length = 11.39, tag = $rectangleSegmentA002) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA002) - 90, length = 105.26) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA002), length = -segLen(rectangleSegmentA002)) | ||||
|   |> angledLine([0, 11.39], %, $rectangleSegmentA002) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA002) - 90, | ||||
|        105.26 | ||||
|      ], %) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA002), | ||||
|        -segLen(rectangleSegmentA002) | ||||
|      ], %) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| `, | ||||
| @ -441,7 +479,6 @@ profile001 = startProfileAt([205.96, 254.59], sketch002) | ||||
|     await page.setBodyDimensions(viewPortSize) | ||||
|  | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.connectionEstablished() | ||||
|  | ||||
|     // Constants and locators | ||||
|     // These are mappings from screenspace to KCL coordinates, | ||||
| @ -500,7 +537,8 @@ profile001 = startProfileAt([205.96, 254.59], sketch002) | ||||
|       await toolbar.startSketchPlaneSelection() | ||||
|       await moveToXzPlane() | ||||
|       await clickOnXzPlane() | ||||
|       await toolbar.waitUntilSketchingReady() | ||||
|       // timeout wait for engine animation is unavoidable | ||||
|       await page.waitForTimeout(600) | ||||
|       await editor.expectEditor.toContain(expectedCodeSnippets.sketchOnXzPlane) | ||||
|     }) | ||||
|     await test.step(`Place a point a few pixels off the middle, verify it still snaps to 0,0`, async () => { | ||||
| @ -542,8 +580,9 @@ profile001 = startProfileAt([205.96, 254.59], sketch002) | ||||
|     editor, | ||||
|     toolbar, | ||||
|     scene, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     const initialCode = `closedSketch = startSketchOn(XZ) | ||||
|   |> circle(center = [8, 5], radius = 2) | ||||
| openSketch = startSketchOn(XY) | ||||
| @ -560,6 +599,8 @@ openSketch = startSketchOn(XY) | ||||
|     }, initialCode) | ||||
|  | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|     await page.waitForTimeout(1000) | ||||
|  | ||||
|     const pointInsideCircle = { | ||||
|       x: viewPortSize.width * 0.63, | ||||
| @ -584,16 +625,15 @@ openSketch = startSketchOn(XY) | ||||
|     const exitSketch = async () => { | ||||
|       await test.step(`Exit sketch mode`, async () => { | ||||
|         await toolbar.exitSketchBtn.click() | ||||
|         await expect(toolbar.exitSketchBtn).not.toBeVisible() | ||||
|         await expect(toolbar.startSketchBtn).toBeEnabled() | ||||
|       }) | ||||
|     } | ||||
|  | ||||
|     await test.step(`Double-click on the closed sketch`, async () => { | ||||
|       await scene.settled(cmdBar) | ||||
|       await moveToCircle() | ||||
|       await page.waitForTimeout(1000) | ||||
|       await dblClickCircle() | ||||
|       await page.waitForTimeout(1000) | ||||
|       await expect(toolbar.startSketchBtn).not.toBeVisible() | ||||
|       await expect(toolbar.exitSketchBtn).toBeVisible() | ||||
|       await editor.expectState({ | ||||
|         activeLines: [`|>circle(center=[8,5],radius=2)`], | ||||
| @ -630,6 +670,7 @@ openSketch = startSketchOn(XY) | ||||
|       // There is a full execution after exiting sketch that clears the scene. | ||||
|       await page.waitForTimeout(500) | ||||
|       await dblClickOpenPath() | ||||
|       await expect(toolbar.startSketchBtn).not.toBeVisible() | ||||
|       await expect(toolbar.exitSketchBtn).toBeVisible() | ||||
|       // Wait for enter sketch mode to complete | ||||
|       await page.waitForTimeout(500) | ||||
| @ -990,9 +1031,6 @@ openSketch = startSketchOn(XY) | ||||
|     }) | ||||
|     await test.step(`Go through the command bar flow`, async () => { | ||||
|       await toolbar.offsetPlaneButton.click() | ||||
|       await expect | ||||
|         .poll(() => page.getByText('Please select one').count()) | ||||
|         .toBe(1) | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'arguments', | ||||
|         currentArgKey: 'plane', | ||||
| @ -1050,7 +1088,6 @@ openSketch = startSketchOn(XY) | ||||
|     const expectedLine = `axis=X,` | ||||
|  | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.connectionEstablished() | ||||
|  | ||||
|     await test.step(`Go through the command bar flow`, async () => { | ||||
|       await toolbar.helixButton.click() | ||||
| @ -1069,7 +1106,6 @@ openSketch = startSketchOn(XY) | ||||
|         commandName: 'Helix', | ||||
|       }) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await expect.poll(() => page.getByText('Axis').count()).toBe(6) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
| @ -1197,7 +1233,6 @@ openSketch = startSketchOn(XY) | ||||
|         }, initialCode) | ||||
|         await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|         await homePage.goToModelingScene() | ||||
|         await scene.settled(cmdBar) | ||||
|  | ||||
|         await test.step(`Go through the command bar flow`, async () => { | ||||
|           await toolbar.closePane('code') | ||||
| @ -1217,22 +1252,15 @@ openSketch = startSketchOn(XY) | ||||
|             commandName: 'Helix', | ||||
|           }) | ||||
|           await cmdBar.selectOption({ name: 'Edge' }).click() | ||||
|           await expect | ||||
|             .poll(() => page.getByText('Please select one').count()) | ||||
|             .toBe(1) | ||||
|           await clickOnEdge() | ||||
|           await page.waitForTimeout(1000) | ||||
|           await cmdBar.progressCmdBar() | ||||
|           await page.waitForTimeout(1000) | ||||
|           await cmdBar.argumentInput.focus() | ||||
|           await page.waitForTimeout(1000) | ||||
|           await page.keyboard.insertText('20') | ||||
|           await cmdBar.progressCmdBar() | ||||
|           await page.keyboard.insertText('0') | ||||
|           await cmdBar.progressCmdBar() | ||||
|           await page.keyboard.insertText('1') | ||||
|           await cmdBar.progressCmdBar() | ||||
|           await page.keyboard.insertText('100') | ||||
|           await cmdBar.expectState({ | ||||
|             stage: 'review', | ||||
|             headerArguments: { | ||||
| @ -1246,7 +1274,6 @@ openSketch = startSketchOn(XY) | ||||
|             commandName: 'Helix', | ||||
|           }) | ||||
|           await cmdBar.progressCmdBar() | ||||
|           await page.waitForTimeout(1000) | ||||
|         }) | ||||
|  | ||||
|         await test.step(`Confirm code is added to the editor, scene has changed`, async () => { | ||||
| @ -1342,7 +1369,7 @@ extrude001 = extrude(profile001, length = 100) | ||||
|     }, initialCode) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // One dumb hardcoded screen pixel value | ||||
|     const testPoint = { x: 620, y: 257 } | ||||
| @ -1503,9 +1530,6 @@ extrude001 = extrude(profile001, length = 100) | ||||
|       if (!shouldPreselect) { | ||||
|         await test.step(`Go through the command bar flow without preselected sketches`, async () => { | ||||
|           await toolbar.loftButton.click() | ||||
|           await expect | ||||
|             .poll(() => page.getByText('Please select one').count()) | ||||
|             .toBe(1) | ||||
|           await cmdBar.expectState({ | ||||
|             stage: 'arguments', | ||||
|             currentArgKey: 'selection', | ||||
| @ -1555,7 +1579,6 @@ extrude001 = extrude(profile001, length = 100) | ||||
|     page, | ||||
|     homePage, | ||||
|     scene, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const initialCode = `sketch001 = startSketchOn(XZ) | ||||
|   |> circle(center = [0, 0], radius = 30) | ||||
| @ -1569,7 +1592,7 @@ loft001 = loft([sketch001, sketch002]) | ||||
|     }, initialCode) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // One dumb hardcoded screen pixel value | ||||
|     const testPoint = { x: 575, y: 200 } | ||||
| @ -1632,9 +1655,15 @@ sketch002 = startSketchOn(XZ) | ||||
|       initialCode: `@settings(defaultLengthUnit = in) | ||||
| sketch001 = startSketchOn(YZ) | ||||
| profile001 = startProfileAt([-400, -400], sketch001) | ||||
|   |> angledLine(angle = 0, length = 800, tag = $rectangleSegmentA001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001) + 90, length = 800) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001)) | ||||
|   |> angledLine([0, 800], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) + 90, | ||||
|        800 | ||||
|      ], %) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| sketch002 = startSketchOn(XZ) | ||||
| @ -1658,7 +1687,7 @@ sketch002 = startSketchOn(XZ) | ||||
|       }, initialCode) | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|  | ||||
|       // One dumb hardcoded screen pixel value | ||||
|       const [clickOnSketch1] = scene.makeMouseHelpers(testPoint.x, testPoint.y) | ||||
| @ -1678,9 +1707,6 @@ sketch002 = startSketchOn(XZ) | ||||
|  | ||||
|       await test.step(`Go through the command bar flow`, async () => { | ||||
|         await toolbar.sweepButton.click() | ||||
|         await expect | ||||
|           .poll(() => page.getByText('Please select one').count()) | ||||
|           .toBe(1) | ||||
|         await cmdBar.expectState({ | ||||
|           commandName: 'Sweep', | ||||
|           currentArgKey: 'target', | ||||
| @ -1800,7 +1826,7 @@ sketch002 = startSketchOn(XZ) | ||||
|     }, initialCode) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // One dumb hardcoded screen pixel value | ||||
|     const testPoint = { x: 700, y: 250 } | ||||
| @ -1817,9 +1843,6 @@ sketch002 = startSketchOn(XZ) | ||||
|  | ||||
|     await test.step(`Go through the command bar flow and fail validation with a toast`, async () => { | ||||
|       await toolbar.sweepButton.click() | ||||
|       await expect | ||||
|         .poll(() => page.getByText('Please select one').count()) | ||||
|         .toBe(1) | ||||
|       await cmdBar.expectState({ | ||||
|         commandName: 'Sweep', | ||||
|         currentArgKey: 'target', | ||||
| @ -2036,9 +2059,6 @@ extrude001 = extrude(sketch001, length = -12) | ||||
|     await test.step(`Open fillet UI without selecting edges`, async () => { | ||||
|       await page.waitForTimeout(100) | ||||
|       await toolbar.filletButton.click() | ||||
|       await expect | ||||
|         .poll(() => page.getByText('Please select one').count()) | ||||
|         .toBe(1) | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'arguments', | ||||
|         currentArgKey: 'selection', | ||||
| @ -2164,7 +2184,6 @@ extrude001 = extrude(sketch001, length = -12) | ||||
|     homePage, | ||||
|     scene, | ||||
|     toolbar, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const initialCode = `sketch001 = startSketchOn(XY) | ||||
| profile001 = circle( | ||||
| @ -2181,7 +2200,7 @@ fillet001 = fillet(extrude001, radius = 5, tags = [getOppositeEdge(seg01)]) | ||||
|     }, initialCode) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     await test.step('Double-click in feature tree and expect error toast', async () => { | ||||
|       await toolbar.openPane('feature-tree') | ||||
| @ -2502,7 +2521,7 @@ extrude001 = extrude(sketch001, length = -12) | ||||
|       }, initialCode) | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|     }) | ||||
|  | ||||
|     // Test 1: Command bar flow with preselected edges | ||||
| @ -2535,7 +2554,6 @@ extrude001 = extrude(sketch001, length = -12) | ||||
|         stage: 'arguments', | ||||
|       }) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await page.waitForTimeout(1000) | ||||
|       await cmdBar.expectState({ | ||||
|         commandName: 'Chamfer', | ||||
|         highlightedHeaderArg: 'length', | ||||
| @ -2547,10 +2565,7 @@ extrude001 = extrude(sketch001, length = -12) | ||||
|         }, | ||||
|         stage: 'arguments', | ||||
|       }) | ||||
|       await cmdBar.argumentInput.focus() | ||||
|       await page.waitForTimeout(1000) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await page.waitForTimeout(1000) | ||||
|       await cmdBar.expectState({ | ||||
|         commandName: 'Chamfer', | ||||
|         headerArguments: { | ||||
| @ -2634,9 +2649,6 @@ extrude001 = extrude(sketch001, length = -12) | ||||
|     await test.step(`Open chamfer UI without selecting edges`, async () => { | ||||
|       await page.waitForTimeout(100) | ||||
|       await toolbar.chamferButton.click() | ||||
|       await expect | ||||
|         .poll(() => page.getByText('Please select one').count()) | ||||
|         .toBe(1) | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'arguments', | ||||
|         currentArgKey: 'selection', | ||||
| @ -2759,7 +2771,6 @@ extrude001 = extrude(sketch001, length = -12) | ||||
|     scene, | ||||
|     editor, | ||||
|     toolbar, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     // Code samples | ||||
|     const initialCode = `@settings(defaultLengthUnit = in) | ||||
| @ -2803,7 +2814,7 @@ chamfer04 = chamfer(extrude001, length = 5, tags = [getOppositeEdge(seg02)]) | ||||
|       }, initialCode) | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|  | ||||
|       // verify modeling scene is loaded | ||||
|       await scene.expectPixelColor( | ||||
| @ -2925,11 +2936,9 @@ extrude001 = extrude(sketch001, length = 30) | ||||
|       await context.addInitScript((initialCode) => { | ||||
|         localStorage.setItem('persistCode', initialCode) | ||||
|       }, initialCode) | ||||
|  | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.connectionEstablished() | ||||
|       await scene.waitForExecutionDone() | ||||
|  | ||||
|       // One dumb hardcoded screen pixel value | ||||
|       const testPoint = { x: 575, y: 200 } | ||||
| @ -2946,9 +2955,6 @@ extrude001 = extrude(sketch001, length = 30) | ||||
|       if (!shouldPreselect) { | ||||
|         await test.step(`Go through the command bar flow without preselected faces`, async () => { | ||||
|           await toolbar.shellButton.click() | ||||
|           await expect | ||||
|             .poll(() => page.getByText('Please select one').count()) | ||||
|             .toBe(1) | ||||
|           await cmdBar.expectState({ | ||||
|             stage: 'arguments', | ||||
|             currentArgKey: 'selection', | ||||
| @ -3009,6 +3015,7 @@ extrude001 = extrude(sketch001, length = 30) | ||||
|       }) | ||||
|  | ||||
|       await test.step('Edit shell via feature tree selection works', async () => { | ||||
|         await toolbar.closePane('code') | ||||
|         await toolbar.openPane('feature-tree') | ||||
|         const operationButton = await toolbar.getFeatureTreeOperation( | ||||
|           'Shell', | ||||
| @ -3037,6 +3044,7 @@ extrude001 = extrude(sketch001, length = 30) | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await toolbar.closePane('feature-tree') | ||||
|         await scene.expectPixelColor([150, 150, 150], testPoint, 15) | ||||
|         await toolbar.openPane('code') | ||||
|         await editor.expectEditor.toContain(editedShellDeclaration) | ||||
|         await editor.expectState({ | ||||
|           diagnostics: [], | ||||
| @ -3071,7 +3079,7 @@ extrude001 = extrude(sketch001, length = 40) | ||||
|     }, initialCode) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // One dumb hardcoded screen pixel value | ||||
|     const testPoint = { x: 580, y: 180 } | ||||
| @ -3089,9 +3097,6 @@ extrude001 = extrude(sketch001, length = 40) | ||||
|  | ||||
|     await test.step(`Go through the command bar flow, selecting a wall and keeping default thickness`, async () => { | ||||
|       await toolbar.shellButton.click() | ||||
|       await expect | ||||
|         .poll(() => page.getByText('Please select one').count()) | ||||
|         .toBe(1) | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'arguments', | ||||
|         currentArgKey: 'selection', | ||||
| @ -3103,9 +3108,6 @@ extrude001 = extrude(sketch001, length = 40) | ||||
|         highlightedHeaderArg: 'selection', | ||||
|         commandName: 'Shell', | ||||
|       }) | ||||
|       await expect | ||||
|         .poll(() => page.getByText('Please select one').count()) | ||||
|         .toBe(1) | ||||
|       await clickOnCap() | ||||
|       await page.keyboard.down('Shift') | ||||
|       await clickOnWall() | ||||
| @ -3114,7 +3116,6 @@ extrude001 = extrude(sketch001, length = 40) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await page.waitForTimeout(500) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await page.waitForTimeout(500) | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'review', | ||||
|         headerArguments: { | ||||
| @ -3123,9 +3124,7 @@ extrude001 = extrude(sketch001, length = 40) | ||||
|         }, | ||||
|         commandName: 'Shell', | ||||
|       }) | ||||
|       await page.waitForTimeout(500) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await page.waitForTimeout(500) | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Confirm code is added to the editor, scene has changed`, async () => { | ||||
| @ -3140,6 +3139,7 @@ extrude001 = extrude(sketch001, length = 40) | ||||
|     }) | ||||
|  | ||||
|     await test.step('Edit shell via feature tree selection works', async () => { | ||||
|       await editor.closePane() | ||||
|       const operationButton = await toolbar.getFeatureTreeOperation('Shell', 0) | ||||
|       await operationButton.dblclick({ button: 'left' }) | ||||
|       await cmdBar.expectState({ | ||||
| @ -3154,7 +3154,6 @@ extrude001 = extrude(sketch001, length = 40) | ||||
|       }) | ||||
|       await page.keyboard.insertText('1') | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await page.waitForTimeout(500) | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'review', | ||||
|         headerArguments: { | ||||
| @ -3165,6 +3164,7 @@ extrude001 = extrude(sketch001, length = 40) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await toolbar.closePane('feature-tree') | ||||
|       await scene.expectPixelColor([150, 150, 150], testPoint, 15) | ||||
|       await toolbar.openPane('code') | ||||
|       await editor.expectEditor.toContain(editedShellDeclaration) | ||||
|       await editor.expectState({ | ||||
|         diagnostics: [], | ||||
| @ -3218,7 +3218,7 @@ extrude002 = extrude(sketch002, length = 50) | ||||
|       }, initialCode) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|  | ||||
|       // One dumb hardcoded screen pixel value | ||||
|       const testPoint = { x: 580, y: 320 } | ||||
| @ -3243,13 +3243,12 @@ extrude002 = extrude(sketch002, length = 50) | ||||
|           highlightedHeaderArg: 'selection', | ||||
|           commandName: 'Shell', | ||||
|         }) | ||||
|         await expect | ||||
|           .poll(() => page.getByText('Please select one').count()) | ||||
|           .toBe(1) | ||||
|         await clickOnCap() | ||||
|         await page.waitForTimeout(1000) | ||||
|         await page.waitForTimeout(500) | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await page.waitForTimeout(500) | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await page.waitForTimeout(500) | ||||
|         await cmdBar.expectState({ | ||||
|           stage: 'review', | ||||
|           headerArguments: { | ||||
| @ -3307,7 +3306,7 @@ profile001 = startProfileAt([-20, 20], sketch001) | ||||
|       }, initialCode) | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|       await toolbar.openPane('feature-tree') | ||||
|  | ||||
|       // One dumb hardcoded screen pixel value | ||||
| @ -3387,7 +3386,7 @@ sweep001 = sweep(sketch001, path = sketch002) | ||||
|     }, initialCode) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // One dumb hardcoded screen pixel value | ||||
|     const testPoint = { x: 500, y: 250 } | ||||
| @ -3400,9 +3399,6 @@ sweep001 = sweep(sketch001, path = sketch002) | ||||
|  | ||||
|     await test.step(`Go through the Shell flow and fail validation with a toast`, async () => { | ||||
|       await toolbar.shellButton.click() | ||||
|       await expect | ||||
|         .poll(() => page.getByText('Please select one').count()) | ||||
|         .toBe(1) | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'arguments', | ||||
|         currentArgKey: 'selection', | ||||
| @ -3437,29 +3433,26 @@ sweep001 = sweep(sketch001, path = sketch002) | ||||
|       const initialCode = ` | ||||
| sketch001 = startSketchOn(XZ) | ||||
| |> startProfileAt([-100.0, 100.0], %) | ||||
| |> angledLine(angle = 0, length = 200.0, tag = $rectangleSegmentA001) | ||||
| |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 200, tag = $rectangleSegmentB001) | ||||
| |> angledLine( | ||||
| angle=segAng(rectangleSegmentA001), | ||||
| length=-segLen(rectangleSegmentA001), | ||||
| tag=$rectangleSegmentC001, | ||||
| ) | ||||
| |> angledLine([0, 200.0], %, $rectangleSegmentA001) | ||||
| |> angledLine([segAng(rectangleSegmentA001) - 90, 200], %, $rectangleSegmentB001) | ||||
| |> angledLine([ | ||||
| segAng(rectangleSegmentA001), | ||||
| -segLen(rectangleSegmentA001) | ||||
| ], %, $rectangleSegmentC001) | ||||
| |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
| |> close() | ||||
| extrude001 = extrude(sketch001, length = 200) | ||||
| sketch002 = startSketchOn(extrude001, rectangleSegmentA001) | ||||
| |> startProfileAt([-66.77, 84.81], %) | ||||
| |> angledLine(angle = 180, length = 27.08, tag = $rectangleSegmentA002) | ||||
| |> angledLine( | ||||
| angle=segAng(rectangleSegmentA002) - 90, | ||||
| length=27.8, | ||||
| tag=$rectangleSegmentB002, | ||||
| ) | ||||
| |> angledLine( | ||||
| angle=segAng(rectangleSegmentA002), | ||||
| length=-segLen(rectangleSegmentA002), | ||||
| tag=$rectangleSegmentC002, | ||||
| ) | ||||
| |> angledLine([180, 27.08], %, $rectangleSegmentA002) | ||||
| |> angledLine([ | ||||
| segAng(rectangleSegmentA002) - 90, | ||||
| 27.8 | ||||
| ], %, $rectangleSegmentB002) | ||||
| |> angledLine([ | ||||
| segAng(rectangleSegmentA002), | ||||
| -segLen(rectangleSegmentA002) | ||||
| ], %, $rectangleSegmentC002) | ||||
| |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
| |> close() | ||||
| ` | ||||
| @ -3469,13 +3462,12 @@ tag=$rectangleSegmentC002, | ||||
|       }, initialCode) | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|  | ||||
|       // select line of code | ||||
|       const codeToSelection = `segAng(rectangleSegmentA002) - 90,` | ||||
|       const codeToSelecton = `segAng(rectangleSegmentA002) - 90,` | ||||
|       // revolve | ||||
|       await editor.scrollToText(codeToSelection) | ||||
|       await page.getByText(codeToSelection).click() | ||||
|       await page.getByText(codeToSelecton).click() | ||||
|       await toolbar.revolveButton.click() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
| @ -3529,32 +3521,35 @@ tag=$rectangleSegmentC002, | ||||
|     }) => { | ||||
|       const initialCode = `sketch001 = startSketchOn(XZ) | ||||
|   |> startProfileAt([-102.57, 101.72], %) | ||||
|   |> angledLine(angle = 0, length = 202.6, tag = $rectangleSegmentA001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 202.6, tag = $rectangleSegmentB001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001), tag = $rectangleSegmentC001) | ||||
|   |> angledLine([0, 202.6], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        202.6 | ||||
|      ], %, $rectangleSegmentB001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $rectangleSegmentC001) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| extrude001 = extrude(sketch001, length = 50) | ||||
| sketch002 = startSketchOn(extrude001, rectangleSegmentA001) | ||||
|   |> circle(center = [-11.34, 10.0], radius = 8.69) | ||||
|  | ||||
| ` | ||||
|       await context.addInitScript((initialCode) => { | ||||
|         localStorage.setItem('persistCode', initialCode) | ||||
|       }, initialCode) | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.connectionEstablished() | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|  | ||||
|       // select line of code | ||||
|       const codeToSelection = `center = [-11.34, 10.0]` | ||||
|       const codeToSelecton = `center = [-11.34, 10.0]` | ||||
|       // revolve | ||||
|       await editor.scrollToText(codeToSelection) | ||||
|       await page.getByText(codeToSelection).click() | ||||
|       await page.getByText(codeToSelecton).click() | ||||
|       await toolbar.revolveButton.click() | ||||
|       await page.getByText('Edge', { exact: true }).click() | ||||
|       const lineCodeToSelection = `angledLine(angle = 0, length = 202.6, tag = $rectangleSegmentA001)` | ||||
|       const lineCodeToSelection = `|> angledLine([0, 202.6], %, $rectangleSegmentA001)` | ||||
|       await page.getByText(lineCodeToSelection).click() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
| @ -3600,7 +3595,6 @@ sketch002 = startSketchOn(extrude001, rectangleSegmentA001) | ||||
|       await editor.expectEditor.toContain( | ||||
|         newCodeToFind.replace('angle = 360', 'angle = angle001') | ||||
|       ) | ||||
|       expect(editor.expectEditor.toContain(newCodeToFind)).toBeTruthy() | ||||
|     }) | ||||
|     test('revolve sketch circle around line segment from startProfileAt sketch', async ({ | ||||
|       context, | ||||
| @ -3611,23 +3605,22 @@ sketch002 = startSketchOn(extrude001, rectangleSegmentA001) | ||||
|       toolbar, | ||||
|       cmdBar, | ||||
|     }) => { | ||||
|       const initialCode = ` | ||||
|     sketch002 = startSketchOn(XY) | ||||
|       |> startProfileAt([-2.02, 1.79], %) | ||||
|       |> xLine(length = 2.6) | ||||
|     sketch001 = startSketchOn('-XY') | ||||
|       |> startProfileAt([-0.48, 1.25], %) | ||||
|       |> angledLine(angle = 0, length = 2.38, tag = $rectangleSegmentA001) | ||||
|       |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 2.4, tag = $rectangleSegmentB001) | ||||
|       |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001), tag = $rectangleSegmentC001) | ||||
|       |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|       |> close() | ||||
|     extrude001 = extrude(sketch001, length = 5) | ||||
|     sketch003 = startSketchOn(extrude001, 'START') | ||||
|       |> circle( | ||||
|         center = [-0.69, 0.56], | ||||
|         radius = 0.28 | ||||
|       ) | ||||
|       const initialCode = `sketch002 = startSketchOn(XY) | ||||
|   |> startProfileAt([-2.02, 1.79], %) | ||||
|   |> xLine(length = 2.6) | ||||
| sketch001 = startSketchOn(-XY) | ||||
|   |> startProfileAt([-0.48, 1.25], %) | ||||
|   |> angledLine([0, 2.38], %, $rectangleSegmentA001) | ||||
|   |> angledLine([segAng(rectangleSegmentA001) - 90, 2.4], %, $rectangleSegmentB001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $rectangleSegmentC001) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| extrude001 = extrude(sketch001, length = 5) | ||||
| sketch003 = startSketchOn(extrude001, 'START') | ||||
|   |> circle(center = [-0.69, 0.56], radius = 0.28) | ||||
| ` | ||||
|  | ||||
|       await context.addInitScript((initialCode) => { | ||||
| @ -3635,20 +3628,15 @@ sketch002 = startSketchOn(extrude001, rectangleSegmentA001) | ||||
|       }, initialCode) | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.connectionEstablished() | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|  | ||||
|       // select line of code | ||||
|       const codeToSelection = `center = [-0.69, 0.56]` | ||||
|       const codeToSelecton = `center = [-0.69, 0.56]` | ||||
|       // revolve | ||||
|       await page.getByText(codeToSelecton).click() | ||||
|       await toolbar.revolveButton.click() | ||||
|       await page.waitForTimeout(1000) | ||||
|       await editor.scrollToText(codeToSelection) | ||||
|       await page.getByText(codeToSelection).click() | ||||
|       await expect.poll(() => page.getByText('AxisOrEdge').count()).toBe(2) | ||||
|       await page.getByText('Edge', { exact: true }).click() | ||||
|       const lineCodeToSelection = `length = 2.6` | ||||
|       await editor.scrollToText(lineCodeToSelection) | ||||
|       const lineCodeToSelection = `|> xLine(length = 2.6)` | ||||
|       await page.getByText(lineCodeToSelection).click() | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await cmdBar.progressCmdBar() | ||||
| @ -3715,22 +3703,21 @@ extrude001 = extrude(profile001, length = 100) | ||||
|     }, initialCode) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     // One dumb hardcoded screen pixel value | ||||
|     const testPoint = { x: 500, y: 250 } | ||||
|     const initialColor: [number, number, number] = [123, 123, 123] | ||||
|     const tolerance = 50 | ||||
|  | ||||
|     await test.step(`Confirm extrude exists with default appearance`, async () => { | ||||
|       await toolbar.closePane('code') | ||||
|       await scene.expectPixelColor(initialColor, testPoint, tolerance) | ||||
|       await scene.expectPixelColor(initialColor, testPoint, 15) | ||||
|     }) | ||||
|  | ||||
|     async function setApperanceAndCheck( | ||||
|       option: string, | ||||
|       hex: string, | ||||
|       shapeColor?: [number, number, number] | ||||
|       shapeColor: [number, number, number] | ||||
|     ) { | ||||
|       await toolbar.openPane('feature-tree') | ||||
|       const enterAppearanceFlow = async (stepName: string) => | ||||
| @ -3779,9 +3766,7 @@ extrude001 = extrude(profile001, length = 100) | ||||
|       }) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await toolbar.closePane('feature-tree') | ||||
|       if (shapeColor) { | ||||
|         await scene.expectPixelColor(shapeColor, testPoint, tolerance) | ||||
|       } | ||||
|       await scene.expectPixelColor(shapeColor, testPoint, 10) | ||||
|       await toolbar.openPane('code') | ||||
|       if (hex === 'default') { | ||||
|         const anyAppearanceDeclaration = `|> appearance(` | ||||
| @ -3800,17 +3785,16 @@ extrude001 = extrude(profile001, length = 100) | ||||
|     } | ||||
|  | ||||
|     await test.step(`Go through the Set Appearance flow for all options`, async () => { | ||||
|       await setApperanceAndCheck('Red', '#FF0000', [180, 30, 30]) | ||||
|       // Not checking the scene color every time cause that's not really deterministic. Red seems reliable though | ||||
|       await setApperanceAndCheck('Green', '#00FF00') | ||||
|       await setApperanceAndCheck('Blue', '#0000FF') | ||||
|       await setApperanceAndCheck('Turquoise', '#00FFFF') | ||||
|       await setApperanceAndCheck('Purple', '#FF00FF') | ||||
|       await setApperanceAndCheck('Yellow', '#FFFF00') | ||||
|       await setApperanceAndCheck('Black', '#000000') | ||||
|       await setApperanceAndCheck('Dark Grey', '#080808') | ||||
|       await setApperanceAndCheck('Light Grey', '#D3D3D3') | ||||
|       await setApperanceAndCheck('White', '#FFFFFF') | ||||
|       await setApperanceAndCheck('Red', '#FF0000', [180, 0, 0]) | ||||
|       await setApperanceAndCheck('Green', '#00FF00', [0, 180, 0]) | ||||
|       await setApperanceAndCheck('Blue', '#0000FF', [0, 0, 180]) | ||||
|       await setApperanceAndCheck('Turquoise', '#00FFFF', [0, 180, 180]) | ||||
|       await setApperanceAndCheck('Purple', '#FF00FF', [180, 0, 180]) | ||||
|       await setApperanceAndCheck('Yellow', '#FFFF00', [180, 180, 0]) | ||||
|       await setApperanceAndCheck('Black', '#000000', [0, 0, 0]) | ||||
|       await setApperanceAndCheck('Dark Grey', '#080808', [0x33, 0x33, 0x33]) | ||||
|       await setApperanceAndCheck('Light Grey', '#D3D3D3', [176, 176, 176]) | ||||
|       await setApperanceAndCheck('White', '#FFFFFF', [184, 184, 184]) | ||||
|       await setApperanceAndCheck( | ||||
|         'Default (clear appearance)', | ||||
|         'default', | ||||
|  | ||||
| @ -83,7 +83,7 @@ test( | ||||
| test( | ||||
|   'click help/keybindings from project page', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ scene, cmdBar, context, page }, testInfo) => { | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const bracketDir = path.join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
| @ -95,11 +95,17 @@ test( | ||||
|  | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     page.on('console', console.log) | ||||
|  | ||||
|     // expect to see the text bracket | ||||
|     await expect(page.getByText('bracket')).toBeVisible() | ||||
|  | ||||
|     await page.getByText('bracket').click() | ||||
|  | ||||
|     await scene.settled(cmdBar) | ||||
|     await expect(page.getByTestId('loading')).toBeAttached() | ||||
|     await expect(page.getByTestId('loading')).not.toBeAttached({ | ||||
|       timeout: 20_000, | ||||
|     }) | ||||
|  | ||||
|     // click ? button | ||||
|     await page.getByTestId('help-button').click() | ||||
| @ -114,7 +120,7 @@ test( | ||||
| test( | ||||
|   'open a file in a project works and renders, open another file in different project with errors, it should clear the scene', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ scene, cmdBar, context, page, editor }, testInfo) => { | ||||
|   async ({ context, page, editor }, testInfo) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const bracketDir = path.join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
| @ -143,7 +149,24 @@ test( | ||||
|  | ||||
|       await page.getByText('bracket').click() | ||||
|  | ||||
|       await scene.settled(cmdBar) | ||||
|       await expect(page.getByTestId('loading')).toBeAttached() | ||||
|       await expect(page.getByTestId('loading')).not.toBeAttached({ | ||||
|         timeout: 20_000, | ||||
|       }) | ||||
|  | ||||
|       await expect( | ||||
|         page.getByRole('button', { name: 'Start Sketch' }) | ||||
|       ).toBeEnabled({ | ||||
|         timeout: 20_000, | ||||
|       }) | ||||
|  | ||||
|       // gray at this pixel means the stream has loaded in the most | ||||
|       // user way we can verify it (pixel color) | ||||
|       await expect | ||||
|         .poll(() => u.getGreatestPixDiff(pointOnModel, [110, 110, 110]), { | ||||
|           timeout: 10_000, | ||||
|         }) | ||||
|         .toBeLessThan(20) | ||||
|     }) | ||||
|  | ||||
|     await test.step('Clicking the logo takes us back to the projects page / home', async () => { | ||||
| @ -186,7 +209,7 @@ test( | ||||
| test( | ||||
|   'open a file in a project works and renders, open another file in different project that is empty, it should clear the scene', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ scene, cmdBar, context, page }, testInfo) => { | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const bracketDir = path.join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
| @ -212,7 +235,24 @@ test( | ||||
|  | ||||
|       await page.getByText('bracket').click() | ||||
|  | ||||
|       await scene.settled(cmdBar) | ||||
|       await expect(page.getByTestId('loading')).toBeAttached() | ||||
|       await expect(page.getByTestId('loading')).not.toBeAttached({ | ||||
|         timeout: 20_000, | ||||
|       }) | ||||
|  | ||||
|       await expect( | ||||
|         page.getByRole('button', { name: 'Start Sketch' }) | ||||
|       ).toBeEnabled({ | ||||
|         timeout: 20_000, | ||||
|       }) | ||||
|  | ||||
|       // gray at this pixel means the stream has loaded in the most | ||||
|       // user way we can verify it (pixel color) | ||||
|       await expect | ||||
|         .poll(() => u.getGreatestPixDiff(pointOnModel, [125, 125, 125]), { | ||||
|           timeout: 10_000, | ||||
|         }) | ||||
|         .toBeLessThan(15) | ||||
|     }) | ||||
|  | ||||
|     await test.step('Clicking the logo takes us back to the projects page / home', async () => { | ||||
| @ -312,7 +352,7 @@ test( | ||||
| test( | ||||
|   'open a file in a project works and renders, open another file in the same project with errors, it should clear the scene', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ scene, cmdBar, context, page }, testInfo) => { | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     if (runningOnWindows()) { | ||||
|       test.fixme(orRunWhenFullSuiteEnabled()) | ||||
|     } | ||||
| @ -340,7 +380,10 @@ test( | ||||
|  | ||||
|       await page.getByText('bracket').click() | ||||
|  | ||||
|       await scene.settled(cmdBar) | ||||
|       await expect(page.getByTestId('loading')).toBeAttached() | ||||
|       await expect(page.getByTestId('loading')).not.toBeAttached({ | ||||
|         timeout: 20_000, | ||||
|       }) | ||||
|  | ||||
|       await expect( | ||||
|         page.getByRole('button', { name: 'Start Sketch' }) | ||||
| @ -400,10 +443,10 @@ test( | ||||
|     await expect(page.getByText('broken-code')).toBeVisible() | ||||
|     await page.getByText('broken-code').click() | ||||
|  | ||||
|     // Gotcha: You can not use scene.settled() since the KCL code is going to fail | ||||
|     await expect( | ||||
|       page.getByTestId('model-state-indicator-playing') | ||||
|     ).toBeAttached() | ||||
|     // Gotcha: You can not use scene.waitForExecutionDone() since the KCL code is going to fail | ||||
|     await expect(page.getByTestId('loading')).not.toBeAttached({ | ||||
|       timeout: 20_000, | ||||
|     }) | ||||
|  | ||||
|     // Gotcha: Scroll to the text content in code mirror because CodeMirror lazy loads DOM content | ||||
|     await editor.scrollToText( | ||||
| @ -426,7 +469,7 @@ test.describe('Can export from electron app', () => { | ||||
|     test( | ||||
|       `Can export using ${method}`, | ||||
|       { tag: ['@electron', '@skipLocalEngine'] }, | ||||
|       async ({ scene, cmdBar, context, page, tronApp }, testInfo) => { | ||||
|       async ({ context, page, tronApp }, testInfo) => { | ||||
|         if (!tronApp) { | ||||
|           fail() | ||||
|         } | ||||
| @ -456,7 +499,10 @@ test.describe('Can export from electron app', () => { | ||||
|  | ||||
|           await page.getByText('bracket').click() | ||||
|  | ||||
|           await scene.settled(cmdBar) | ||||
|           await expect(page.getByTestId('loading')).toBeAttached() | ||||
|           await expect(page.getByTestId('loading')).not.toBeAttached({ | ||||
|             timeout: 20_000, | ||||
|           }) | ||||
|  | ||||
|           await expect( | ||||
|             page.getByRole('button', { name: 'Start Sketch' }) | ||||
| @ -766,7 +812,7 @@ test.describe(`Project management commands`, () => { | ||||
|   test( | ||||
|     `Rename from project page`, | ||||
|     { tag: '@electron' }, | ||||
|     async ({ context, page, scene, cmdBar }, testInfo) => { | ||||
|     async ({ context, page }, testInfo) => { | ||||
|       const projectName = `my_project_to_rename` | ||||
|       await context.folderSetupFn(async (dir) => { | ||||
|         await fsp.mkdir(`${dir}/${projectName}`, { recursive: true }) | ||||
| @ -775,6 +821,7 @@ test.describe(`Project management commands`, () => { | ||||
|           `${dir}/${projectName}/main.kcl` | ||||
|         ) | ||||
|       }) | ||||
|       const u = await getUtils(page) | ||||
|  | ||||
|       // Constants and locators | ||||
|       const projectHomeLink = page.getByTestId('project-link') | ||||
| @ -796,7 +843,7 @@ test.describe(`Project management commands`, () => { | ||||
|         page.on('console', console.log) | ||||
|  | ||||
|         await projectHomeLink.click() | ||||
|         await scene.settled(cmdBar) | ||||
|         await u.waitForPageLoad() | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Run rename command via command palette`, async () => { | ||||
| @ -835,6 +882,7 @@ test.describe(`Project management commands`, () => { | ||||
|           `${dir}/${projectName}/main.kcl` | ||||
|         ) | ||||
|       }) | ||||
|       const u = await getUtils(page) | ||||
|  | ||||
|       // Constants and locators | ||||
|       const projectHomeLink = page.getByTestId('project-link') | ||||
| @ -852,9 +900,9 @@ test.describe(`Project management commands`, () => { | ||||
|         await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|         page.on('console', console.log) | ||||
|  | ||||
|         await page.waitForTimeout(3000) | ||||
|  | ||||
|         await projectHomeLink.click() | ||||
|         await u.waitForPageLoad() | ||||
|         await scene.connectionEstablished() | ||||
|         await scene.settled(cmdBar) | ||||
|       }) | ||||
|  | ||||
| @ -878,7 +926,7 @@ test.describe(`Project management commands`, () => { | ||||
|   test( | ||||
|     `Rename from home page`, | ||||
|     { tag: '@electron' }, | ||||
|     async ({ context, page, homePage, scene, cmdBar }, testInfo) => { | ||||
|     async ({ context, page, homePage }, testInfo) => { | ||||
|       const projectName = `my_project_to_rename` | ||||
|       await context.folderSetupFn(async (dir) => { | ||||
|         await fsp.mkdir(`${dir}/${projectName}`, { recursive: true }) | ||||
| @ -934,7 +982,7 @@ test.describe(`Project management commands`, () => { | ||||
|   test( | ||||
|     `Delete from home page`, | ||||
|     { tag: '@electron' }, | ||||
|     async ({ context, page, scene, cmdBar }, testInfo) => { | ||||
|     async ({ context, page }, testInfo) => { | ||||
|       const projectName = `my_project_to_delete` | ||||
|       await context.folderSetupFn(async (dir) => { | ||||
|         await fsp.mkdir(`${dir}/${projectName}`, { recursive: true }) | ||||
| @ -985,7 +1033,6 @@ test.describe(`Project management commands`, () => { | ||||
|     homePage, | ||||
|     toolbar, | ||||
|     cmdBar, | ||||
|     scene, | ||||
|   }) => { | ||||
|     const projectName = 'test-project' | ||||
|     await test.step(`Setup`, async () => { | ||||
| @ -1025,11 +1072,10 @@ test.describe(`Project management commands`, () => { | ||||
|       }) | ||||
|       await cmdBar.argumentInput.fill(projectName) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await scene.settled(cmdBar) | ||||
|       await toolbar.logoLink.click() | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Check the project was created with a non-colliding name`, async () => { | ||||
|       await toolbar.logoLink.click() | ||||
|       await homePage.expectState({ | ||||
|         projectCards: [ | ||||
|           { | ||||
| @ -1060,11 +1106,10 @@ test.describe(`Project management commands`, () => { | ||||
|       }) | ||||
|       await cmdBar.argumentInput.fill(projectName) | ||||
|       await cmdBar.progressCmdBar() | ||||
|       await scene.settled(cmdBar) | ||||
|       await toolbar.logoLink.click() | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Check the second project was created with a non-colliding name`, async () => { | ||||
|       await toolbar.logoLink.click() | ||||
|       await homePage.expectState({ | ||||
|         projectCards: [ | ||||
|           { | ||||
| @ -1150,7 +1195,7 @@ test( | ||||
| test( | ||||
|   'Nested directories in project without main.kcl do not create main.kcl', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ scene, cmdBar, context, page }, testInfo) => { | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     let testDir: string | undefined | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       await fsp.mkdir(path.join(dir, 'router-template-slate', 'nested'), { | ||||
| @ -1173,7 +1218,10 @@ test( | ||||
|  | ||||
|     await test.step('Open the project', async () => { | ||||
|       await page.getByText('router-template-slate').click() | ||||
|       await scene.settled(cmdBar) | ||||
|       await expect(page.getByTestId('loading')).toBeAttached() | ||||
|       await expect(page.getByTestId('loading')).not.toBeAttached({ | ||||
|         timeout: 20_000, | ||||
|       }) | ||||
|  | ||||
|       // It actually loads. | ||||
|       await expect(u.codeLocator).toContainText('mounting bracket') | ||||
| @ -1286,7 +1334,7 @@ test( | ||||
| test( | ||||
|   'Can load a file with CRLF line endings', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ context, page, scene, cmdBar }, testInfo) => { | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     if (runningOnWindows()) { | ||||
|       test.fixme(orRunWhenFullSuiteEnabled()) | ||||
|     } | ||||
| @ -1309,8 +1357,13 @@ test( | ||||
|     const u = await getUtils(page) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     page.on('console', console.log) | ||||
|  | ||||
|     await page.getByText('router-template-slate').click() | ||||
|     await scene.settled(cmdBar) | ||||
|     await expect(page.getByTestId('loading')).toBeAttached() | ||||
|     await expect(page.getByTestId('loading')).not.toBeAttached({ | ||||
|       timeout: 20_000, | ||||
|     }) | ||||
|  | ||||
|     await expect(u.codeLocator).toContainText('routerDiameter') | ||||
|     await expect(u.codeLocator).toContainText('templateGap') | ||||
| @ -1525,7 +1578,7 @@ extrude001 = extrude(sketch001, length = 200)`) | ||||
| test( | ||||
|   'Opening a project should successfully load the stream, (regression test that this also works when switching between projects)', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ context, page, cmdBar, homePage, scene }, testInfo) => { | ||||
|   async ({ context, page, cmdBar, homePage }, testInfo) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       await fsp.mkdir(path.join(dir, 'router-template-slate'), { | ||||
|         recursive: true, | ||||
| @ -1554,10 +1607,13 @@ test( | ||||
|         path.join(dir, 'bracket', 'main.kcl') | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     page.on('console', console.log) | ||||
|  | ||||
|     const pointOnModel = { x: 630, y: 280 } | ||||
|  | ||||
|     await test.step('Opening the bracket project via command palette should load the stream', async () => { | ||||
|       await homePage.expectState({ | ||||
|         projectCards: [ | ||||
| @ -1591,7 +1647,15 @@ test( | ||||
|         stage: 'commandBarClosed', | ||||
|       }) | ||||
|  | ||||
|       await scene.settled(cmdBar) | ||||
|       await u.waitForPageLoad() | ||||
|  | ||||
|       // gray at this pixel means the stream has loaded in the most | ||||
|       // user way we can verify it (pixel color) | ||||
|       await expect | ||||
|         .poll(() => u.getGreatestPixDiff(pointOnModel, [85, 85, 85]), { | ||||
|           timeout: 10_000, | ||||
|         }) | ||||
|         .toBeLessThan(15) | ||||
|     }) | ||||
|  | ||||
|     await test.step('Clicking the logo takes us back to the projects page / home', async () => { | ||||
| @ -1608,7 +1672,15 @@ test( | ||||
|  | ||||
|       await page.getByText('router-template-slate').click() | ||||
|  | ||||
|       await scene.settled(cmdBar) | ||||
|       await u.waitForPageLoad() | ||||
|  | ||||
|       // gray at this pixel means the stream has loaded in the most | ||||
|       // user way we can verify it (pixel color) | ||||
|       await expect | ||||
|         .poll(() => u.getGreatestPixDiff(pointOnModel, [143, 143, 143]), { | ||||
|           timeout: 10_000, | ||||
|         }) | ||||
|         .toBeLessThan(15) | ||||
|     }) | ||||
|  | ||||
|     await test.step('The projects on the home page should still be normal', async () => { | ||||
| @ -1661,6 +1733,8 @@ test( | ||||
|     }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     page.on('console', console.log) | ||||
|  | ||||
|     // we'll grab this from the settings on screen before we switch | ||||
|     let originalProjectDirName: string | ||||
|     const newProjectDirName = testInfo.outputPath( | ||||
| @ -1801,7 +1875,7 @@ test( | ||||
| test( | ||||
|   'file pane is scrollable when there are many files', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ scene, cmdBar, context, page }, testInfo) => { | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const testDir = path.join(dir, 'testProject') | ||||
|       await fsp.mkdir(testDir, { recursive: true }) | ||||
| @ -1880,8 +1954,10 @@ test( | ||||
|  | ||||
|     await test.step('setup, open file pane', async () => { | ||||
|       await page.getByText('testProject').click() | ||||
|  | ||||
|       await scene.settled(cmdBar) | ||||
|       await expect(page.getByTestId('loading')).toBeAttached() | ||||
|       await expect(page.getByTestId('loading')).not.toBeAttached({ | ||||
|         timeout: 20_000, | ||||
|       }) | ||||
|  | ||||
|       await page.getByTestId('files-pane-button').click() | ||||
|     }) | ||||
|  | ||||
| @ -41,13 +41,15 @@ sketch002 = startSketchOn(XZ) | ||||
| extrude002 = extrude(sketch002, length = 50) | ||||
| sketch003 = startSketchOn(XY) | ||||
|   |> startProfileAt([52.92, 157.81], %) | ||||
|   |> angledLine(angle = 0, length = 176.4, tag = $rectangleSegmentA001) | ||||
|   |> angledLine( | ||||
|        angle = segAng(rectangleSegmentA001) - 90, | ||||
|        length = 53.4, | ||||
|        tag = $rectangleSegmentB001, | ||||
|      ) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001), tag = $rectangleSegmentC001) | ||||
|   |> angledLine([0, 176.4], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        53.4 | ||||
|      ], %, $rectangleSegmentB001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $rectangleSegmentC001) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| extrude003 = extrude(sketch003, length = 20) | ||||
| @ -61,7 +63,7 @@ test.describe('edit with AI example snapshots', () => { | ||||
|         localStorage.setItem('persistCode', file) | ||||
|       }, file) | ||||
|       await homePage.goToModelingScene() | ||||
|       await scene.settled(cmdBar) | ||||
|       await scene.waitForExecutionDone() | ||||
|  | ||||
|       const body1CapCoords = { x: 571, y: 351 } | ||||
|       const [clickBody1Cap] = scene.makeMouseHelpers( | ||||
| @ -72,6 +74,7 @@ test.describe('edit with AI example snapshots', () => { | ||||
|       const submittingToast = page.getByText('Submitting to Text-to-CAD API...') | ||||
|  | ||||
|       await test.step('wait for scene to load select body and check selection came through', async () => { | ||||
|         await scene.expectPixelColor([134, 134, 134], body1CapCoords, 15) | ||||
|         await clickBody1Cap() | ||||
|         await scene.expectPixelColor(yellow, body1CapCoords, 20) | ||||
|         await editor.expectState({ | ||||
|  | ||||
| @ -22,9 +22,15 @@ sketch002 = startSketchOn(XZ) | ||||
| extrude002 = extrude(sketch002, length = 50) | ||||
| sketch003 = startSketchOn(XY) | ||||
|   |> startProfileAt([52.92, 157.81], %) | ||||
|   |> angledLine(angle = 0, length = 176.4, tag = $rectangleSegmentA001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001) - 90, length = 53.4, tag = $rectangleSegmentB001) | ||||
|   |> angledLine(angle = segAng(rectangleSegmentA001), length = -segLen(rectangleSegmentA001), tag = $rectangleSegmentC001) | ||||
|   |> angledLine([0, 176.4], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        53.4 | ||||
|      ], %, $rectangleSegmentB001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $rectangleSegmentC001) | ||||
|   |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) | ||||
|   |> close() | ||||
| extrude003 = extrude(sketch003, length = 20) | ||||
| @ -51,12 +57,11 @@ test.describe('Prompt-to-edit tests', { tag: '@skipWin' }, () => { | ||||
|         page, | ||||
|         scene, | ||||
|       }) => { | ||||
|         test.fixme(orRunWhenFullSuiteEnabled()) | ||||
|         await context.addInitScript((file) => { | ||||
|           localStorage.setItem('persistCode', file) | ||||
|         }, file) | ||||
|         await homePage.goToModelingScene() | ||||
|         await scene.settled(cmdBar) | ||||
|         await scene.waitForExecutionDone() | ||||
|  | ||||
|         const body1CapCoords = { x: 571, y: 311 } | ||||
|         const greenCheckCoords = { x: 565, y: 305 } | ||||
| @ -151,7 +156,7 @@ test.describe('Prompt-to-edit tests', { tag: '@skipWin' }, () => { | ||||
|       localStorage.setItem('persistCode', file) | ||||
|     }, file) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     const body1CapCoords = { x: 571, y: 311 } | ||||
|     const [clickBody1Cap] = scene.makeMouseHelpers( | ||||
| @ -207,7 +212,7 @@ test.describe('Prompt-to-edit tests', { tag: '@skipWin' }, () => { | ||||
|       localStorage.setItem('persistCode', file) | ||||
|     }, file) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     const submittingToast = page.getByText('Submitting to Text-to-CAD API...') | ||||
|     const successToast = page.getByText('Prompt to edit successful') | ||||
| @ -260,7 +265,6 @@ test.describe('Prompt-to-edit tests', { tag: '@skipWin' }, () => { | ||||
|     page, | ||||
|     scene, | ||||
|   }) => { | ||||
|     test.fixme(orRunWhenFullSuiteEnabled()) | ||||
|     const body1CapCoords = { x: 571, y: 311 } | ||||
|     const body2WallCoords = { x: 620, y: 152 } | ||||
|     const [clickBody1Cap] = scene.makeMouseHelpers( | ||||
| @ -277,7 +281,7 @@ test.describe('Prompt-to-edit tests', { tag: '@skipWin' }, () => { | ||||
|       localStorage.setItem('persistCode', file) | ||||
|     }, file) | ||||
|     await homePage.goToModelingScene() | ||||
|     await scene.settled(cmdBar) | ||||
|     await scene.waitForExecutionDone() | ||||
|  | ||||
|     const submittingToast = page.getByText('Submitting to Text-to-CAD API...') | ||||
|     const successToast = page.getByText('Prompt to edit successful') | ||||
|  | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	