Compare commits
	
		
			93 Commits
		
	
	
		
			nightly-v2
			...
			move-tests
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ff64367729 | |||
| c440bbc0fe | |||
| c2bb81ad1f | |||
| 82ec52b317 | |||
| f5a14166a1 | |||
| 6ac8cef180 | |||
| f0c8dbd5a2 | |||
| c20ce60c9a | |||
| f2b8e66952 | |||
| cdd73e952a | |||
| edbff28296 | |||
| 006fcd7490 | |||
| 4795de789d | |||
| 1a04a4dcfb | |||
| 119cacf6a8 | |||
| 12b00aca34 | |||
| 5c56f94fbb | |||
| 329e60dda8 | |||
| 2844f4c4d6 | |||
| 0c48006793 | |||
| 534d1ddecc | |||
| 4f5766d423 | |||
| 253556d6c7 | |||
| 10019a180d | |||
| 728314ccda | |||
| 7e229099a0 | |||
| da217b6c1a | |||
| ff4e1a6c73 | |||
| a7c2548645 | |||
| ccfc592b62 | |||
| b3cc43ae2f | |||
| f061a9a15f | |||
| 2b92f00d3c | |||
| 4b0823dbbf | |||
| 08a4016fda | |||
| cb56fc7555 | |||
| d7d822cc11 | |||
| 8e450378c3 | |||
| fd39fcdb25 | |||
| 24011dd100 | |||
| a9b78fb2a4 | |||
| 765e27c02b | |||
| 05baf9884d | |||
| 1ced492deb | |||
| 804124b07f | |||
| b3112025b9 | |||
| 898b4ed016 | |||
| 4b21a5b667 | |||
| 00e97257ae | |||
| aeb656d176 | |||
| ac49ebd6e0 | |||
| b40f03ad25 | |||
| a8ad86e645 | |||
| 87f50cd5e9 | |||
| 0400e6228e | |||
| 26f150fd6c | |||
| 3049f405f5 | |||
| 53d40301dc | |||
| 671c01e36f | |||
| e80151979b | |||
| 668e2afb99 | |||
| 548c664db0 | |||
| d3a3f4410c | |||
| 22eb343171 | |||
| f2cfa4d5cf | |||
| 3f1f40eeba | |||
| ff2d161606 | |||
| 210c78029d | |||
| e27840219b | |||
| c943a3f192 | |||
| 6aa588f09f | |||
| 59a6333aad | |||
| 403f1507ae | |||
| eac7b83504 | |||
| 667500d1b9 | |||
| b15aac9f48 | |||
| 54153aa646 | |||
| 943cf21d34 | |||
| 5a6728c45a | |||
| ff2103d493 | |||
| 2dfa8f2176 | |||
| 29ed330326 | |||
| ca2cc825a6 | |||
| 83fe1b7ce0 | |||
| 157b76cc78 | |||
| cf957d880e | |||
| dfc3d19677 | |||
| dd370a9365 | |||
| 2274d6459c | |||
| 32ce857119 | |||
| 88b51da417 | |||
| 30d365aeb3 | |||
| 7af62399ac | 
| @ -1,59 +0,0 @@ | ||||
| # bash strict mode | ||||
| set -euo pipefail | ||||
|  | ||||
| if [[ ! -f "test-results/.last-run.json" ]]; then | ||||
|     # if no last run artifact, than run plawright normally | ||||
|     echo "run playwright normally" | ||||
|     if [[ "$3" == ubuntu-latest* ]]; then | ||||
|         yarn test:playwright:browser:chrome:ubuntu -- --shard=$1/$2 || true | ||||
|     elif [[ "$3" == windows-latest* ]]; then | ||||
|         yarn test:playwright:browser:chrome:windows -- --shard=$1/$2 || true | ||||
|     else | ||||
|         echo "Do not run playwright. Unable to detect os runtime." | ||||
|         exit 1 | ||||
|     fi | ||||
|     # # send to axiom | ||||
|     node playwrightProcess.mjs | tee /tmp/github-actions.log > /dev/null 2>&1 | ||||
| fi | ||||
|  | ||||
| retry=1 | ||||
| max_retrys=4 | ||||
|  | ||||
| # retry failed tests, doing our own retries because using inbuilt playwright retries causes connection issues | ||||
| while [[ $retry -le $max_retrys ]]; do | ||||
|     if [[ -f "test-results/.last-run.json" ]]; then | ||||
|         failed_tests=$(jq '.failedTests | length' test-results/.last-run.json) | ||||
|         if [[ $failed_tests -gt 0 ]]; then | ||||
|             echo "retried=true" >>$GITHUB_OUTPUT | ||||
|             echo "run playwright with last failed tests and retry $retry" | ||||
|             if [[ "$3" == ubuntu-latest* ]]; then | ||||
|                 yarn test:playwright:browser:chrome:ubuntu -- --last-failed || true | ||||
|             elif [[ "$3" == windows-latest* ]]; then | ||||
|                 yarn test:playwright:browser:chrome:windows -- --last-failed || true | ||||
|             else | ||||
|                 echo "Do not run playwright. Unable to detect os runtime." | ||||
|                 exit 1 | ||||
|             fi | ||||
|             # send to axiom | ||||
|             node playwrightProcess.mjs | tee /tmp/github-actions.log > /dev/null 2>&1 | ||||
|             retry=$((retry + 1)) | ||||
|         else | ||||
|             echo "retried=false" >>$GITHUB_OUTPUT | ||||
|             exit 0 | ||||
|         fi | ||||
|     else | ||||
|         echo "retried=false" >>$GITHUB_OUTPUT | ||||
|         exit 0 | ||||
|     fi | ||||
| done | ||||
|  | ||||
| echo "retried=false" >>$GITHUB_OUTPUT | ||||
|  | ||||
| if [[ -f "test-results/.last-run.json" ]]; then | ||||
|     failed_tests=$(jq '.failedTests | length' test-results/.last-run.json) | ||||
|     if [[ $failed_tests -gt 0 ]]; then | ||||
|         # if it still fails after 3 retrys, then fail the job | ||||
|         exit 1 | ||||
|     fi | ||||
| fi | ||||
| exit 0 | ||||
							
								
								
									
										20
									
								
								.github/ci-cd-scripts/playwright-electron.sh
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -1,15 +1,17 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
| # bash strict mode | ||||
| set -euo pipefail | ||||
|  | ||||
| if [[ ! -f "test-results/.last-run.json" ]]; then | ||||
|     # if no last run artifact, than run plawright normally | ||||
|     echo "run playwright normally" | ||||
|         if [[ "$1" == ubuntu-latest* ]]; then | ||||
|             xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn test:playwright:electron:ubuntu || true | ||||
|         elif [[ "$1" == windows-latest* ]]; then | ||||
|             yarn test:playwright:electron:windows || true | ||||
|         elif [[ "$1" == macos-14* ]]; then | ||||
|             yarn test:playwright:electron:macos || true | ||||
|         if [[ "$3" == ubuntu-latest* ]]; then | ||||
|             xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn test:playwright:electron:ubuntu -- --shard=$1/$2 || true | ||||
|         elif [[ "$3" == windows-latest* ]]; then | ||||
|             yarn test:playwright:electron:windows -- --shard=$1/$2 || true | ||||
|         elif [[ "$3" == macos-14* ]]; then | ||||
|             yarn test:playwright:electron:macos  -- --shard=$1/$2 || true | ||||
|         else | ||||
|             echo "Do not run playwright. Unable to detect os runtime." | ||||
|             exit 1 | ||||
| @ -28,11 +30,11 @@ while [[ $retry -le $max_retrys ]]; do | ||||
|         if [[ $failed_tests -gt 0 ]]; then | ||||
|             echo "retried=true" >>$GITHUB_OUTPUT | ||||
|             echo "run playwright with last failed tests and retry $retry" | ||||
|             if [[ "$1" == ubuntu-latest* ]]; then | ||||
|             if [[ "$3" == ubuntu-latest* ]]; then | ||||
|                 xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn test:playwright:electron:ubuntu -- --last-failed || true | ||||
|             elif [[ "$1" == windows-latest* ]]; then | ||||
|             elif [[ "$3" == windows-latest* ]]; then | ||||
|                 yarn test:playwright:electron:windows -- --last-failed || true | ||||
|             elif [[ "$1" == macos-14* ]]; then | ||||
|             elif [[ "$3" == macos-14* ]]; then | ||||
|                 yarn test:playwright:electron:macos -- --last-failed || true | ||||
|             else | ||||
|                 echo "Do not run playwright. Unable to detect os runtime." | ||||
|  | ||||
							
								
								
									
										9
									
								
								.github/workflows/build-apps.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -165,7 +165,6 @@ jobs: | ||||
|       - name: Build the app (release) | ||||
|         if: ${{ env.IS_RELEASE == 'true' || env.IS_NIGHTLY == 'true' }} | ||||
|         env: | ||||
|           PUBLISH_FOR_PULL_REQUEST: true | ||||
|           APPLE_ID: ${{ secrets.APPLE_ID }} | ||||
|           APPLE_PASSWORD: ${{ secrets.APPLE_PASSWORD }} | ||||
|           APPLE_APP_SPECIFIC_PASSWORD: ${{ secrets.APPLE_PASSWORD }} | ||||
| @ -173,7 +172,6 @@ jobs: | ||||
|           CSC_LINK: ${{ secrets.APPLE_CERTIFICATE }} | ||||
|           CSC_KEY_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }} | ||||
|           CSC_KEYCHAIN: ${{ secrets.APPLE_SIGNING_IDENTITY }} | ||||
|           CSC_FOR_PULL_REQUEST: true | ||||
|           WINDOWS_CERTIFICATE_THUMBPRINT: ${{ secrets.WINDOWS_CERTIFICATE_THUMBPRINT }} | ||||
|         run: yarn electron-builder --config --publish always | ||||
|  | ||||
| @ -229,7 +227,6 @@ jobs: | ||||
|           CSC_LINK: ${{ secrets.APPLE_CERTIFICATE }} | ||||
|           CSC_KEY_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }} | ||||
|           CSC_KEYCHAIN: ${{ secrets.APPLE_SIGNING_IDENTITY }} | ||||
|           CSC_FOR_PULL_REQUEST: true | ||||
|           WINDOWS_CERTIFICATE_THUMBPRINT: ${{ secrets.WINDOWS_CERTIFICATE_THUMBPRINT }} | ||||
|         run: yarn electron-builder --config --publish always | ||||
|  | ||||
| @ -365,7 +362,7 @@ jobs: | ||||
|       - name: Set more complete nightly release notes | ||||
|         if: ${{ env.IS_NIGHTLY == 'true' }} | ||||
|         run: | | ||||
|           # Note: prefered going this way instead of a full clone in the checkout step, | ||||
|           # Note: preferred going this way instead of a full clone in the checkout step, | ||||
|           # see https://github.com/actions/checkout/issues/1471 | ||||
|           git fetch --prune --unshallow --tags | ||||
|           export TAG="nightly-${VERSION}" | ||||
| @ -394,6 +391,10 @@ jobs: | ||||
|           parent: false | ||||
|           destination: 'dl.kittycad.io/releases/modeling-app/nightly' | ||||
|  | ||||
|       - name: Invalidate bucket cache on latest*.yml and last_download.json files | ||||
|         if: ${{ env.IS_NIGHTLY == 'true' }} | ||||
|         run: yarn files:invalidate-bucket:nightly | ||||
|  | ||||
|       - name: Tag nightly commit | ||||
|         if: ${{ env.IS_NIGHTLY == 'true' }} | ||||
|         uses: actions/github-script@v7 | ||||
|  | ||||
							
								
								
									
										154
									
								
								.github/workflows/e2e-tests.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -33,13 +33,13 @@ jobs: | ||||
|             rust: | ||||
|               - 'src/wasm-lib/**' | ||||
|  | ||||
|   browser: | ||||
|   electron: | ||||
|     timeout-minutes: ${{ matrix.os == 'macos-14' && 60 || 50 }} | ||||
|     name: playwright:browser:${{ matrix.os }} ${{ matrix.shardIndex }} ${{ matrix.shardTotal }} | ||||
|     name: playwright:electron:${{ matrix.os }} ${{ matrix.shardIndex }} ${{ matrix.shardTotal }} | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         os: [ubuntu-latest-8-cores, windows-latest-8-cores] | ||||
|         os: [ubuntu-latest-8-cores, windows-latest-8-cores, macos-14-large] | ||||
|         shardIndex: [1, 2, 3, 4] | ||||
|         shardTotal: [4] | ||||
|     runs-on: ${{ matrix.os }} | ||||
| @ -123,13 +123,13 @@ jobs: | ||||
|       if: steps.download-wasm.outcome == 'failure' | ||||
|       shell: bash | ||||
|       run: yarn build:wasm | ||||
|     - name: build web | ||||
|       run: yarn build:local | ||||
|     - name: build electron | ||||
|       shell: bash | ||||
|       run: yarn tron:package | ||||
|     - name: Run ubuntu/chrome snapshots | ||||
|       shell: bash | ||||
|       run: | | ||||
|         yarn playwright test --project="Google Chrome" --config=playwright.ci.config.ts --retries="3" --update-snapshots --grep=@snapshot  --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} | ||||
|         PLATFORM=web yarn playwright test --config=playwright.config.ts --retries="3" --update-snapshots --grep=@snapshot  --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} | ||||
|       env: | ||||
|         CI: true | ||||
|         NODE_ENV: development | ||||
| @ -186,12 +186,12 @@ jobs: | ||||
|       with: | ||||
|         name: test-results-${{ matrix.os }}-${{ matrix.shardIndex }}-${{ github.sha }} | ||||
|         path: test-results/ | ||||
|     - name: Run playwright/chrome flow (with retries) | ||||
|     - name: Run playwright/electron flow (with retries) | ||||
|       id: retry | ||||
|       if: ${{ !cancelled() && (success() || failure()) }} | ||||
|       shell: bash | ||||
|       run: | | ||||
|         .github/ci-cd-scripts/playwright-browser-chrome.sh ${{matrix.shardIndex}} ${{matrix.shardTotal}} ${{matrix.os}} | ||||
|         .github/ci-cd-scripts/playwright-electron.sh ${{matrix.shardIndex}} ${{matrix.shardTotal}} ${{matrix.os}} | ||||
|       env: | ||||
|         CI: true | ||||
|         FAIL_ON_CONSOLE_ERRORS: true | ||||
| @ -199,11 +199,6 @@ jobs: | ||||
|         VITE_KC_DEV_TOKEN: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|         VITE_KC_SKIP_AUTH: true | ||||
|         token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|     - name: send to axiom | ||||
|       if: always() | ||||
|       shell: bash | ||||
|       run: | | ||||
|         node playwrightProcess.mjs | tee /tmp/github-actions.log | ||||
|     - uses: actions/upload-artifact@v4 | ||||
|       if: always() | ||||
|       with: | ||||
| @ -221,136 +216,3 @@ jobs: | ||||
|         retention-days: 30 | ||||
|         overwrite: true | ||||
|  | ||||
|  | ||||
|   electron: | ||||
|     name: playwright:electron:${{matrix.os}} | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         os: [ubuntu-latest-8-cores, windows-latest-8-cores, macos-14-large] | ||||
|     timeout-minutes: 60 | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     needs: check-rust-changes | ||||
|     steps: | ||||
|     - name: Tune GitHub-hosted runner network | ||||
|       uses: smorimoto/tune-github-hosted-runner-network@v1 | ||||
|     - uses: actions/checkout@v4 | ||||
|     - uses: actions/setup-node@v4 | ||||
|       with: | ||||
|         node-version-file: '.nvmrc' | ||||
|         cache: 'yarn' | ||||
|     - uses: KittyCAD/action-install-cli@main | ||||
|     - name: Install dependencies | ||||
|       shell: bash | ||||
|       run: yarn | ||||
|     - name: Cache Playwright Browsers | ||||
|       uses: actions/cache@v4 | ||||
|       with: | ||||
|         path: | | ||||
|           ~/.cache/ms-playwright/ | ||||
|         key: ${{ runner.os }}-playwright-${{ hashFiles('yarn.lock') }} | ||||
|     - name: Install Playwright Browsers | ||||
|       shell: bash | ||||
|       run: yarn playwright install chromium --with-deps | ||||
|     - name: Download Wasm Cache | ||||
|       id: download-wasm | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'false' | ||||
|       uses: dawidd6/action-download-artifact@v7 | ||||
|       continue-on-error: true | ||||
|       with: | ||||
|         github_token: ${{secrets.GITHUB_TOKEN}} | ||||
|         name: wasm-bundle | ||||
|         workflow: build-and-store-wasm.yml | ||||
|         branch: main | ||||
|         path: src/wasm-lib/pkg | ||||
|     - name: copy wasm blob | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'false' | ||||
|       shell: bash | ||||
|       run: cp src/wasm-lib/pkg/wasm_lib_bg.wasm public | ||||
|       continue-on-error: true | ||||
|     - name: Setup Rust | ||||
|       uses: dtolnay/rust-toolchain@stable | ||||
|     - name: Cache Wasm (because rust diff) | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'true' | ||||
|       uses: Swatinem/rust-cache@v2 | ||||
|       with: | ||||
|         workspaces: './src/wasm-lib' | ||||
|     - name: OR Cache Wasm (because wasm cache failed) | ||||
|       if: steps.download-wasm.outcome == 'failure' | ||||
|       uses: Swatinem/rust-cache@v2 | ||||
|       with: | ||||
|         workspaces: './src/wasm-lib' | ||||
|     - name: install good sed | ||||
|       if:  ${{ startsWith(matrix.os, 'macos') }} | ||||
|       shell: bash | ||||
|       run: | | ||||
|         brew install gnu-sed | ||||
|         echo "/opt/homebrew/opt/gnu-sed/libexec/gnubin" >> $GITHUB_PATH | ||||
|     - name: Install vector | ||||
|       if:  ${{ startsWith(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 & | ||||
|     - name: Build Wasm (because rust diff) | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'true' | ||||
|       shell: bash | ||||
|       run: yarn build:wasm | ||||
|     - name: OR Build Wasm (because wasm cache failed) | ||||
|       if: steps.download-wasm.outcome == 'failure' | ||||
|       shell: bash | ||||
|       run: yarn build:wasm | ||||
|     - name: build electron | ||||
|       shell: bash | ||||
|       run: yarn tron:package | ||||
|     - uses: actions/download-artifact@v4 | ||||
|       if: ${{ !cancelled() && (success() || failure()) }} | ||||
|       continue-on-error: true | ||||
|       with: | ||||
|         name: test-results-electron-${{ matrix.os }}-${{ github.sha }} | ||||
|         path: test-results/ | ||||
|     - name: Run electron tests (with retries) | ||||
|       id: retry | ||||
|       if: ${{ !cancelled() && (success() || failure()) }} | ||||
|       shell: bash | ||||
|       run: | | ||||
|         .github/ci-cd-scripts/playwright-electron.sh ${{ matrix.os }} | ||||
|       env: | ||||
|         CI: true | ||||
|         FAIL_ON_CONSOLE_ERRORS: true | ||||
|         NODE_ENV: development | ||||
|         VITE_KC_DEV_TOKEN: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|         VITE_KC_SKIP_AUTH: true | ||||
|         IS_UBUNTU: ${{ startsWith(matrix.os, 'ubuntu') && 'true' || 'false' }} | ||||
|         #DEBUG: 'pw:browser*' | ||||
|     - name: send to axiom | ||||
|       if: ${{ !cancelled() && (success() || failure()) && !startsWith(matrix.os, 'windows') }} | ||||
|       shell: bash | ||||
|       run: | | ||||
|         node playwrightProcess.mjs | tee /tmp/github-actions.log | ||||
|     - uses: actions/upload-artifact@v4 | ||||
|       if: ${{ !cancelled() && (success() || failure()) }} | ||||
|       with: | ||||
|         name: test-results-electron-${{ matrix.os }}-${{ github.sha }} | ||||
|         path: test-results/ | ||||
|         include-hidden-files: true | ||||
|         retention-days: 30 | ||||
|         overwrite: true | ||||
|     - uses: actions/upload-artifact@v4 | ||||
|       if: ${{ !cancelled() && (success() || failure()) }} | ||||
|       with: | ||||
|         name: playwright-report-electron-${{ matrix.os }}-${{ github.sha }} | ||||
|         path: playwright-report/ | ||||
|         include-hidden-files: true | ||||
|         retention-days: 30 | ||||
|         overwrite: true | ||||
|  | ||||
							
								
								
									
										6
									
								
								.github/workflows/publish-apps-release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -126,11 +126,7 @@ jobs: | ||||
|           destination: 'dl.kittycad.io/releases/modeling-app' | ||||
|  | ||||
|       - name: Invalidate bucket cache on latest*.yml and last_download.json files | ||||
|         run: | | ||||
|           gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="/releases/modeling-app/last_download.json" --async | ||||
|           gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="/releases/modeling-app/latest-linux-arm64.yml" --async | ||||
|           gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="/releases/modeling-app/latest-mac.yml" --async | ||||
|           gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="/releases/modeling-app/latest.yml" --async | ||||
|         run: yarn files:invalidate-bucket | ||||
|  | ||||
|       - name: Upload release files to Github | ||||
|         if: ${{ github.event_name == 'release' }} | ||||
|  | ||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -61,6 +61,7 @@ Mac_App_Distribution.provisionprofile | ||||
| *.tsbuildinfo | ||||
| src/wasm-lib/pkg | ||||
|  | ||||
| .eslintcache | ||||
| venv | ||||
| .vite/ | ||||
|  | ||||
|  | ||||
							
								
								
									
										43
									
								
								INSTALL.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,43 @@ | ||||
| # Setting Up Zoo Modeling App | ||||
|  | ||||
| Compared to other CAD software, getting Zoo Modeling App up and running is quick and straightforward across platforms. It's about 100MB to download and is quick to install. | ||||
|  | ||||
| ## Windows | ||||
|  | ||||
| 1. Download the [Zoo Modeling App installer](https://zoo.dev/modeling-app/download) for Windows and for your processor type. | ||||
|  | ||||
| 2. Once downloaded, run the installer `Zoo Modeling App-{version}-{arch}-win.exe` which should take a few seconds. | ||||
|  | ||||
| 3. The installation happens at `C:\Program Files\Zoo Modeling App`. A shortcut in the start menu is also created so you can run the app easily by clicking on it. | ||||
|  | ||||
| ## macOS | ||||
|  | ||||
| 1. Download the [Zoo Modeling App installer](https://zoo.dev/modeling-app/download) for macOS and for your processor type. | ||||
|  | ||||
| 2. Once downloaded, open the disk image `Zoo Modeling App-{version}-{arch}-mac.dmg` and drag the applications to your `Applications` directory. | ||||
|  | ||||
| 3. You can then open your `Applications` directory and double-click on `Zoo Modeling App` to open. | ||||
|  | ||||
|  | ||||
| ## Linux  | ||||
|  | ||||
| 1. Download the [Zoo Modeling App installer](https://zoo.dev/modeling-app/download) for Linux and for your processor type. | ||||
|  | ||||
| 2. Install the dependencies needed to run the [AppImage format](https://appimage.org/). | ||||
|     -  On Ubuntu, install the FUSE library with these commands in a terminal. | ||||
|        ```bash | ||||
|        sudo apt update | ||||
|        sudo apt install libfuse2 | ||||
|        ``` | ||||
|     - Optionally, follow [these steps](https://github.com/probonopd/go-appimage/blob/master/src/appimaged/README.md#initial-setup) to install `appimaged`. It is a daemon that makes interacting with AppImage files more seamless.  | ||||
|     - Once installed, copy the downloaded `Zoo Modeling App-{version}-{arch}-linux.AppImage` to the directory of your choice, for instance `~/Applications`. | ||||
|  | ||||
|    - `appimaged` should automatically find it and make it executable. If not, run: | ||||
|      ```bash | ||||
|      chmod a+x ~/Applications/Zoo\ Modeling\ App-{version}-{arch}-linux.AppImage | ||||
|      ``` | ||||
|  | ||||
| 3. You can double-click on the AppImage to run it, or in a terminal with this command: | ||||
|    ```bash | ||||
|     ~/Applications/Zoo\ Modeling\ App-{version}-{arch}-linux.AppImage | ||||
|    ``` | ||||
							
								
								
									
										17
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @ -388,23 +388,6 @@ yarn test:unit:local | ||||
|  | ||||
| #### E2E Tests | ||||
|  | ||||
| **Playwright Browser** | ||||
|  | ||||
| These E2E tests run in a browser (without electron). | ||||
| There are tests that are skipped if they are ran in a windows OS or Linux OS. We can use playwright tags to implement test skipping. | ||||
|  | ||||
| Breaking down the command `yarn test:playwright:browser:chrome:windows` | ||||
| - The application is `playwright` | ||||
| - The runtime is a `browser` | ||||
| - The specific `browser` is `chrome` | ||||
| - The test should run in a `windows` environment. It will skip tests that are broken or flaky in the windows OS. | ||||
|  | ||||
| ``` | ||||
| yarn test:playwright:browser:chrome | ||||
| yarn test:playwright:browser:chrome:windows | ||||
| yarn test:playwright:browser:chrome:ubuntu | ||||
| ``` | ||||
|  | ||||
| **Playwright Electron** | ||||
|  | ||||
| These E2E tests run in electron. There are tests that are skipped if they are ran in a windows, linux, or macos environment. We can use playwright tags to implement test skipping. | ||||
|  | ||||
| @ -22,3 +22,5 @@ once fixed in engine will just start working here with no language changes. | ||||
|  | ||||
| - **Chamfers**: Chamfers cannot intersect, you will get an error. Only simple | ||||
|     chamfer cases work currently. | ||||
|  | ||||
| - **Appearance**: Changing the appearance on a loft does not work. | ||||
|  | ||||
							
								
								
									
										210
									
								
								docs/kcl/appearance.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -19,6 +19,7 @@ layout: manual | ||||
| * [`angledLineThatIntersects`](kcl/angledLineThatIntersects) | ||||
| * [`angledLineToX`](kcl/angledLineToX) | ||||
| * [`angledLineToY`](kcl/angledLineToY) | ||||
| * [`appearance`](kcl/appearance) | ||||
| * [`arc`](kcl/arc) | ||||
| * [`arcTo`](kcl/arcTo) | ||||
| * [`asin`](kcl/asin) | ||||
|  | ||||
| @ -45,7 +45,7 @@ circles = map([1..3], drawCircle) | ||||
| ```js | ||||
| r = 10 // radius | ||||
| // Call `map`, using an anonymous function instead of a named one. | ||||
| circles = map([1..3], (id) { | ||||
| circles = map([1..3], fn(id) { | ||||
|   return startSketchOn("XY") | ||||
|     |> circle({ center = [id * 2 * r, 0], radius = r }, %) | ||||
| }) | ||||
|  | ||||
| @ -61,7 +61,7 @@ assertEqual(sum([1, 2, 3]), 6, 0.00001, "1 + 2 + 3 summed is 6") | ||||
| // an anonymous `add` function as its parameter, instead of declaring a | ||||
| // named function outside. | ||||
| arr = [1, 2, 3] | ||||
| sum = reduce(arr, 0, (i, result_so_far) { | ||||
| sum = reduce(arr, 0, fn(i, result_so_far) { | ||||
|   return i + result_so_far | ||||
| }) | ||||
|  | ||||
| @ -84,7 +84,7 @@ fn decagon(radius) { | ||||
|   // Use a `reduce` to draw the remaining decagon sides. | ||||
|   // For each number in the array 1..10, run the given function, | ||||
|   // which takes a partially-sketched decagon and adds one more edge to it. | ||||
|   fullDecagon = reduce([1..10], startOfDecagonSketch, (i, partialDecagon) { | ||||
|   fullDecagon = reduce([1..10], startOfDecagonSketch, fn(i, partialDecagon) { | ||||
|     // Draw one edge of the decagon. | ||||
|     x = cos(stepAngle * i) * radius | ||||
|     y = sin(stepAngle * i) * radius | ||||
|  | ||||
							
								
								
									
										2891
									
								
								docs/kcl/std.json
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										23
									
								
								docs/kcl/types/AppearanceData.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,23 @@ | ||||
| --- | ||||
| title: "AppearanceData" | ||||
| excerpt: "Data for appearance." | ||||
| layout: manual | ||||
| --- | ||||
|  | ||||
| Data for appearance. | ||||
|  | ||||
| **Type:** `object` | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Properties | ||||
|  | ||||
| | Property | Type | Description | Required | | ||||
| |----------|------|-------------|----------| | ||||
| | `color` |`string`| Color of the new material, a hex string like "#ff0000". | No | | ||||
| | `metalness` |`number` (**maximum:** 100.0)| Metalness of the new material, a percentage like 95.7. | No | | ||||
| | `roughness` |`number` (**maximum:** 100.0)| Roughness of the new material, a percentage like 95.7. | No | | ||||
|  | ||||
|  | ||||
| @ -1,22 +1,11 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
|  | ||||
| import { setupElectron, tearDown } from './test-utils' | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
| import { test, expect } from './zoo-test' | ||||
|  | ||||
| test.describe('Electron app header tests', () => { | ||||
|   test( | ||||
|     'Open Command Palette button has correct shortcut', | ||||
|     { tag: '@electron' }, | ||||
|     async ({ browserName }, testInfo) => { | ||||
|       const { electronApp, page } = await setupElectron({ | ||||
|         testInfo, | ||||
|         folderSetupFn: async () => {}, | ||||
|       }) | ||||
|  | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     async ({ page }, testInfo) => { | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|       // No space before the shortcut since it checks textContent. | ||||
|       let text | ||||
| @ -34,21 +23,14 @@ test.describe('Electron app header tests', () => { | ||||
|       const commandsButton = page.getByRole('button', { name: 'Commands' }) | ||||
|       await expect(commandsButton).toBeVisible() | ||||
|       await expect(commandsButton).toHaveText(text) | ||||
|  | ||||
|       await electronApp.close() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test( | ||||
|     'User settings has correct shortcut', | ||||
|     { tag: '@electron' }, | ||||
|     async ({ browserName }, testInfo) => { | ||||
|       const { electronApp, page } = await setupElectron({ | ||||
|         testInfo, | ||||
|         folderSetupFn: async () => {}, | ||||
|       }) | ||||
|  | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     async ({ page }, testInfo) => { | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|       // Open the user sidebar menu. | ||||
|       await page.getByTestId('user-sidebar-toggle').click() | ||||
| @ -59,8 +41,6 @@ test.describe('Electron app header tests', () => { | ||||
|       const userSettingsButton = page.getByTestId('user-settings') | ||||
|       await expect(userSettingsButton).toBeVisible() | ||||
|       await expect(userSettingsButton).toHaveText(text) | ||||
|  | ||||
|       await electronApp.close() | ||||
|     } | ||||
|   ) | ||||
| }) | ||||
|  | ||||
| @ -1,29 +1,26 @@ | ||||
| import { test, expect, Page } from '@playwright/test' | ||||
| import { test, expect, Page } from './zoo-test' | ||||
| import { | ||||
|   getUtils, | ||||
|   TEST_COLORS, | ||||
|   setup, | ||||
|   tearDown, | ||||
|   commonPoints, | ||||
|   PERSIST_MODELING_CONTEXT, | ||||
| } from './test-utils' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
| import { HomePageFixture } from './fixtures/homePageFixture' | ||||
|  | ||||
| test.setTimeout(120000) | ||||
|  | ||||
| async function doBasicSketch(page: Page, openPanes: string[]) { | ||||
| async function doBasicSketch( | ||||
|   page: Page, | ||||
|   homePage: HomePageFixture, | ||||
|   openPanes: string[] | ||||
| ) { | ||||
|   const u = await getUtils(page) | ||||
|   await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|   await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|   const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|  | ||||
|   await u.waitForAuthSkipAppStart() | ||||
|   await homePage.goToModelingScene() | ||||
|   await u.waitForPageLoad() | ||||
|   await page.waitForTimeout() | ||||
|   await u.openDebugPanel() | ||||
|  | ||||
|   // If we have the code pane open, we should see the code. | ||||
| @ -148,13 +145,11 @@ async function doBasicSketch(page: Page, openPanes: string[]) { | ||||
| } | ||||
|  | ||||
| test.describe('Basic sketch', () => { | ||||
|   test('code pane open at start', { tag: ['@skipWin'] }, async ({ page }) => { | ||||
|     // Skip on windows it is being weird. | ||||
|     test.skip(process.platform === 'win32', 'Skip on windows') | ||||
|     await doBasicSketch(page, ['code']) | ||||
|   test.fixme('code pane open at start', async ({ page, homePage }) => { | ||||
|     await doBasicSketch(page, homePage, ['code']) | ||||
|   }) | ||||
|  | ||||
|   test('code pane closed at start', async ({ page }) => { | ||||
|   test.fixme('code pane closed at start', async ({ page, homePage }) => { | ||||
|     // Load the app with the code panes | ||||
|     await page.addInitScript(async (persistModelingContext) => { | ||||
|       localStorage.setItem( | ||||
| @ -162,6 +157,6 @@ test.describe('Basic sketch', () => { | ||||
|         JSON.stringify({ openPanes: [] }) | ||||
|       ) | ||||
|     }, PERSIST_MODELING_CONTEXT) | ||||
|     await doBasicSketch(page, []) | ||||
|     await doBasicSketch(page, homePage, []) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -1,27 +1,21 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { getUtils, setup, tearDown } from './test-utils' | ||||
| import { test, expect, Page } from './zoo-test' | ||||
| import { HomePageFixture } from './fixtures/homePageFixture' | ||||
| import { getUtils } from './test-utils' | ||||
| import { EngineCommand } from 'lang/std/artifactGraph' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.describe('Can create sketches on all planes and their back sides', () => { | ||||
|   const sketchOnPlaneAndBackSideTest = async ( | ||||
|     page: any, | ||||
|     page: Page, | ||||
|     homePage: HomePageFixture, | ||||
|     plane: string, | ||||
|     clickCoords: { x: number; y: number } | ||||
|   ) => { | ||||
|     const u = await getUtils(page) | ||||
|     const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.openDebugPanel() | ||||
|  | ||||
|     const coord = | ||||
| @ -83,32 +77,39 @@ test.describe('Can create sketches on all planes and their back sides', () => { | ||||
|     await u.clearCommandLogs() | ||||
|     await u.removeCurrentCode() | ||||
|   } | ||||
|   test('XY', async ({ page }) => { | ||||
|   test('XY', async ({ page, homePage }) => { | ||||
|     await sketchOnPlaneAndBackSideTest( | ||||
|       page, | ||||
|       homePage, | ||||
|       'XY', | ||||
|       { x: 600, y: 388 } // red plane | ||||
|       // { x: 600, y: 400 }, // red plane // clicks grid helper and that causes problems, should fix so that these coords work too. | ||||
|     ) | ||||
|   }) | ||||
|  | ||||
|   test('YZ', async ({ page }) => { | ||||
|     await sketchOnPlaneAndBackSideTest(page, 'YZ', { x: 700, y: 250 }) // green plane | ||||
|   test('YZ', async ({ page, homePage }) => { | ||||
|     await sketchOnPlaneAndBackSideTest(page, homePage, 'YZ', { x: 700, y: 250 }) // green plane | ||||
|   }) | ||||
|  | ||||
|   test('XZ', async ({ page }) => { | ||||
|     await sketchOnPlaneAndBackSideTest(page, '-XZ', { x: 700, y: 80 }) // blue plane | ||||
|   test('XZ', async ({ page, homePage }) => { | ||||
|     await sketchOnPlaneAndBackSideTest(page, homePage, '-XZ', { x: 700, y: 80 }) // blue plane | ||||
|   }) | ||||
|  | ||||
|   test('-XY', async ({ page }) => { | ||||
|     await sketchOnPlaneAndBackSideTest(page, '-XY', { x: 600, y: 118 }) // back of red plane | ||||
|   test('-XY', async ({ page, homePage }) => { | ||||
|     await sketchOnPlaneAndBackSideTest(page, homePage, '-XY', { | ||||
|       x: 600, | ||||
|       y: 118, | ||||
|     }) // back of red plane | ||||
|   }) | ||||
|  | ||||
|   test('-YZ', async ({ page }) => { | ||||
|     await sketchOnPlaneAndBackSideTest(page, '-YZ', { x: 700, y: 219 }) // back of green plane | ||||
|   test('-YZ', async ({ page, homePage }) => { | ||||
|     await sketchOnPlaneAndBackSideTest(page, homePage, '-YZ', { | ||||
|       x: 700, | ||||
|       y: 219, | ||||
|     }) // back of green plan | ||||
|   }) | ||||
|  | ||||
|   test('-XZ', async ({ page }) => { | ||||
|     await sketchOnPlaneAndBackSideTest(page, 'XZ', { x: 700, y: 427 }) // back of blue plane | ||||
|   test('-XZ', async ({ page, homePage }) => { | ||||
|     await sketchOnPlaneAndBackSideTest(page, homePage, 'XZ', { x: 700, y: 427 }) // back of blue plane | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -1,28 +1,15 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
|  | ||||
| import { | ||||
|   getUtils, | ||||
|   setup, | ||||
|   setupElectron, | ||||
|   tearDown, | ||||
|   executorInputPath, | ||||
| } from './test-utils' | ||||
| import { getUtils, executorInputPath } from './test-utils' | ||||
| import { join } from 'path' | ||||
| import { bracket } from 'lib/exampleKcl' | ||||
| import { TEST_CODE_LONG_WITH_ERROR_OUT_OF_VIEW } from './storageStates' | ||||
| import fsp from 'fs/promises' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.describe('Code pane and errors', () => { | ||||
|   test('Typing KCL errors induces a badge on the code pane button', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
| @ -31,18 +18,18 @@ test.describe('Code pane and errors', () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `// Extruded Triangle | ||||
| sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([10, 0], %) | ||||
|   |> line([-5, 10], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| extrude001 = extrude(5, sketch001)` | ||||
|   sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt([0, 0], %) | ||||
|     |> line([10, 0], %) | ||||
|     |> line([-5, 10], %) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|   extrude001 = extrude(5, sketch001)` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // wait for execution done | ||||
|     await u.openDebugPanel() | ||||
| @ -62,11 +49,11 @@ extrude001 = extrude(5, sketch001)` | ||||
|     await expect(codePaneButtonHolder).toContainText('notification') | ||||
|   }) | ||||
|  | ||||
|   test('Opening and closing the code pane will consistently show error diagnostics', async ({ | ||||
|   test.skip('Opening and closing the code pane will consistently show error diagnostics', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|     editor, | ||||
|   }) => { | ||||
|     await page.goto('http://localhost:3000') | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     // Load the app with the working starter code | ||||
| @ -74,8 +61,8 @@ extrude001 = extrude(5, sketch001)` | ||||
|       localStorage.setItem('persistCode', code) | ||||
|     }, bracket) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 900 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page.setBodyDimensions({ width: 1200, height: 900 }) | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // wait for execution done | ||||
|     await u.openDebugPanel() | ||||
| @ -91,8 +78,9 @@ extrude001 = extrude(5, sketch001)` | ||||
|     await expect(codePaneButtonHolder).not.toContainText('notification') | ||||
|  | ||||
|     // Delete a character to break the KCL | ||||
|     await u.openKclCodePanel() | ||||
|     await page.getByText('thickness, bracketLeg1Sketch)').click() | ||||
|     await editor.openPane() | ||||
|     await editor.scrollToText('thickness, bracketLeg1Sketch)') | ||||
|     await page.getByText('extrude(thickness, bracketLeg1Sketch)').click() | ||||
|     await page.keyboard.press('Backspace') | ||||
|  | ||||
|     // Ensure that a badge appears on the button | ||||
| @ -116,7 +104,10 @@ extrude001 = extrude(5, sketch001)` | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|  | ||||
|     // Open the code pane | ||||
|     await u.openKclCodePanel() | ||||
|     await editor.openPane() | ||||
|  | ||||
|     // Go to our problematic code again (missing closing paren!) | ||||
|     await editor.scrollToText('extrude(thickness, bracketLeg1Sketch') | ||||
|  | ||||
|     // Ensure that a badge appears on the button | ||||
|     await expect(codePaneButtonHolder).toContainText('notification') | ||||
| @ -129,59 +120,58 @@ extrude001 = extrude(5, sketch001)` | ||||
|     await expect(page.locator('.cm-tooltip').first()).toBeVisible() | ||||
|   }) | ||||
|  | ||||
|   test('When error is not in view you can click the badge to scroll to it', async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|   test.fixme( | ||||
|     'When error is not in view you can click the badge to scroll to it', | ||||
|     async ({ page, homePage, context }) => { | ||||
|       // Load the app with the working starter code | ||||
|       await context.addInitScript((code) => { | ||||
|         localStorage.setItem('persistCode', code) | ||||
|       }, TEST_CODE_LONG_WITH_ERROR_OUT_OF_VIEW) | ||||
|  | ||||
|     // Load the app with the working starter code | ||||
|     await page.addInitScript((code) => { | ||||
|       localStorage.setItem('persistCode', code) | ||||
|     }, TEST_CODE_LONG_WITH_ERROR_OUT_OF_VIEW) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|       await page.waitForTimeout(1000) | ||||
|  | ||||
|     await page.waitForTimeout(1000) | ||||
|       // Ensure badge is present | ||||
|       const codePaneButtonHolder = page.locator('#code-button-holder') | ||||
|       await expect(codePaneButtonHolder).toContainText('notification') | ||||
|  | ||||
|     // Ensure badge is present | ||||
|     const codePaneButtonHolder = page.locator('#code-button-holder') | ||||
|     await expect(codePaneButtonHolder).toContainText('notification') | ||||
|       // Ensure we have no errors in the gutter, since error out of view. | ||||
|       await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|  | ||||
|     // Ensure we have no errors in the gutter, since error out of view. | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|       // Click the badge. | ||||
|       const badge = page.locator('#code-badge') | ||||
|       await expect(badge).toBeVisible() | ||||
|       await badge.click() | ||||
|  | ||||
|     // Click the badge. | ||||
|     const badge = page.locator('#code-badge') | ||||
|     await expect(badge).toBeVisible() | ||||
|     await badge.click() | ||||
|       // Ensure we have an error diagnostic. | ||||
|       await expect(page.locator('.cm-lint-marker-error').first()).toBeVisible() | ||||
|  | ||||
|     // Ensure we have an error diagnostic. | ||||
|     await expect(page.locator('.cm-lint-marker-error').first()).toBeVisible() | ||||
|  | ||||
|     // Hover over the error to see the error message | ||||
|     await page.hover('.cm-lint-marker-error') | ||||
|     await expect( | ||||
|       page | ||||
|         .getByText( | ||||
|           'sketch profile must lie entirely on one side of the revolution axis' | ||||
|         ) | ||||
|         .first() | ||||
|     ).toBeVisible() | ||||
|   }) | ||||
|       // Hover over the error to see the error message | ||||
|       await page.hover('.cm-lint-marker-error') | ||||
|       await expect( | ||||
|         page | ||||
|           .getByText( | ||||
|             'Modeling command failed: [ApiError { error_code: InternalEngine, message: "Solid3D revolve failed:  sketch profile must lie entirely on one side of the revolution axis" }]' | ||||
|           ) | ||||
|           .first() | ||||
|       ).toBeVisible() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test('When error is not in view WITH LINTS you can click the badge to scroll to it', async ({ | ||||
|     context, | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     // Load the app with the working starter code | ||||
|     await page.addInitScript((code) => { | ||||
|     await context.addInitScript((code) => { | ||||
|       localStorage.setItem('persistCode', code) | ||||
|     }, TEST_CODE_LONG_WITH_ERROR_OUT_OF_VIEW) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await page.waitForTimeout(1000) | ||||
|  | ||||
| @ -241,32 +231,29 @@ extrude001 = extrude(5, sketch001)` | ||||
| test( | ||||
|   'Opening multiple panes persists when switching projects', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browserName }, testInfo) => { | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     // Setup multiple projects. | ||||
|     const { electronApp, page } = await setupElectron({ | ||||
|       testInfo, | ||||
|       folderSetupFn: async (dir) => { | ||||
|         const routerTemplateDir = join(dir, 'router-template-slate') | ||||
|         const bracketDir = join(dir, 'bracket') | ||||
|         await Promise.all([ | ||||
|           fsp.mkdir(routerTemplateDir, { recursive: true }), | ||||
|           fsp.mkdir(bracketDir, { recursive: true }), | ||||
|         ]) | ||||
|         await Promise.all([ | ||||
|           fsp.copyFile( | ||||
|             executorInputPath('router-template-slate.kcl'), | ||||
|             join(routerTemplateDir, 'main.kcl') | ||||
|           ), | ||||
|           fsp.copyFile( | ||||
|             executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|             join(bracketDir, 'main.kcl') | ||||
|           ), | ||||
|         ]) | ||||
|       }, | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const routerTemplateDir = join(dir, 'router-template-slate') | ||||
|       const bracketDir = join(dir, 'bracket') | ||||
|       await Promise.all([ | ||||
|         fsp.mkdir(routerTemplateDir, { recursive: true }), | ||||
|         fsp.mkdir(bracketDir, { recursive: true }), | ||||
|       ]) | ||||
|       await Promise.all([ | ||||
|         fsp.copyFile( | ||||
|           executorInputPath('router-template-slate.kcl'), | ||||
|           join(routerTemplateDir, 'main.kcl') | ||||
|         ), | ||||
|         fsp.copyFile( | ||||
|           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|           join(bracketDir, 'main.kcl') | ||||
|         ), | ||||
|       ]) | ||||
|     }) | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await test.step('Opening the bracket project should load', async () => { | ||||
|       await expect(page.getByText('bracket')).toBeVisible() | ||||
| @ -309,30 +296,21 @@ test( | ||||
|       await expect(page.locator('#variables-pane')).toBeVisible() | ||||
|       await expect(page.locator('#logs-pane')).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
|  | ||||
| test( | ||||
|   'external change of file contents are reflected in editor', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browserName }, testInfo) => { | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     const PROJECT_DIR_NAME = 'lee-was-here' | ||||
|     const { | ||||
|       electronApp, | ||||
|       page, | ||||
|       dir: projectsDir, | ||||
|     } = await setupElectron({ | ||||
|       testInfo, | ||||
|       folderSetupFn: async (dir) => { | ||||
|         const aProjectDir = join(dir, PROJECT_DIR_NAME) | ||||
|         await fsp.mkdir(aProjectDir, { recursive: true }) | ||||
|       }, | ||||
|     const { dir: projectsDir } = await context.folderSetupFn(async (dir) => { | ||||
|       const aProjectDir = join(dir, PROJECT_DIR_NAME) | ||||
|       await fsp.mkdir(aProjectDir, { recursive: true }) | ||||
|     }) | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await test.step('Open the project', async () => { | ||||
|       await expect(page.getByText(PROJECT_DIR_NAME)).toBeVisible() | ||||
| @ -351,7 +329,5 @@ test( | ||||
|       ) | ||||
|       await u.editorTextMatches(content) | ||||
|     }) | ||||
|  | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
|  | ||||
| @ -1,37 +1,30 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
|  | ||||
| import { getUtils, setup, tearDown } from './test-utils' | ||||
| import { getUtils } from './test-utils' | ||||
| import { KCL_DEFAULT_LENGTH } from 'lib/constants' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.describe('Command bar tests', () => { | ||||
|   test('Extrude from command bar selects extrude line after', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XY') | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> xLine(-20, %) | ||||
|     |> close(%) | ||||
|       ` | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> xLine(-20, %) | ||||
|   |> close(%) | ||||
|     ` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
| @ -52,24 +45,24 @@ test.describe('Command bar tests', () => { | ||||
|     ) | ||||
|   }) | ||||
|  | ||||
|   test('Fillet from command bar', async ({ page }) => { | ||||
|   test('Fillet from command bar', async ({ page, homePage }) => { | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-5, -5], %) | ||||
|   |> line([0, 10], %) | ||||
|   |> line([10, 0], %) | ||||
|   |> line([0, -10], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| extrude001 = extrude(-10, sketch001)` | ||||
|     |> startProfileAt([-5, -5], %) | ||||
|     |> line([0, 10], %) | ||||
|     |> line([10, 0], %) | ||||
|     |> line([0, -10], %) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|   extrude001 = extrude(-10, sketch001)` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
| @ -87,16 +80,16 @@ extrude001 = extrude(-10, sketch001)` | ||||
|     await page.keyboard.press('Enter') // submit | ||||
|     await page.waitForTimeout(100) | ||||
|     await expect(page.locator('.cm-activeLine')).toContainText( | ||||
|       `fillet({ radius = ${KCL_DEFAULT_LENGTH}, tags = [seg01] }, %)` | ||||
|       `fillet({ radius: ${KCL_DEFAULT_LENGTH}, tags: [seg01] }, %)` | ||||
|     ) | ||||
|   }) | ||||
|  | ||||
|   test('Command bar can change a setting, and switch back and forth between arguments', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     const commandBarButton = page.getByRole('button', { name: 'Commands' }) | ||||
|     const cmdSearchBar = page.getByPlaceholder('Search commands') | ||||
| @ -153,7 +146,7 @@ extrude001 = extrude(-10, sketch001)` | ||||
|     // Check that the visibility changed | ||||
|     await expect(paneSelector).not.toBeVisible() | ||||
|  | ||||
|     commandOptionInput = page.getByPlaceholder('off') | ||||
|     commandOptionInput = page.locator('[id="option-input"]') | ||||
|  | ||||
|     // Test case for https://github.com/KittyCAD/modeling-app/issues/2882 | ||||
|     await commandBarButton.click() | ||||
| @ -174,10 +167,10 @@ extrude001 = extrude(-10, sketch001)` | ||||
|  | ||||
|   test('Command bar keybinding works from code editor and can change a setting', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await expect( | ||||
|       page.getByRole('button', { name: 'Start Sketch' }) | ||||
| @ -221,25 +214,25 @@ extrude001 = extrude(-10, sketch001)` | ||||
|     await expect(page.locator('body')).not.toHaveClass(`body-bg dark`) | ||||
|   }) | ||||
|  | ||||
|   test('Can extrude from the command bar', async ({ page }) => { | ||||
|   test('Can extrude from the command bar', async ({ page, homePage }) => { | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `distance = sqrt(20) | ||||
|       sketch001 = startSketchOn('XZ') | ||||
|       |> startProfileAt([-6.95, 10.98], %) | ||||
|       |> line([25.1, 0.41], %) | ||||
|       |> line([0.73, -20.93], %) | ||||
|       |> line([-23.44, 0.52], %) | ||||
|       |> close(%) | ||||
|           ` | ||||
|     sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt([-6.95, 10.98], %) | ||||
|     |> line([25.1, 0.41], %) | ||||
|     |> line([0.73, -20.93], %) | ||||
|     |> line([-23.44, 0.52], %) | ||||
|     |> close(%) | ||||
|         ` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // Make sure the stream is up | ||||
|     await u.openDebugPanel() | ||||
| @ -293,26 +286,19 @@ extrude001 = extrude(-10, sketch001)` | ||||
|     await continueButton.click() | ||||
|     await submitButton.click() | ||||
|  | ||||
|     // Check that the code was updated | ||||
|     await u.waitForCmdReceive('extrude') | ||||
|     // Unfortunately this indentation seems to matter for the test | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `distance = sqrt(20) | ||||
| distance001 = ${KCL_DEFAULT_LENGTH} | ||||
| sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt([-6.95, 10.98], %) | ||||
|     |> line([25.1, 0.41], %) | ||||
|     |> line([0.73, -20.93], %) | ||||
|     |> line([-23.44, 0.52], %) | ||||
|     |> close(%) | ||||
| extrude001 = extrude(distance001, sketch001)`.replace(/(\r\n|\n|\r)/gm, '') // remove newlines | ||||
|  | ||||
|     await expect(page.locator('.cm-content')).toContainText( | ||||
|       'extrude001 = extrude(distance001, sketch001)' | ||||
|     ) | ||||
|   }) | ||||
|  | ||||
|   test('Can switch between sketch tools via command bar', async ({ page }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|   test('Can switch between sketch tools via command bar', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     const sketchButton = page.getByRole('button', { name: 'Start Sketch' }) | ||||
|     const cmdBarButton = page.getByRole('button', { name: 'Commands' }) | ||||
|  | ||||
| @ -1,23 +1,16 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { getUtils, setup, tearDown } from './test-utils' | ||||
| import { test, expect } from './zoo-test' | ||||
| import { getUtils } from './test-utils' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
| test.describe('Copilot ghost text', () => { | ||||
|   // eslint-disable-next-line jest/valid-title | ||||
|   test.skip(true, 'Needs to get covered again') | ||||
|  | ||||
|   test('completes code in empty file', async ({ page }) => { | ||||
|   test('completes code in empty file', async ({ page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
| @ -52,12 +45,13 @@ test.describe('Copilot ghost text', () => { | ||||
|  | ||||
|   test.skip('copilot disabled in sketch mode no select plane', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
| @ -101,12 +95,13 @@ test.describe('Copilot ghost text', () => { | ||||
|  | ||||
|   test('copilot disabled in sketch mode after selecting plane', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
| @ -184,12 +179,12 @@ test.describe('Copilot ghost text', () => { | ||||
|     await expect(page.locator('.cm-ghostText')).not.toBeVisible() | ||||
|   }) | ||||
|  | ||||
|   test('ArrowUp in code rejects the suggestion', async ({ page }) => { | ||||
|   test('ArrowUp in code rejects the suggestion', async ({ page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
| @ -212,12 +207,15 @@ test.describe('Copilot ghost text', () => { | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|   }) | ||||
|  | ||||
|   test('ArrowDown in code rejects the suggestion', async ({ page }) => { | ||||
|   test('ArrowDown in code rejects the suggestion', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
| @ -240,12 +238,15 @@ test.describe('Copilot ghost text', () => { | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|   }) | ||||
|  | ||||
|   test('ArrowLeft in code rejects the suggestion', async ({ page }) => { | ||||
|   test('ArrowLeft in code rejects the suggestion', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
| @ -268,12 +269,15 @@ test.describe('Copilot ghost text', () => { | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|   }) | ||||
|  | ||||
|   test('ArrowRight in code rejects the suggestion', async ({ page }) => { | ||||
|   test('ArrowRight in code rejects the suggestion', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
| @ -296,12 +300,12 @@ test.describe('Copilot ghost text', () => { | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|   }) | ||||
|  | ||||
|   test('Enter in code scoots it down', async ({ page }) => { | ||||
|   test('Enter in code scoots it down', async ({ page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
| @ -326,12 +330,15 @@ test.describe('Copilot ghost text', () => { | ||||
|     ) | ||||
|   }) | ||||
|  | ||||
|   test('Ctrl+shift+z in code rejects the suggestion', async ({ page }) => { | ||||
|   test('Ctrl+shift+z in code rejects the suggestion', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
| @ -360,12 +367,13 @@ test.describe('Copilot ghost text', () => { | ||||
|  | ||||
|   test('Ctrl+z in code rejects the suggestion and undos the last code', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await page.waitForTimeout(800) | ||||
|     await u.codeLocator.click() | ||||
| @ -420,98 +428,107 @@ test.describe('Copilot ghost text', () => { | ||||
|     await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() | ||||
|  | ||||
|     // TODO when we make codemirror a widget, we can test this. | ||||
|     //await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|   }) | ||||
|     //await expect(page.locator('.cm-content')).toHaveText(``) }) | ||||
|  | ||||
|   test('delete in code rejects the suggestion', async ({ page }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     test('delete in code rejects the suggestion', async ({ | ||||
|       page, | ||||
|       homePage, | ||||
|     }) => { | ||||
|       const u = await getUtils(page) | ||||
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|       await u.codeLocator.click() | ||||
|       await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|  | ||||
|     await expect(page.locator('.cm-ghostText')).not.toBeVisible() | ||||
|     await page.waitForTimeout(500) | ||||
|     await page.keyboard.press('Enter') | ||||
|     await page.keyboard.press('Enter') | ||||
|     await page.keyboard.press('Enter') | ||||
|     await expect(page.locator('.cm-ghostText').first()).toBeVisible() | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `fn cube = (pos, scale) => {  sg = startSketchOn('XY')    |> startProfileAt(pos, %)    |> line([0, scale], %)    |> line([scale, 0], %)    |> line([0, -scale], %)  return sg}part001 = cube([0,0], 20)    |> close(%)    |> extrude(20, %)` | ||||
|     ) | ||||
|     await expect(page.locator('.cm-ghostText').first()).toHaveText( | ||||
|       `fn cube = (pos, scale) => {` | ||||
|     ) | ||||
|       await expect(page.locator('.cm-ghostText')).not.toBeVisible() | ||||
|       await page.waitForTimeout(500) | ||||
|       await page.keyboard.press('Enter') | ||||
|       await page.keyboard.press('Enter') | ||||
|       await page.keyboard.press('Enter') | ||||
|       await expect(page.locator('.cm-ghostText').first()).toBeVisible() | ||||
|       await expect(page.locator('.cm-content')).toHaveText( | ||||
|         `fn cube = (pos, scale) => {  sg = startSketchOn('XY')    |> startProfileAt(pos, %)    |> line([0, scale], %)    |> line([scale, 0], %)    |> line([0, -scale], %)  return sg}part001 = cube([0,0], 20)    |> close(%)    |> extrude(20, %)` | ||||
|       ) | ||||
|       await expect(page.locator('.cm-ghostText').first()).toHaveText( | ||||
|         `fn cube = (pos, scale) => {` | ||||
|       ) | ||||
|  | ||||
|     // Going elsewhere in the code should hide the ghost text. | ||||
|     await page.keyboard.press('Delete') | ||||
|     await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() | ||||
|       // Going elsewhere in the code should hide the ghost text. | ||||
|       await page.keyboard.press('Delete') | ||||
|       await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() | ||||
|  | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|   }) | ||||
|       await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|     }) | ||||
|  | ||||
|   test('backspace in code rejects the suggestion', async ({ page }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     test('backspace in code rejects the suggestion', async ({ | ||||
|       page, | ||||
|       homePage, | ||||
|     }) => { | ||||
|       const u = await getUtils(page) | ||||
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|       await u.codeLocator.click() | ||||
|       await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|  | ||||
|     await expect(page.locator('.cm-ghostText')).not.toBeVisible() | ||||
|     await page.waitForTimeout(500) | ||||
|     await page.keyboard.press('Enter') | ||||
|     await page.keyboard.press('Enter') | ||||
|     await page.keyboard.press('Enter') | ||||
|     await expect(page.locator('.cm-ghostText').first()).toBeVisible() | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `fn cube = (pos, scale) => {  sg = startSketchOn('XY')    |> startProfileAt(pos, %)    |> line([0, scale], %)    |> line([scale, 0], %)    |> line([0, -scale], %)  return sg}part001 = cube([0,0], 20)    |> close(%)    |> extrude(20, %)` | ||||
|     ) | ||||
|     await expect(page.locator('.cm-ghostText').first()).toHaveText( | ||||
|       `fn cube = (pos, scale) => {` | ||||
|     ) | ||||
|       await expect(page.locator('.cm-ghostText')).not.toBeVisible() | ||||
|       await page.waitForTimeout(500) | ||||
|       await page.keyboard.press('Enter') | ||||
|       await page.keyboard.press('Enter') | ||||
|       await page.keyboard.press('Enter') | ||||
|       await expect(page.locator('.cm-ghostText').first()).toBeVisible() | ||||
|       await expect(page.locator('.cm-content')).toHaveText( | ||||
|         `fn cube = (pos, scale) => {  sg = startSketchOn('XY')    |> startProfileAt(pos, %)    |> line([0, scale], %)    |> line([scale, 0], %)    |> line([0, -scale], %)  return sg}part001 = cube([0,0], 20)    |> close(%)    |> extrude(20, %)` | ||||
|       ) | ||||
|       await expect(page.locator('.cm-ghostText').first()).toHaveText( | ||||
|         `fn cube = (pos, scale) => {` | ||||
|       ) | ||||
|  | ||||
|     // Going elsewhere in the code should hide the ghost text. | ||||
|     await page.keyboard.press('Backspace') | ||||
|     await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() | ||||
|       // Going elsewhere in the code should hide the ghost text. | ||||
|       await page.keyboard.press('Backspace') | ||||
|       await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() | ||||
|  | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|   }) | ||||
|       await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|     }) | ||||
|  | ||||
|   test('focus outside code pane rejects the suggestion', async ({ page }) => { | ||||
|     const u = await getUtils(page) | ||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     test('focus outside code pane rejects the suggestion', async ({ | ||||
|       page, | ||||
|       homePage, | ||||
|     }) => { | ||||
|       const u = await getUtils(page) | ||||
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|       await u.codeLocator.click() | ||||
|       await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|  | ||||
|     await expect(page.locator('.cm-ghostText')).not.toBeVisible() | ||||
|     await page.waitForTimeout(500) | ||||
|     await page.keyboard.press('Enter') | ||||
|     await expect(page.locator('.cm-ghostText').first()).toBeVisible() | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `fn cube = (pos, scale) => {  sg = startSketchOn('XY')    |> startProfileAt(pos, %)    |> line([0, scale], %)    |> line([scale, 0], %)    |> line([0, -scale], %)  return sg}part001 = cube([0,0], 20)    |> close(%)    |> extrude(20, %)` | ||||
|     ) | ||||
|     await expect(page.locator('.cm-ghostText').first()).toHaveText( | ||||
|       `fn cube = (pos, scale) => {` | ||||
|     ) | ||||
|       await expect(page.locator('.cm-ghostText')).not.toBeVisible() | ||||
|       await page.waitForTimeout(500) | ||||
|       await page.keyboard.press('Enter') | ||||
|       await expect(page.locator('.cm-ghostText').first()).toBeVisible() | ||||
|       await expect(page.locator('.cm-content')).toHaveText( | ||||
|         `fn cube = (pos, scale) => {  sg = startSketchOn('XY')    |> startProfileAt(pos, %)    |> line([0, scale], %)    |> line([scale, 0], %)    |> line([0, -scale], %)  return sg}part001 = cube([0,0], 20)    |> close(%)    |> extrude(20, %)` | ||||
|       ) | ||||
|       await expect(page.locator('.cm-ghostText').first()).toHaveText( | ||||
|         `fn cube = (pos, scale) => {` | ||||
|       ) | ||||
|  | ||||
|     // Going outside the editor should hide the ghost text. | ||||
|     await page.mouse.move(0, 0) | ||||
|     await page | ||||
|       .getByRole('button', { name: 'Start Sketch' }) | ||||
|       .waitFor({ state: 'visible' }) | ||||
|     await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||
|     await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() | ||||
|       // Going outside the editor should hide the ghost text. | ||||
|       await page.mouse.move(0, 0) | ||||
|       await page | ||||
|         .getByRole('button', { name: 'Start Sketch' }) | ||||
|         .waitFor({ state: 'visible' }) | ||||
|       await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||
|       await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() | ||||
|  | ||||
|     await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|       await expect(page.locator('.cm-content')).toHaveText(``) | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -1,14 +1,6 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
|  | ||||
| import { getUtils, setup, tearDown } from './test-utils' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
| import { getUtils } from './test-utils' | ||||
|  | ||||
| function countNewlines(input: string): number { | ||||
|   let count = 0 | ||||
| @ -24,13 +16,14 @@ test.describe('Debug pane', () => { | ||||
|   test('Artifact IDs in the artifact graph are stable across code edits', async ({ | ||||
|     page, | ||||
|     context, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const code = `sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([0, 0], %) | ||||
| |> line([1, 1], %) | ||||
| ` | ||||
|     |> startProfileAt([0, 0], %) | ||||
|   |> line([1, 1], %) | ||||
|   ` | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     const tree = page.getByTestId('debug-feature-tree') | ||||
|     const segment = tree.locator('li', { | ||||
| @ -39,7 +32,7 @@ test.describe('Debug pane', () => { | ||||
|     }) | ||||
|  | ||||
|     await test.step('Test setup', async () => { | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await homePage.goToModelingScene() | ||||
|       await u.openKclCodePanel() | ||||
|       await u.openDebugPanel() | ||||
|       // Set the code in the code editor. | ||||
|  | ||||
| @ -1,39 +1,31 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { join } from 'path' | ||||
| import { test, expect } from './zoo-test' | ||||
| import path from 'path' | ||||
| import { | ||||
|   getUtils, | ||||
|   setupElectron, | ||||
|   tearDown, | ||||
|   executorInputPath, | ||||
|   getPlaywrightDownloadDir, | ||||
| } from './test-utils' | ||||
| import fsp from 'fs/promises' | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test( | ||||
|   'export works on the first try', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browserName }, testInfo) => { | ||||
|     const { electronApp, page } = await setupElectron({ | ||||
|       testInfo, | ||||
|       folderSetupFn: async (dir) => { | ||||
|         const bracketDir = join(dir, 'bracket') | ||||
|         await Promise.all([fsp.mkdir(bracketDir, { recursive: true })]) | ||||
|         await Promise.all([ | ||||
|           fsp.copyFile( | ||||
|             executorInputPath('router-template-slate.kcl'), | ||||
|             join(bracketDir, 'other.kcl') | ||||
|           ), | ||||
|           fsp.copyFile( | ||||
|             executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|             join(bracketDir, 'main.kcl') | ||||
|           ), | ||||
|         ]) | ||||
|       }, | ||||
|   async ({ page, context }, testInfo) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const bracketDir = path.join(dir, 'bracket') | ||||
|       await Promise.all([fsp.mkdir(bracketDir, { recursive: true })]) | ||||
|       await Promise.all([ | ||||
|         fsp.copyFile( | ||||
|           executorInputPath('router-template-slate.kcl'), | ||||
|           path.join(bracketDir, 'other.kcl') | ||||
|         ), | ||||
|         fsp.copyFile( | ||||
|           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|           path.join(bracketDir, 'main.kcl') | ||||
|         ), | ||||
|       ]) | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     page.on('console', console.log) | ||||
|  | ||||
| @ -93,12 +85,16 @@ test( | ||||
|       await expect(successToastMessage).toBeVisible() | ||||
|       await expect(exportingToastMessage).not.toBeVisible() | ||||
|  | ||||
|       const firstFileFullPath = path.resolve( | ||||
|         getPlaywrightDownloadDir(page), | ||||
|         exportFileName | ||||
|       ) | ||||
|       await test.step('Check the export size', async () => { | ||||
|         await expect | ||||
|           .poll( | ||||
|             async () => { | ||||
|               try { | ||||
|                 const outputGltf = await fsp.readFile(exportFileName) | ||||
|                 const outputGltf = await fsp.readFile(firstFileFullPath) | ||||
|                 return outputGltf.byteLength | ||||
|               } catch (e) { | ||||
|                 return 0 | ||||
| @ -107,9 +103,6 @@ test( | ||||
|             { timeout: 15_000 } | ||||
|           ) | ||||
|           .toBeGreaterThan(300_000) | ||||
|  | ||||
|         // clean up exported file | ||||
|         await fsp.rm(exportFileName) | ||||
|       }) | ||||
|     }) | ||||
|  | ||||
| @ -170,12 +163,16 @@ test( | ||||
|           expect(exportingToastMessage).not.toBeVisible(), | ||||
|         ])) | ||||
|  | ||||
|       const secondFileFullPath = path.resolve( | ||||
|         getPlaywrightDownloadDir(page), | ||||
|         exportFileName | ||||
|       ) | ||||
|       await test.step('Check the export size', async () => { | ||||
|         await expect | ||||
|           .poll( | ||||
|             async () => { | ||||
|               try { | ||||
|                 const outputGltf = await fsp.readFile(exportFileName) | ||||
|                 const outputGltf = await fsp.readFile(secondFileFullPath) | ||||
|                 return outputGltf.byteLength | ||||
|               } catch (e) { | ||||
|                 return 0 | ||||
| @ -184,13 +181,7 @@ test( | ||||
|             { timeout: 15_000 } | ||||
|           ) | ||||
|           .toBeGreaterThan(100_000) | ||||
|  | ||||
|         // clean up exported file | ||||
|         await fsp.rm(exportFileName) | ||||
|       }) | ||||
|       await electronApp.close() | ||||
|     }) | ||||
|  | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
| import fsp from 'fs/promises' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import { | ||||
| @ -6,37 +6,27 @@ import { | ||||
|   darkModePlaneColorXZ, | ||||
|   executorInputPath, | ||||
|   getUtils, | ||||
|   setup, | ||||
|   setupElectron, | ||||
|   tearDown, | ||||
| } from './test-utils' | ||||
|  | ||||
| import { join } from 'path' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.describe('Editor tests', () => { | ||||
|   test('can comment out code with ctrl+/', async ({ page }) => { | ||||
|   test('can comment out code with ctrl+/', async ({ page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // check no error to begin with | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await page.keyboard.type(`sketch001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   |> close(%)`) | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> line([-20, 0], %) | ||||
|     |> close(%)`) | ||||
|  | ||||
|     await page.keyboard.down('ControlOrMeta') | ||||
|     await page.keyboard.press('/') | ||||
| @ -44,11 +34,11 @@ test.describe('Editor tests', () => { | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn('XY') | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> line([-20, 0], %) | ||||
|     // |> close(%)`) | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   // |> close(%)`) | ||||
|  | ||||
|     // uncomment the code | ||||
|     await page.keyboard.down('ControlOrMeta') | ||||
| @ -57,61 +47,63 @@ test.describe('Editor tests', () => { | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn('XY') | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> line([-20, 0], %) | ||||
|     |> close(%)`) | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   |> close(%)`) | ||||
|   }) | ||||
|  | ||||
|   test('if you click the format button it formats your code', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // check no error to begin with | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await page.keyboard.type(`sketch001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   |> close(%)`) | ||||
|     await page.locator('#code-pane button:first-child').click() | ||||
|     await page.locator('button:has-text("Format code")').click() | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn('XY') | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> line([-20, 0], %) | ||||
|     |> close(%)`) | ||||
|     await page.locator('#code-pane button:first-child').click() | ||||
|     await page.locator('button:has-text("Format code")').click() | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   |> close(%)`) | ||||
|   }) | ||||
|  | ||||
|   test('if you click the format button it formats your code and executes so lints are still there', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // check no error to begin with | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|  | ||||
|     await u.codeLocator.click() | ||||
|     await page.keyboard.type(`sketch_001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   |> close(%)`) | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> line([-20, 0], %) | ||||
|     |> close(%)`) | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
| @ -135,11 +127,11 @@ test.describe('Editor tests', () => { | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch_001 = startSketchOn('XY') | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> line([-20, 0], %) | ||||
|     |> close(%)`) | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   |> close(%)`) | ||||
|  | ||||
|     // error in guter | ||||
|     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible() | ||||
| @ -151,29 +143,27 @@ test.describe('Editor tests', () => { | ||||
|     ).toBeVisible() | ||||
|   }) | ||||
|  | ||||
|   test('fold gutters work', async ({ page }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|   test('fold gutters work', async ({ page, homePage }) => { | ||||
|     const fullCode = `sketch001 = startSketchOn('XY') | ||||
|      |> startProfileAt([-10, -10], %) | ||||
|      |> line([20, 0], %) | ||||
|      |> line([0, 20], %) | ||||
|      |> line([-20, 0], %) | ||||
|      |> close(%)` | ||||
|    |> startProfileAt([-10, -10], %) | ||||
|    |> line([20, 0], %) | ||||
|    |> line([0, 20], %) | ||||
|    |> line([-20, 0], %) | ||||
|    |> close(%)` | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XY') | ||||
|      |> startProfileAt([-10, -10], %) | ||||
|      |> line([20, 0], %) | ||||
|      |> line([0, 20], %) | ||||
|      |> line([-20, 0], %) | ||||
|      |> close(%)` | ||||
|    |> startProfileAt([-10, -10], %) | ||||
|    |> line([20, 0], %) | ||||
|    |> line([0, 20], %) | ||||
|    |> line([-20, 0], %) | ||||
|    |> close(%)` | ||||
|       ) | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // TODO: Jess needs to fix this but you have to mod the code to get them to show | ||||
|     // up, its an annoying codemirror thing. | ||||
| @ -224,22 +214,25 @@ test.describe('Editor tests', () => { | ||||
|     await expect(foldGutterFoldLine).not.toBeVisible() | ||||
|   }) | ||||
|  | ||||
|   test('hover over functions shows function description', async ({ page }) => { | ||||
|   test('hover over functions shows function description', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   |> close(%)` | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> line([-20, 0], %) | ||||
|     |> close(%)` | ||||
|       ) | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // check no error to begin with | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
| @ -268,23 +261,24 @@ test.describe('Editor tests', () => { | ||||
|  | ||||
|   test('if you use the format keyboard binding it formats your code', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   |> close(%)` | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> line([-20, 0], %) | ||||
|     |> close(%)` | ||||
|       ) | ||||
|       localStorage.setItem('disableAxis', 'true') | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // check no error to begin with | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
| @ -301,32 +295,33 @@ test.describe('Editor tests', () => { | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn('XY') | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> line([-20, 0], %) | ||||
|     |> close(%)`) | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   |> close(%)`) | ||||
|   }) | ||||
|  | ||||
|   test('if you use the format keyboard binding it formats your code and executes so lints are shown', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch_001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   |> close(%)` | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> line([-20, 0], %) | ||||
|     |> close(%)` | ||||
|       ) | ||||
|       localStorage.setItem('disableAxis', 'true') | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
| @ -353,11 +348,11 @@ test.describe('Editor tests', () => { | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch_001 = startSketchOn('XY') | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> line([-20, 0], %) | ||||
|     |> close(%)`) | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> line([-20, 0], %) | ||||
|   |> close(%)`) | ||||
|  | ||||
|     // error in guter | ||||
|     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible() | ||||
| @ -369,11 +364,14 @@ test.describe('Editor tests', () => { | ||||
|     ).toBeVisible() | ||||
|   }) | ||||
|  | ||||
|   test('if you write kcl with lint errors you get lints', async ({ page }) => { | ||||
|   test('if you write kcl with lint errors you get lints', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // check no error to begin with | ||||
|     await expect(page.locator('.cm-lint-marker-info')).not.toBeVisible() | ||||
| @ -409,23 +407,26 @@ test.describe('Editor tests', () => { | ||||
|     await expect(page.locator('.cm-lint-marker-info')).not.toBeVisible() | ||||
|   }) | ||||
|  | ||||
|   test('if you fixup kcl errors you clear lints', async ({ page }) => { | ||||
|   test('if you fixup kcl errors you clear lints', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([3.29, 7.86], %) | ||||
|   |> line([2.48, 2.44], %) | ||||
|   |> line([2.66, 1.17], %) | ||||
|   |> close(%) | ||||
|   ` | ||||
|     |> startProfileAt([3.29, 7.86], %) | ||||
|     |> line([2.48, 2.44], %) | ||||
|     |> line([2.66, 1.17], %) | ||||
|     |> close(%) | ||||
|     ` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // check no error to begin with | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
| @ -447,20 +448,23 @@ test.describe('Editor tests', () => { | ||||
|     ).not.toBeVisible() | ||||
|   }) | ||||
|  | ||||
|   test('if you write invalid kcl you get inlined errors', async ({ page }) => { | ||||
|   test('if you write invalid kcl you get inlined errors', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // check no error to begin with | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|  | ||||
|     /* add the following code to the editor ($ error is not a valid line) | ||||
|       $ error | ||||
|       const topAng = 30 | ||||
|       const bottomAng = 25 | ||||
|      */ | ||||
|     $ error | ||||
|     const topAng = 30 | ||||
|     const bottomAng = 25 | ||||
|    */ | ||||
|     await u.codeLocator.click() | ||||
|     await page.keyboard.type('$ error') | ||||
|  | ||||
| @ -518,106 +522,108 @@ test.describe('Editor tests', () => { | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|   }) | ||||
|  | ||||
|   // TODO currently multiple source ranges are not supported | ||||
|   test.skip('error with 2 source ranges gets 2 diagnostics', async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `length = .750 | ||||
|   width = 0.500 | ||||
|   height = 0.500 | ||||
|   dia = 4 | ||||
|   test.fixme( | ||||
|     'error with 2 source ranges gets 2 diagnostics', | ||||
|     async ({ page, homePage }) => { | ||||
|       const u = await getUtils(page) | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `length = .750 | ||||
|     width = 0.500 | ||||
|     height = 0.500 | ||||
|     dia = 4 | ||||
|    | ||||
|     fn squareHole = (l, w) => { | ||||
|   squareHoleSketch = startSketchOn('XY') | ||||
|   |> startProfileAt([-width / 2, -length / 2], %) | ||||
|   |> lineTo([width / 2, -length / 2], %) | ||||
|   |> lineTo([width / 2, length / 2], %) | ||||
|   |> lineTo([-width / 2, length / 2], %) | ||||
|   |> close(%) | ||||
|   return squareHoleSketch | ||||
|     } | ||||
|     ` | ||||
|         ) | ||||
|       }) | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|   fn squareHole = (l, w) => { | ||||
|     squareHoleSketch = startSketchOn('XY') | ||||
|     |> startProfileAt([-width / 2, -length / 2], %) | ||||
|     |> lineTo([width / 2, -length / 2], %) | ||||
|     |> lineTo([width / 2, length / 2], %) | ||||
|     |> lineTo([-width / 2, length / 2], %) | ||||
|     |> close(%) | ||||
|     return squareHoleSketch | ||||
|   } | ||||
|   ` | ||||
|       ) | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await u.waitForPageLoad() | ||||
|       await page.waitForTimeout(1000) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|       await u.openDebugPanel() | ||||
|       await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|       await u.closeDebugPanel() | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|       // check no error to begin with | ||||
|       await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|  | ||||
|     // check no error to begin with | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|       // Click on the bottom of the code editor to add a new line | ||||
|       await u.codeLocator.click() | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('ArrowDown') | ||||
|       await page.keyboard.press('Enter') | ||||
|       await page.keyboard.type(`extrusion = startSketchOn('XY') | ||||
|   |> circle({ center: [0, 0], radius: dia/2 }, %) | ||||
|     |> hole(squareHole(length, width, height), %) | ||||
|     |> extrude(height, %)`) | ||||
|  | ||||
|     // Click on the bottom of the code editor to add a new line | ||||
|     await u.codeLocator.click() | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('Enter') | ||||
|     await page.keyboard.type(`extrusion = startSketchOn('XY') | ||||
|     |> circle({ center = [0, 0], radius = dia/2 }, %) | ||||
|   |> hole(squareHole(length, width, height), %) | ||||
|   |> extrude(height, %)`) | ||||
|       // error in gutter | ||||
|       await expect(page.locator('.cm-lint-marker-error').first()).toBeVisible() | ||||
|       await page.hover('.cm-lint-marker-error:first-child') | ||||
|       await expect( | ||||
|         page.getByText('Expected 2 arguments, got 3').first() | ||||
|       ).toBeVisible() | ||||
|  | ||||
|     // error in gutter | ||||
|     await expect(page.locator('.cm-lint-marker-error').first()).toBeVisible() | ||||
|     await page.hover('.cm-lint-marker-error:first-child') | ||||
|     await expect( | ||||
|       page.getByText('Expected 2 arguments, got 3').first() | ||||
|     ).toBeVisible() | ||||
|  | ||||
|     // Make sure there are two diagnostics | ||||
|     await expect(page.locator('.cm-lint-marker-error')).toHaveCount(2) | ||||
|   }) | ||||
|       // Make sure there are two diagnostics | ||||
|       await expect(page.locator('.cm-lint-marker-error')).toHaveCount(2) | ||||
|     } | ||||
|   ) | ||||
|   test('if your kcl gets an error from the engine it is inlined', async ({ | ||||
|     context, | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|     await context.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `box = startSketchOn('XY') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([0, 10], %) | ||||
|   |> line([10, 0], %) | ||||
|   |> line([0, -10], %, $revolveAxis) | ||||
|   |> close(%) | ||||
|   |> extrude(10, %) | ||||
|  | ||||
|   sketch001 = startSketchOn(box, revolveAxis) | ||||
|   |> startProfileAt([5, 10], %) | ||||
|   |> line([0, -10], %) | ||||
|   |> line([2, 0], %) | ||||
|   |> line([0, -10], %) | ||||
|   |> close(%) | ||||
|   |> revolve({ | ||||
|   axis = revolveAxis, | ||||
|   angle = 90 | ||||
|   }, %) | ||||
|       ` | ||||
|     |> startProfileAt([0, 0], %) | ||||
|     |> line([0, 10], %) | ||||
|     |> line([10, 0], %) | ||||
|     |> line([0, -10], %, $revolveAxis) | ||||
|     |> close(%) | ||||
|     |> extrude(10, %) | ||||
|    | ||||
|     sketch001 = startSketchOn(box, revolveAxis) | ||||
|     |> startProfileAt([5, 10], %) | ||||
|     |> line([0, -10], %) | ||||
|     |> line([2, 0], %) | ||||
|     |> line([0, -10], %) | ||||
|     |> close(%) | ||||
|     |> revolve({ | ||||
|     axis: revolveAxis, | ||||
|     angle: 90 | ||||
|     }, %) | ||||
|     ` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await page.goto('/') | ||||
|     await u.waitForPageLoad() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await expect(page.locator('.cm-lint-marker-error')).toBeVisible() | ||||
|  | ||||
| @ -628,12 +634,15 @@ test.describe('Editor tests', () => { | ||||
|     await expect(page.getByText(searchText)).toBeVisible() | ||||
|   }) | ||||
|   test.describe('Autocomplete works', () => { | ||||
|     test('with enter/click to accept the completion', async ({ page }) => { | ||||
|     test('with enter/click to accept the completion', async ({ | ||||
|       page, | ||||
|       homePage, | ||||
|     }) => { | ||||
|       const u = await getUtils(page) | ||||
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|       // tests clicking on an option, selection the first option | ||||
|       // and arrowing down to an option | ||||
| @ -695,19 +704,19 @@ test.describe('Editor tests', () => { | ||||
|  | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt([3.14, 12], %) | ||||
|     |> xLine(5, %) // lin`) | ||||
|         |> startProfileAt([3.14, 12], %) | ||||
|         |> xLine(5, %) // lin`) | ||||
|  | ||||
|       // expect there to be no KCL errors | ||||
|       await expect(page.locator('.cm-lint-marker-error')).toHaveCount(0) | ||||
|     }) | ||||
|  | ||||
|     test('with tab to accept the completion', async ({ page }) => { | ||||
|     test('with tab to accept the completion', async ({ page, homePage }) => { | ||||
|       const u = await getUtils(page) | ||||
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|       // this test might be brittle as we add and remove functions | ||||
|       // but should also be easy to update. | ||||
| @ -769,26 +778,30 @@ test.describe('Editor tests', () => { | ||||
|  | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt([3.14, 12], %) | ||||
|     |> xLine(5, %) // lin`) | ||||
|         |> startProfileAt([3.14, 12], %) | ||||
|         |> xLine(5, %) // lin`) | ||||
|     }) | ||||
|   }) | ||||
|   test('Can undo a click and point extrude with ctrl+z', async ({ page }) => { | ||||
|   test('Can undo a click and point extrude with ctrl+z', async ({ | ||||
|     page, | ||||
|     context, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|     await context.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt([4.61, -14.01], %) | ||||
|     |> line([12.73, -0.09], %) | ||||
|     |> tangentialArcTo([24.95, -5.38], %) | ||||
|     |> close(%)` | ||||
|   |> startProfileAt([4.61, -14.01], %) | ||||
|   |> line([12.73, -0.09], %) | ||||
|   |> tangentialArcTo([24.95, -5.38], %) | ||||
|   |> close(%)` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await expect( | ||||
|       page.getByRole('button', { name: 'Start Sketch' }) | ||||
|     ).not.toBeDisabled() | ||||
| @ -841,29 +854,32 @@ test.describe('Editor tests', () => { | ||||
|     await page.waitForTimeout(100) | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt([4.61, -14.01], %) | ||||
|     |> line([12.73, -0.09], %) | ||||
|     |> tangentialArcTo([24.95, -5.38], %) | ||||
|     |> close(%)`) | ||||
|   |> startProfileAt([4.61, -14.01], %) | ||||
|   |> line([12.73, -0.09], %) | ||||
|   |> tangentialArcTo([24.95, -5.38], %) | ||||
|   |> close(%)`) | ||||
|   }) | ||||
|  | ||||
|   test('Can undo a sketch modification with ctrl+z', async ({ page }) => { | ||||
|   test('Can undo a sketch modification with ctrl+z', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt([4.61, -10.01], %) | ||||
|     |> line([12.73, -0.09], %) | ||||
|     |> tangentialArcTo([24.95, -0.38], %) | ||||
|     |> close(%) | ||||
|     |> extrude(5, %)` | ||||
|   |> startProfileAt([4.61, -10.01], %) | ||||
|   |> line([12.73, -0.09], %) | ||||
|   |> tangentialArcTo([24.95, -0.38], %) | ||||
|   |> close(%) | ||||
|   |> extrude(5, %)` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await expect( | ||||
|       page.getByRole('button', { name: 'Start Sketch' }) | ||||
|     ).not.toBeDisabled() | ||||
| @ -890,7 +906,7 @@ test.describe('Editor tests', () => { | ||||
|     }) | ||||
|     await page.waitForTimeout(100) | ||||
|  | ||||
|     const startPX = [665, 397] | ||||
|     const startPX = [1200 / 2, 500 / 2] | ||||
|  | ||||
|     const dragPX = 40 | ||||
|  | ||||
| @ -904,9 +920,9 @@ test.describe('Editor tests', () => { | ||||
|  | ||||
|     await expect(page.getByTestId('segment-overlay')).toHaveCount(2) | ||||
|  | ||||
|     // drag startProfieAt handle | ||||
|     // drag startProfileAt handle | ||||
|     await page.dragAndDrop('#stream', '#stream', { | ||||
|       sourcePosition: { x: startPX[0], y: startPX[1] }, | ||||
|       sourcePosition: { x: startPX[0] + 68, y: startPX[1] + 147 }, | ||||
|       targetPosition: { x: startPX[0] + dragPX, y: startPX[1] + dragPX }, | ||||
|     }) | ||||
|     await page.waitForTimeout(100) | ||||
| @ -944,12 +960,12 @@ test.describe('Editor tests', () => { | ||||
|     // expect the code to have changed | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([7.12, -12.68], %) | ||||
|   |> line([15.39, -2.78], %) | ||||
|   |> tangentialArcTo([27.6, -3.05], %) | ||||
|   |> close(%) | ||||
|   |> extrude(5, %) | ||||
| `) | ||||
|     |> startProfileAt([2.71, -2.71], %) | ||||
|     |> line([15.4, -2.78], %) | ||||
|     |> tangentialArcTo([27.6, -3.05], %) | ||||
|     |> close(%) | ||||
|     |> extrude(5, %) | ||||
|   `) | ||||
|  | ||||
|     // Hit undo | ||||
|     await page.keyboard.down('Control') | ||||
| @ -958,11 +974,11 @@ test.describe('Editor tests', () => { | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([7.12, -12.68], %) | ||||
|   |> line([15.39, -2.78], %) | ||||
|   |> tangentialArcTo([24.95, -0.38], %) | ||||
|   |> close(%) | ||||
|   |> extrude(5, %)`) | ||||
|     |> startProfileAt([2.71, -2.71], %) | ||||
|     |> line([15.4, -2.78], %) | ||||
|     |> tangentialArcTo([24.95, -0.38], %) | ||||
|     |> close(%) | ||||
|     |> extrude(5, %)`) | ||||
|  | ||||
|     // Hit undo again. | ||||
|     await page.keyboard.down('Control') | ||||
| @ -971,12 +987,12 @@ test.describe('Editor tests', () => { | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([7.12, -12.68], %) | ||||
|   |> line([12.73, -0.09], %) | ||||
|   |> tangentialArcTo([24.95, -0.38], %) | ||||
|   |> close(%) | ||||
|   |> extrude(5, %) | ||||
| `) | ||||
|     |> startProfileAt([2.71, -2.71], %) | ||||
|     |> line([12.73, -0.09], %) | ||||
|     |> tangentialArcTo([24.95, -0.38], %) | ||||
|     |> close(%) | ||||
|     |> extrude(5, %) | ||||
|   `) | ||||
|  | ||||
|     // Hit undo again. | ||||
|     await page.keyboard.down('Control') | ||||
| @ -986,31 +1002,29 @@ test.describe('Editor tests', () => { | ||||
|     await page.waitForTimeout(100) | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt([4.61, -10.01], %) | ||||
|     |> line([12.73, -0.09], %) | ||||
|     |> tangentialArcTo([24.95, -0.38], %) | ||||
|     |> close(%) | ||||
|     |> extrude(5, %)`) | ||||
|   |> startProfileAt([4.61, -10.01], %) | ||||
|   |> line([12.73, -0.09], %) | ||||
|   |> tangentialArcTo([24.95, -0.38], %) | ||||
|   |> close(%) | ||||
|   |> extrude(5, %)`) | ||||
|   }) | ||||
|  | ||||
|   test.fixme( | ||||
|     `Can use the import stdlib function on a local OBJ file`, | ||||
|     { tag: '@electron' }, | ||||
|     async ({ browserName }, testInfo) => { | ||||
|       const { electronApp, page } = await setupElectron({ | ||||
|         testInfo, | ||||
|         folderSetupFn: async (dir) => { | ||||
|           const bracketDir = join(dir, 'cube') | ||||
|           await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|           await fsp.copyFile( | ||||
|             executorInputPath('cube.obj'), | ||||
|             join(bracketDir, 'cube.obj') | ||||
|           ) | ||||
|           await fsp.writeFile(join(bracketDir, 'main.kcl'), '') | ||||
|         }, | ||||
|     async ({ page, context }, testInfo) => { | ||||
|       await context.folderSetupFn(async (dir) => { | ||||
|         const bracketDir = join(dir, 'cube') | ||||
|         await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           executorInputPath('cube.obj'), | ||||
|           join(bracketDir, 'cube.obj') | ||||
|         ) | ||||
|         await fsp.writeFile(join(bracketDir, 'main.kcl'), '') | ||||
|       }) | ||||
|  | ||||
|       const viewportSize = { width: 1200, height: 500 } | ||||
|       await page.setViewportSize(viewportSize) | ||||
|       await page.setBodyDimensions(viewportSize) | ||||
|  | ||||
|       // Locators and constants | ||||
|       const u = await getUtils(page) | ||||
| @ -1068,8 +1082,6 @@ test.describe('Editor tests', () => { | ||||
|           }) | ||||
|           .toBeGreaterThan(15) | ||||
|       }) | ||||
|  | ||||
|       await electronApp.close() | ||||
|     } | ||||
|   ) | ||||
| }) | ||||
|  | ||||
| @ -29,7 +29,7 @@ export class EditorFixture { | ||||
|   reConstruct = (page: Page) => { | ||||
|     this.page = page | ||||
|  | ||||
|     this.codeContent = page.locator('.cm-content') | ||||
|     this.codeContent = page.locator('.cm-content[data-language="kcl"]') | ||||
|     this.diagnosticsTooltip = page.locator('.cm-tooltip-lint') | ||||
|     this.diagnosticsGutterIcon = page.locator('.cm-lint-marker-error') | ||||
|     this.activeLine = this.page.locator('.cm-activeLine') | ||||
| @ -54,13 +54,13 @@ export class EditorFixture { | ||||
|         } | ||||
|       } | ||||
|       if (!shouldNormalise) { | ||||
|         const expectStart = expect(this.codeContent) | ||||
|         const expectStart = expect.poll(() => this.codeContent.textContent()) | ||||
|         if (not) { | ||||
|           const result = await expectStart.not.toContainText(code, { timeout }) | ||||
|           const result = await expectStart.not.toContain(code) | ||||
|           await resetPane() | ||||
|           return result | ||||
|         } | ||||
|         const result = await expectStart.toContainText(code, { timeout }) | ||||
|         const result = await expectStart.toContain(code) | ||||
|         await resetPane() | ||||
|         return result | ||||
|       } | ||||
| @ -147,4 +147,20 @@ export class EditorFixture { | ||||
|   openPane() { | ||||
|     return openPane(this.page, this.paneButtonTestId) | ||||
|   } | ||||
|   scrollToText(text: string) { | ||||
|     return this.page.evaluate((scrollToText: string) => { | ||||
|       // editorManager is available on the window object. | ||||
|       // @ts-ignore | ||||
|       let index = editorManager._editorView.docView.view.state.doc | ||||
|         .toString() | ||||
|         .indexOf(scrollToText) | ||||
|       // @ts-ignore | ||||
|       editorManager._editorView.dispatch({ | ||||
|         selection: { | ||||
|           anchor: index, | ||||
|         }, | ||||
|         scrollIntoView: true, | ||||
|       }) | ||||
|     }, text) | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -1,11 +1,11 @@ | ||||
| import type { | ||||
|   BrowserContext, | ||||
|   ElectronApplication, | ||||
|   Page, | ||||
|   TestInfo, | ||||
|   Page, | ||||
| } from '@playwright/test' | ||||
| import { test as base } from '@playwright/test' | ||||
| import { getUtils, setup, setupElectron, tearDown } from '../test-utils' | ||||
|  | ||||
| import { getUtils, setup, setupElectron } from '../test-utils' | ||||
| import fsp from 'fs/promises' | ||||
| import { join } from 'path' | ||||
| import { CmdBarFixture } from './cmdBarFixture' | ||||
| @ -20,11 +20,11 @@ export class AuthenticatedApp { | ||||
|   public readonly page: Page | ||||
|   public readonly context: BrowserContext | ||||
|   public readonly testInfo: TestInfo | ||||
|   public readonly viewPortSize = { width: 1000, height: 500 } | ||||
|   public readonly viewPortSize = { width: 1200, height: 500 } | ||||
|  | ||||
|   constructor(context: BrowserContext, page: Page, testInfo: TestInfo) { | ||||
|     this.page = page | ||||
|     this.context = context | ||||
|     this.page = page | ||||
|     this.testInfo = testInfo | ||||
|   } | ||||
|  | ||||
| @ -49,9 +49,7 @@ export class AuthenticatedApp { | ||||
|   } | ||||
| } | ||||
|  | ||||
| interface Fixtures { | ||||
|   app: AuthenticatedApp | ||||
|   tronApp: AuthenticatedTronApp | ||||
| export interface Fixtures { | ||||
|   cmdBar: CmdBarFixture | ||||
|   editor: EditorFixture | ||||
|   toolbar: ToolbarFixture | ||||
| @ -61,9 +59,11 @@ interface Fixtures { | ||||
| export class AuthenticatedTronApp { | ||||
|   public readonly _page: Page | ||||
|   public page: Page | ||||
|   public readonly context: BrowserContext | ||||
|   public context: BrowserContext | ||||
|   public readonly testInfo: TestInfo | ||||
|   public electronApp?: ElectronApplication | ||||
|   public readonly viewPortSize = { width: 1200, height: 500 } | ||||
|   public dir: string = '' | ||||
|  | ||||
|   constructor(context: BrowserContext, page: Page, testInfo: TestInfo) { | ||||
|     this._page = page | ||||
| @ -79,15 +79,22 @@ export class AuthenticatedTronApp { | ||||
|       appSettings?: Partial<SaveSettingsPayload> | ||||
|     } = { fixtures: {} } | ||||
|   ) { | ||||
|     const { electronApp, page } = await setupElectron({ | ||||
|     const { electronApp, page, context, dir, options } = await setupElectron({ | ||||
|       testInfo: this.testInfo, | ||||
|       folderSetupFn: arg.folderSetupFn, | ||||
|       cleanProjectDir: arg.cleanProjectDir, | ||||
|       appSettings: arg.appSettings, | ||||
|     }) | ||||
|     this.page = page | ||||
|     this.context = context | ||||
|     this.electronApp = electronApp | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     this.dir = dir | ||||
|  | ||||
|     // Easier to access throughout utils | ||||
|     this.page.dir = dir | ||||
|  | ||||
|     // Setup localStorage, addCookies, reload | ||||
|     await setup(this.context, this.page, this.testInfo) | ||||
|  | ||||
|     for (const key of unsafeTypedKeys(arg.fixtures)) { | ||||
|       const fixture = arg.fixtures[key] | ||||
| @ -110,32 +117,20 @@ export class AuthenticatedTronApp { | ||||
|     }) | ||||
| } | ||||
|  | ||||
| export const test = base.extend<Fixtures>({ | ||||
|   app: async ({ page, context }, use, testInfo) => { | ||||
|     await use(new AuthenticatedApp(context, page, testInfo)) | ||||
|   }, | ||||
|   tronApp: async ({ page, context }, use, testInfo) => { | ||||
|     await use(new AuthenticatedTronApp(context, page, testInfo)) | ||||
|   }, | ||||
|   cmdBar: async ({ page }, use) => { | ||||
| export const fixtures = { | ||||
|   cmdBar: async ({ page }: { page: Page }, use: any) => { | ||||
|     await use(new CmdBarFixture(page)) | ||||
|   }, | ||||
|   editor: async ({ page }, use) => { | ||||
|   editor: async ({ page }: { page: Page }, use: any) => { | ||||
|     await use(new EditorFixture(page)) | ||||
|   }, | ||||
|   toolbar: async ({ page }, use) => { | ||||
|   toolbar: async ({ page }: { page: Page }, use: any) => { | ||||
|     await use(new ToolbarFixture(page)) | ||||
|   }, | ||||
|   scene: async ({ page }, use) => { | ||||
|   scene: async ({ page }: { page: Page }, use: any) => { | ||||
|     await use(new SceneFixture(page)) | ||||
|   }, | ||||
|   homePage: async ({ page }, use) => { | ||||
|   homePage: async ({ page }: { page: Page }, use: any) => { | ||||
|     await use(new HomePageFixture(page)) | ||||
|   }, | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| export { expect } from '@playwright/test' | ||||
| } | ||||
|  | ||||
| @ -14,10 +14,14 @@ interface HomePageState { | ||||
| export class HomePageFixture { | ||||
|   public page: Page | ||||
|  | ||||
|   projectSection!: Locator | ||||
|   projectCard!: Locator | ||||
|   projectCardTitle!: Locator | ||||
|   projectCardFile!: Locator | ||||
|   projectCardFolder!: Locator | ||||
|   projectButtonNew!: Locator | ||||
|   projectButtonContinue!: Locator | ||||
|   projectTextName!: Locator | ||||
|   sortByDateBtn!: Locator | ||||
|   sortByNameBtn!: Locator | ||||
|  | ||||
| @ -28,11 +32,19 @@ export class HomePageFixture { | ||||
|   reConstruct = (page: Page) => { | ||||
|     this.page = page | ||||
|  | ||||
|     this.projectSection = this.page.getByTestId('home-section') | ||||
|  | ||||
|     this.projectCard = this.page.getByTestId('project-link') | ||||
|     this.projectCardTitle = this.page.getByTestId('project-title') | ||||
|     this.projectCardFile = this.page.getByTestId('project-file-count') | ||||
|     this.projectCardFolder = this.page.getByTestId('project-folder-count') | ||||
|  | ||||
|     this.projectButtonNew = this.page.getByTestId('home-new-file') | ||||
|     this.projectTextName = this.page.getByTestId('cmd-bar-arg-value') | ||||
|     this.projectButtonContinue = this.page.getByRole('button', { | ||||
|       name: 'Continue', | ||||
|     }) | ||||
|  | ||||
|     this.sortByDateBtn = this.page.getByTestId('home-sort-by-modified') | ||||
|     this.sortByNameBtn = this.page.getByTestId('home-sort-by-name') | ||||
|   } | ||||
| @ -91,10 +103,25 @@ export class HomePageFixture { | ||||
|       .toEqual(expectedState) | ||||
|   } | ||||
|  | ||||
|   createAndGoToProject = async (projectTitle: string) => { | ||||
|     await expect(this.projectSection).not.toHaveText('Loading your Projects...') | ||||
|     await this.projectButtonNew.click() | ||||
|     await this.projectTextName.click() | ||||
|     await this.projectTextName.fill(projectTitle) | ||||
|     await this.projectButtonContinue.click() | ||||
|   } | ||||
|  | ||||
|   openProject = async (projectTitle: string) => { | ||||
|     const projectCard = this.projectCard.locator( | ||||
|       this.page.getByText(projectTitle) | ||||
|     ) | ||||
|     await projectCard.click() | ||||
|   } | ||||
|  | ||||
|   goToModelingScene = async (name: string = 'testDefault') => { | ||||
|     // On web this is a no-op. There is no project view. | ||||
|     if (process.env.PLATFORM === 'web') return | ||||
|  | ||||
|     await this.createAndGoToProject(name) | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -53,8 +53,9 @@ export class SceneFixture { | ||||
|  | ||||
|   expectState = async (expected: SceneSerialised) => { | ||||
|     return expect | ||||
|       .poll(() => this._serialiseScene(), { | ||||
|         message: `Expected scene state to match`, | ||||
|       .poll(async () => await this._serialiseScene(), { | ||||
|         intervals: [1_000, 2_000, 10_000], | ||||
|         timeout: 60000, | ||||
|       }) | ||||
|       .toEqual(expected) | ||||
|   } | ||||
| @ -187,7 +188,10 @@ export class SceneFixture { | ||||
|         type: 'default_camera_get_settings', | ||||
|       }, | ||||
|     }) | ||||
|     await this.waitForExecutionDone() | ||||
|     await this.page | ||||
|       .locator(`[data-receive-command-type="default_camera_get_settings"]`) | ||||
|       .first() | ||||
|       .waitFor() | ||||
|     const position = await Promise.all([ | ||||
|       this.page.getByTestId('cam-x-position').inputValue().then(Number), | ||||
|       this.page.getByTestId('cam-y-position').inputValue().then(Number), | ||||
| @ -238,6 +242,7 @@ export class SceneFixture { | ||||
|   } | ||||
|  | ||||
|   async clickGizmoMenuItem(name: string) { | ||||
|     await this.gizmo.hover() | ||||
|     await this.gizmo.click({ button: 'right' }) | ||||
|     const buttonToTest = this.page.getByRole('button', { | ||||
|       name: name, | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| import type { Page, Locator } from '@playwright/test' | ||||
| import { expect } from './fixtureSetup' | ||||
| import { expect } from '../zoo-test' | ||||
| import { doAndWaitForImageDiff } from '../test-utils' | ||||
|  | ||||
| export class ToolbarFixture { | ||||
| @ -7,6 +7,7 @@ export class ToolbarFixture { | ||||
|  | ||||
|   extrudeButton!: Locator | ||||
|   loftButton!: Locator | ||||
|   shellButton!: Locator | ||||
|   offsetPlaneButton!: Locator | ||||
|   startSketchBtn!: Locator | ||||
|   lineBtn!: Locator | ||||
| @ -28,6 +29,7 @@ export class ToolbarFixture { | ||||
|     this.page = page | ||||
|     this.extrudeButton = page.getByTestId('extrude') | ||||
|     this.loftButton = page.getByTestId('loft') | ||||
|     this.shellButton = page.getByTestId('shell') | ||||
|     this.offsetPlaneButton = page.getByTestId('plane-offset') | ||||
|     this.startSketchBtn = page.getByTestId('sketch') | ||||
|     this.lineBtn = page.getByTestId('line') | ||||
|  | ||||
| @ -1,29 +1,22 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { setupElectron, tearDown, executorInputPath } from './test-utils' | ||||
| import { test, expect } from './zoo-test' | ||||
| import { executorInputPath } from './test-utils' | ||||
| import { join } from 'path' | ||||
| import fsp from 'fs/promises' | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test( | ||||
|   'When machine-api server not found butt is disabled and shows the reason', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browserName }, testInfo) => { | ||||
|     const { electronApp, page } = await setupElectron({ | ||||
|       testInfo, | ||||
|       folderSetupFn: async (dir) => { | ||||
|         const bracketDir = join(dir, 'bracket') | ||||
|         await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|           join(bracketDir, 'main.kcl') | ||||
|         ) | ||||
|       }, | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const bracketDir = join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|       await fsp.copyFile( | ||||
|         executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|         join(bracketDir, 'main.kcl') | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await expect(page.getByText('bracket')).toBeVisible() | ||||
|  | ||||
| @ -47,28 +40,23 @@ test( | ||||
|     // that the machine-api server is not found | ||||
|     await makeButton.hover() | ||||
|     await expect(page.getByText(notFoundText).first()).toBeVisible() | ||||
|  | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
|  | ||||
| test( | ||||
|   'When machine-api server not found home screen & project status shows the reason', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browserName }, testInfo) => { | ||||
|     const { electronApp, page } = await setupElectron({ | ||||
|       testInfo, | ||||
|       folderSetupFn: async (dir) => { | ||||
|         const bracketDir = join(dir, 'bracket') | ||||
|         await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|           join(bracketDir, 'main.kcl') | ||||
|         ) | ||||
|       }, | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const bracketDir = join(dir, 'bracket') | ||||
|       await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|       await fsp.copyFile( | ||||
|         executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|         join(bracketDir, 'main.kcl') | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     const notFoundText = 'Machine API server was not discovered' | ||||
|  | ||||
| @ -91,7 +79,5 @@ test( | ||||
|  | ||||
|     await networkMachineToggle.hover() | ||||
|     await expect(page.getByText(notFoundText).nth(1)).toBeVisible() | ||||
|  | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
|  | ||||
							
								
								
									
										12
									
								
								e2e/playwright/null.spec.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,12 @@ | ||||
| // These tests are meant to simply test starting and stopping the electron | ||||
| // application, check it can make it to the project pane, and nothing more. | ||||
| // It also tests our test wrappers are working. | ||||
| // Additionally this serves as a nice minimal example. | ||||
|  | ||||
| import { test, expect } from './zoo-test' | ||||
|  | ||||
| test.describe('Open the application', () => { | ||||
|   test('see the project view', async ({ page, context }) => { | ||||
|     await expect(page.getByTestId('home-section')).toBeVisible() | ||||
|   }) | ||||
| }) | ||||
| @ -1,79 +1,63 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
| import { join } from 'path' | ||||
| import fsp from 'fs/promises' | ||||
| import { | ||||
|   getUtils, | ||||
|   setup, | ||||
|   setupElectron, | ||||
|   tearDown, | ||||
|   executorInputPath, | ||||
|   createProject, | ||||
| } from './test-utils' | ||||
| import { getUtils, executorInputPath, createProject } from './test-utils' | ||||
| import { bracket } from 'lib/exampleKcl' | ||||
| import { onboardingPaths } from 'routes/Onboarding/paths' | ||||
| import { | ||||
|   TEST_SETTINGS_KEY, | ||||
|   TEST_SETTINGS_ONBOARDING_START, | ||||
|   TEST_SETTINGS_ONBOARDING_EXPORT, | ||||
|   TEST_SETTINGS_ONBOARDING_PARAMETRIC_MODELING, | ||||
|   TEST_SETTINGS_ONBOARDING_USER_MENU, | ||||
| } from './storageStates' | ||||
| import * as TOML from '@iarna/toml' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   if (testInfo.tags.includes('@electron')) { | ||||
|     return | ||||
|   } | ||||
|   await setup(context, page) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
| // Because onboarding relies on an app setting we need to set it as incompletel | ||||
| // for all these tests. | ||||
|  | ||||
| test.describe('Onboarding tests', () => { | ||||
|   test('Onboarding code is shown in the editor', async ({ page }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     // Override beforeEach test setup | ||||
|     await page.addInitScript( | ||||
|       async ({ settingsKey }) => { | ||||
|         // Give no initial code, so that the onboarding start is shown immediately | ||||
|         localStorage.removeItem('persistCode') | ||||
|         localStorage.removeItem(settingsKey) | ||||
|   test( | ||||
|     'Onboarding code is shown in the editor', | ||||
|     { | ||||
|       appSettings: { | ||||
|         app: { | ||||
|           onboardingStatus: 'incomplete', | ||||
|         }, | ||||
|       }, | ||||
|       { settingsKey: TEST_SETTINGS_KEY } | ||||
|     ) | ||||
|       cleanProjectDir: true, | ||||
|     }, | ||||
|     async ({ context, page, homePage }) => { | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       // Test that the onboarding pane loaded | ||||
|       await expect( | ||||
|         page.getByText('Welcome to Modeling App! This') | ||||
|       ).toBeVisible() | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|  | ||||
|     // Test that the onboarding pane loaded | ||||
|     await expect(page.getByText('Welcome to Modeling App! This')).toBeVisible() | ||||
|  | ||||
|     // *and* that the code is shown in the editor | ||||
|     await expect(page.locator('.cm-content')).toContainText('// Shelf Bracket') | ||||
|   }) | ||||
|       // *and* that the code is shown in the editor | ||||
|       await expect(page.locator('.cm-content')).toContainText( | ||||
|         '// Shelf Bracket' | ||||
|       ) | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test( | ||||
|     'Desktop: fresh onboarding executes and loads', | ||||
|     { tag: '@electron' }, | ||||
|     async ({ browserName: _ }, testInfo) => { | ||||
|       const { electronApp, page } = await setupElectron({ | ||||
|         testInfo, | ||||
|         appSettings: { | ||||
|           app: { | ||||
|             onboardingStatus: 'incomplete', | ||||
|           }, | ||||
|     { | ||||
|       tag: '@electron', | ||||
|       appSettings: { | ||||
|         app: { | ||||
|           onboardingStatus: 'incomplete', | ||||
|         }, | ||||
|         cleanProjectDir: true, | ||||
|       }) | ||||
|  | ||||
|       }, | ||||
|       cleanProjectDir: true, | ||||
|     }, | ||||
|     async ({ page, homePage }, testInfo) => { | ||||
|       const u = await getUtils(page) | ||||
|  | ||||
|       const viewportSize = { width: 1200, height: 500 } | ||||
|       await page.setViewportSize(viewportSize) | ||||
|       await page.setBodyDimensions(viewportSize) | ||||
|  | ||||
|       await test.step(`Create a project and open to the onboarding`, async () => { | ||||
|         await createProject({ name: 'project-link', page }) | ||||
| @ -93,320 +77,361 @@ test.describe('Onboarding tests', () => { | ||||
|           '// Shelf Bracket' | ||||
|         ) | ||||
|       }) | ||||
|  | ||||
|       await electronApp.close() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test('Code resets after confirmation', async ({ page }) => { | ||||
|     const initialCode = `sketch001 = startSketchOn('XZ')` | ||||
|  | ||||
|     // Load the page up with some code so we see the confirmation warning | ||||
|     // when we go to replay onboarding | ||||
|     await page.addInitScript((code) => { | ||||
|       localStorage.setItem('persistCode', code) | ||||
|     }, initialCode) | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|  | ||||
|     // Replay the onboarding | ||||
|     await page.getByRole('link', { name: 'Settings' }).last().click() | ||||
|     const replayButton = page.getByRole('button', { name: 'Replay onboarding' }) | ||||
|     await expect(replayButton).toBeVisible() | ||||
|     await replayButton.click() | ||||
|  | ||||
|     // Ensure we see the warning, and that the code has not yet updated | ||||
|     await expect( | ||||
|       page.getByText('Replaying onboarding resets your code') | ||||
|     ).toBeVisible() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(initialCode) | ||||
|  | ||||
|     const nextButton = page.getByTestId('onboarding-next') | ||||
|     await expect(nextButton).toBeVisible() | ||||
|     await nextButton.click() | ||||
|  | ||||
|     // Ensure we see the introduction and that the code has been reset | ||||
|     await expect(page.getByText('Welcome to Modeling App!')).toBeVisible() | ||||
|     await expect(page.locator('.cm-content')).toContainText('// Shelf Bracket') | ||||
|  | ||||
|     // Ensure we persisted the code to local storage. | ||||
|     // Playwright's addInitScript method unfortunately will reset | ||||
|     // this code if we try reloading the page as a test, | ||||
|     // so this is our best way to test persistence afaik. | ||||
|     expect( | ||||
|       await page.evaluate(() => { | ||||
|         return localStorage.getItem('persistCode') | ||||
|       }) | ||||
|     ).toContain('// Shelf Bracket') | ||||
|   }) | ||||
|  | ||||
|   test('Click through each onboarding step', async ({ page }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     // Override beforeEach test setup | ||||
|     await page.addInitScript( | ||||
|       async ({ settingsKey, settings }) => { | ||||
|         // Give no initial code, so that the onboarding start is shown immediately | ||||
|         localStorage.setItem('persistCode', '') | ||||
|         localStorage.setItem(settingsKey, settings) | ||||
|   test( | ||||
|     'Code resets after confirmation', | ||||
|     { | ||||
|       appSettings: { | ||||
|         app: { | ||||
|           onboardingStatus: 'incomplete', | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         settingsKey: TEST_SETTINGS_KEY, | ||||
|         settings: TOML.stringify({ settings: TEST_SETTINGS_ONBOARDING_START }), | ||||
|       } | ||||
|     ) | ||||
|       cleanProjectDir: true, | ||||
|     }, | ||||
|     async ({ context, page, homePage }) => { | ||||
|       const initialCode = `sketch001 = startSketchOn('XZ')` | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 1080 }) | ||||
|       // Load the page up with some code so we see the confirmation warning | ||||
|       // when we go to replay onboarding | ||||
|       await context.addInitScript((code) => { | ||||
|         localStorage.setItem('persistCode', code) | ||||
|       }, initialCode) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|     // Test that the onboarding pane loaded | ||||
|     await expect(page.getByText('Welcome to Modeling App! This')).toBeVisible() | ||||
|       // Replay the onboarding | ||||
|       await page.getByRole('link', { name: 'Settings' }).last().click() | ||||
|       const replayButton = page.getByRole('button', { | ||||
|         name: 'Replay onboarding', | ||||
|       }) | ||||
|       await expect(replayButton).toBeVisible() | ||||
|       await replayButton.click() | ||||
|  | ||||
|     const nextButton = page.getByTestId('onboarding-next') | ||||
|       // Ensure we see the warning, and that the code has not yet updated | ||||
|       await expect(page.getByText('Would you like to create')).toBeVisible() | ||||
|       await expect(page.locator('.cm-content')).toHaveText(initialCode) | ||||
|  | ||||
|     while ((await nextButton.innerText()) !== 'Finish') { | ||||
|       await expect(nextButton).toBeVisible() | ||||
|       const nextButton = page.getByTestId('onboarding-next') | ||||
|       await nextButton.hover() | ||||
|       await nextButton.click() | ||||
|  | ||||
|       // Ensure we see the introduction and that the code has been reset | ||||
|       await expect(page.getByText('Welcome to Modeling App!')).toBeVisible() | ||||
|       await expect(page.locator('.cm-content')).toContainText( | ||||
|         '// Shelf Bracket' | ||||
|       ) | ||||
|  | ||||
|       // There used to be old code here that checked if we stored the reset | ||||
|       // code into localStorage but that isnt the case on desktop. It gets | ||||
|       // saved to the file system, which we have other tests for. | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|     // Finish the onboarding | ||||
|     await expect(nextButton).toBeVisible() | ||||
|     await nextButton.click() | ||||
|  | ||||
|     // Test that the onboarding pane is gone | ||||
|     await expect(page.getByTestId('onboarding-content')).not.toBeVisible() | ||||
|     await expect(page.url()).not.toContain('onboarding') | ||||
|   }) | ||||
|  | ||||
|   test('Onboarding redirects and code updating', async ({ page }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     // Override beforeEach test setup | ||||
|     await page.addInitScript( | ||||
|       async ({ settingsKey, settings }) => { | ||||
|         // Give some initial code, so we can test that it's cleared | ||||
|         localStorage.setItem('persistCode', 'sigmaAllow = 15000') | ||||
|         localStorage.setItem(settingsKey, settings) | ||||
|   test( | ||||
|     'Click through each onboarding step', | ||||
|     { | ||||
|       appSettings: { | ||||
|         app: { | ||||
|           onboardingStatus: 'incomplete', | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         settingsKey: TEST_SETTINGS_KEY, | ||||
|         settings: TOML.stringify({ settings: TEST_SETTINGS_ONBOARDING_EXPORT }), | ||||
|     }, | ||||
|     async ({ context, page, homePage }) => { | ||||
|       // Override beforeEach test setup | ||||
|       await context.addInitScript( | ||||
|         async ({ settingsKey, settings }) => { | ||||
|           // Give no initial code, so that the onboarding start is shown immediately | ||||
|           localStorage.setItem('persistCode', '') | ||||
|           localStorage.setItem(settingsKey, settings) | ||||
|         }, | ||||
|         { | ||||
|           settingsKey: TEST_SETTINGS_KEY, | ||||
|           settings: TOML.stringify({ | ||||
|             settings: TEST_SETTINGS_ONBOARDING_START, | ||||
|           }), | ||||
|         } | ||||
|       ) | ||||
|  | ||||
|       await page.setBodyDimensions({ width: 1200, height: 1080 }) | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|       // Test that the onboarding pane loaded | ||||
|       await expect( | ||||
|         page.getByText('Welcome to Modeling App! This') | ||||
|       ).toBeVisible() | ||||
|  | ||||
|       const nextButton = page.getByTestId('onboarding-next') | ||||
|  | ||||
|       while ((await nextButton.innerText()) !== 'Finish') { | ||||
|         await nextButton.hover() | ||||
|         await nextButton.click() | ||||
|       } | ||||
|     ) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       // Finish the onboarding | ||||
|       await nextButton.hover() | ||||
|       await nextButton.click() | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|  | ||||
|     // Test that the redirect happened | ||||
|     await expect(page.url().split(':3000').slice(-1)[0]).toBe( | ||||
|       `/file/%2Fbrowser%2Fmain.kcl/onboarding/export` | ||||
|     ) | ||||
|  | ||||
|     // Test that you come back to this page when you refresh | ||||
|     await page.reload() | ||||
|     await expect(page.url().split(':3000').slice(-1)[0]).toBe( | ||||
|       `/file/%2Fbrowser%2Fmain.kcl/onboarding/export` | ||||
|     ) | ||||
|  | ||||
|     // Test that the onboarding pane loaded | ||||
|     const title = page.locator('[data-testid="onboarding-content"]') | ||||
|     await expect(title).toBeAttached() | ||||
|  | ||||
|     // Test that the code changes when you advance to the next step | ||||
|     await page.locator('[data-testid="onboarding-next"]').click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText('') | ||||
|  | ||||
|     // Test that the code is not empty when you click on the next step | ||||
|     await page.locator('[data-testid="onboarding-next"]').click() | ||||
|     await expect(page.locator('.cm-content')).toHaveText(/.+/) | ||||
|   }) | ||||
|  | ||||
|   test('Onboarding code gets reset to demo on Interactive Numbers step', async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     test.skip( | ||||
|       process.platform === 'darwin', | ||||
|       "Skip on macOS, because Playwright isn't behaving the same as the actual browser" | ||||
|     ) | ||||
|     const u = await getUtils(page) | ||||
|     const badCode = `// This is bad code we shouldn't see` | ||||
|     // Override beforeEach test setup | ||||
|     await page.addInitScript( | ||||
|       async ({ settingsKey, settings, badCode }) => { | ||||
|         localStorage.setItem('persistCode', badCode) | ||||
|         localStorage.setItem(settingsKey, settings) | ||||
|       }, | ||||
|       { | ||||
|         settingsKey: TEST_SETTINGS_KEY, | ||||
|         settings: TOML.stringify({ | ||||
|           settings: TEST_SETTINGS_ONBOARDING_PARAMETRIC_MODELING, | ||||
|         }), | ||||
|         badCode, | ||||
|       } | ||||
|     ) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 1080 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|  | ||||
|     await page.waitForURL('**' + onboardingPaths.PARAMETRIC_MODELING, { | ||||
|       waitUntil: 'domcontentloaded', | ||||
|     }) | ||||
|  | ||||
|     const bracketNoNewLines = bracket.replace(/\n/g, '') | ||||
|  | ||||
|     // Check the code got reset on load | ||||
|     await expect(page.locator('#code-pane')).toBeVisible() | ||||
|     await expect(u.codeLocator).toHaveText(bracketNoNewLines, { | ||||
|       timeout: 10_000, | ||||
|     }) | ||||
|  | ||||
|     // Mess with the code again | ||||
|     await u.codeLocator.selectText() | ||||
|     await u.codeLocator.fill(badCode) | ||||
|     await expect(u.codeLocator).toHaveText(badCode) | ||||
|  | ||||
|     // Click to the next step | ||||
|     await page.locator('[data-testid="onboarding-next"]').click() | ||||
|     await page.waitForURL('**' + onboardingPaths.INTERACTIVE_NUMBERS, { | ||||
|       waitUntil: 'domcontentloaded', | ||||
|     }) | ||||
|  | ||||
|     // Check that the code has been reset | ||||
|     await expect(u.codeLocator).toHaveText(bracketNoNewLines) | ||||
|   }) | ||||
|  | ||||
|   test('Avatar text updates depending on image load success', async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     // Override beforeEach test setup | ||||
|     await page.addInitScript( | ||||
|       async ({ settingsKey, settings }) => { | ||||
|         localStorage.setItem(settingsKey, settings) | ||||
|       }, | ||||
|       { | ||||
|         settingsKey: TEST_SETTINGS_KEY, | ||||
|         settings: TOML.stringify({ | ||||
|           settings: TEST_SETTINGS_ONBOARDING_USER_MENU, | ||||
|         }), | ||||
|       } | ||||
|     ) | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|  | ||||
|     await page.waitForURL('**/file/**', { waitUntil: 'domcontentloaded' }) | ||||
|  | ||||
|     // Test that the text in this step is correct | ||||
|     const avatarLocator = await page | ||||
|       .getByTestId('user-sidebar-toggle') | ||||
|       .locator('img') | ||||
|     const onboardingOverlayLocator = await page | ||||
|       .getByTestId('onboarding-content') | ||||
|       .locator('div') | ||||
|       .nth(1) | ||||
|  | ||||
|     // Expect the avatar to be visible and for the text to reference it | ||||
|     await expect(avatarLocator).toBeVisible() | ||||
|     await expect(onboardingOverlayLocator).toBeVisible() | ||||
|     await expect(onboardingOverlayLocator).toContainText('your avatar') | ||||
|  | ||||
|     // This is to force the avatar to 404. | ||||
|     // For our test image (only triggers locally. on CI, it's Kurt's / | ||||
|     // gravatar image ) | ||||
|     await page.route('/cat.jpg', async (route) => { | ||||
|       await route.fulfill({ | ||||
|         status: 404, | ||||
|         contentType: 'text/plain', | ||||
|         body: 'Not Found!', | ||||
|       }) | ||||
|     }) | ||||
|  | ||||
|     // 404 the CI avatar image | ||||
|     await page.route('https://lh3.googleusercontent.com/**', async (route) => { | ||||
|       await route.fulfill({ | ||||
|         status: 404, | ||||
|         contentType: 'text/plain', | ||||
|         body: 'Not Found!', | ||||
|       }) | ||||
|     }) | ||||
|  | ||||
|     await page.reload({ waitUntil: 'domcontentloaded' }) | ||||
|  | ||||
|     // Now expect the text to be different | ||||
|     await expect(avatarLocator).not.toBeVisible() | ||||
|     await expect(onboardingOverlayLocator).toBeVisible() | ||||
|     await expect(onboardingOverlayLocator).toContainText('the menu button') | ||||
|   }) | ||||
|  | ||||
|   test("Avatar text doesn't mention avatar when no avatar", async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     // Override beforeEach test setup | ||||
|     await page.addInitScript( | ||||
|       async ({ settingsKey, settings }) => { | ||||
|         localStorage.setItem(settingsKey, settings) | ||||
|         localStorage.setItem('FORCE_NO_IMAGE', 'FORCE_NO_IMAGE') | ||||
|       }, | ||||
|       { | ||||
|         settingsKey: TEST_SETTINGS_KEY, | ||||
|         settings: TOML.stringify({ | ||||
|           settings: TEST_SETTINGS_ONBOARDING_USER_MENU, | ||||
|         }), | ||||
|       } | ||||
|     ) | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|  | ||||
|     await page.waitForURL('**/file/**', { waitUntil: 'domcontentloaded' }) | ||||
|  | ||||
|     // Test that the text in this step is correct | ||||
|     const sidebar = page.getByTestId('user-sidebar-toggle') | ||||
|     const avatar = sidebar.locator('img') | ||||
|     const onboardingOverlayLocator = page | ||||
|       .getByTestId('onboarding-content') | ||||
|       .locator('div') | ||||
|       .nth(1) | ||||
|  | ||||
|     // Expect the avatar to be visible and for the text to reference it | ||||
|     await expect(avatar).not.toBeVisible() | ||||
|     await expect(onboardingOverlayLocator).toBeVisible() | ||||
|     await expect(onboardingOverlayLocator).toContainText('the menu button') | ||||
|  | ||||
|     // Test we mention what else is in this menu for https://github.com/KittyCAD/modeling-app/issues/2939 | ||||
|     // which doesn't deserver its own full test spun up | ||||
|     const userMenuFeatures = [ | ||||
|       'manage your account', | ||||
|       'report a bug', | ||||
|       'request a feature', | ||||
|       'sign out', | ||||
|     ] | ||||
|     for (const feature of userMenuFeatures) { | ||||
|       await expect(onboardingOverlayLocator).toContainText(feature) | ||||
|       // Test that the onboarding pane is gone | ||||
|       await expect(page.getByTestId('onboarding-content')).not.toBeVisible() | ||||
|       await expect.poll(() => page.url()).not.toContain('/onboarding') | ||||
|     } | ||||
|   }) | ||||
|   ) | ||||
|  | ||||
|   test( | ||||
|     'Onboarding redirects and code updating', | ||||
|     { | ||||
|       appSettings: { | ||||
|         app: { | ||||
|           onboardingStatus: '/export', | ||||
|         }, | ||||
|       }, | ||||
|       cleanProjectDir: true, | ||||
|     }, | ||||
|     async ({ context, page, homePage }) => { | ||||
|       const originalCode = 'sigmaAllow = 15000' | ||||
|  | ||||
|       // Override beforeEach test setup | ||||
|       await context.addInitScript( | ||||
|         async ({ settingsKey, settings }) => { | ||||
|           // Give some initial code, so we can test that it's cleared | ||||
|           localStorage.setItem('persistCode', originalCode) | ||||
|           localStorage.setItem(settingsKey, settings) | ||||
|         }, | ||||
|         { | ||||
|           settingsKey: TEST_SETTINGS_KEY, | ||||
|           settings: TOML.stringify({ | ||||
|             settings: TEST_SETTINGS_ONBOARDING_EXPORT, | ||||
|           }), | ||||
|         } | ||||
|       ) | ||||
|  | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|       // Test that the redirect happened | ||||
|       await expect.poll(() => page.url()).toContain('/onboarding/export') | ||||
|  | ||||
|       // Test that you come back to this page when you refresh | ||||
|       await page.reload() | ||||
|       await expect.poll(() => page.url()).toContain('/onboarding/export') | ||||
|  | ||||
|       // Test that the code changes when you advance to the next step | ||||
|       await page.getByTestId('onboarding-next').hover() | ||||
|       await page.getByTestId('onboarding-next').click() | ||||
|  | ||||
|       // Test that the onboarding pane loaded | ||||
|       const title = page.locator('[data-testid="onboarding-content"]') | ||||
|       await expect(title).toBeAttached() | ||||
|  | ||||
|       await expect(page.locator('.cm-content')).not.toHaveText(originalCode) | ||||
|  | ||||
|       // Test that the code is not empty when you click on the next step | ||||
|       await page.locator('[data-testid="onboarding-next"]').hover() | ||||
|       await page.locator('[data-testid="onboarding-next"]').click() | ||||
|       await expect(page.locator('.cm-content')).toHaveText(/.+/) | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test( | ||||
|     'Onboarding code gets reset to demo on Interactive Numbers step', | ||||
|     { | ||||
|       appSettings: { | ||||
|         app: { | ||||
|           onboardingStatus: '/parametric-modeling', | ||||
|         }, | ||||
|       }, | ||||
|       cleanProjectDir: true, | ||||
|     }, | ||||
|  | ||||
|     async ({ context, page, homePage }) => { | ||||
|       const u = await getUtils(page) | ||||
|       const badCode = `// This is bad code we shouldn't see` | ||||
|  | ||||
|       await page.setBodyDimensions({ width: 1200, height: 1080 }) | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|       await expect | ||||
|         .poll(() => page.url()) | ||||
|         .toContain(onboardingPaths.PARAMETRIC_MODELING) | ||||
|  | ||||
|       const bracketNoNewLines = bracket.replace(/\n/g, '') | ||||
|  | ||||
|       // Check the code got reset on load | ||||
|       await expect(page.locator('#code-pane')).toBeVisible() | ||||
|       await expect(u.codeLocator).toHaveText(bracketNoNewLines, { | ||||
|         timeout: 10_000, | ||||
|       }) | ||||
|  | ||||
|       // Mess with the code again | ||||
|       await u.codeLocator.selectText() | ||||
|       await u.codeLocator.fill(badCode) | ||||
|       await expect(u.codeLocator).toHaveText(badCode) | ||||
|  | ||||
|       // Click to the next step | ||||
|       await page.locator('[data-testid="onboarding-next"]').hover() | ||||
|       await page.locator('[data-testid="onboarding-next"]').click() | ||||
|       await page.waitForURL('**' + onboardingPaths.INTERACTIVE_NUMBERS, { | ||||
|         waitUntil: 'domcontentloaded', | ||||
|       }) | ||||
|  | ||||
|       // Check that the code has been reset | ||||
|       await expect(u.codeLocator).toHaveText(bracketNoNewLines) | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   // (lee) The two avatar tests are weird because even on main, we don't have | ||||
|   // anything to do with the avatar inside the onboarding test. Due to the | ||||
|   // low impact of an avatar not showing I'm changing this to fixme. | ||||
|   test.fixme( | ||||
|     'Avatar text updates depending on image load success', | ||||
|     { | ||||
|       appSettings: { | ||||
|         app: { | ||||
|           onboardingStatus: 'incomplete', | ||||
|         }, | ||||
|       }, | ||||
|       cleanProjectDir: true, | ||||
|     }, | ||||
|     async ({ context, page, homePage }) => { | ||||
|       // Override beforeEach test setup | ||||
|       await context.addInitScript( | ||||
|         async ({ settingsKey, settings }) => { | ||||
|           localStorage.setItem(settingsKey, settings) | ||||
|         }, | ||||
|         { | ||||
|           settingsKey: TEST_SETTINGS_KEY, | ||||
|           settings: TOML.stringify({ | ||||
|             settings: TEST_SETTINGS_ONBOARDING_USER_MENU, | ||||
|           }), | ||||
|         } | ||||
|       ) | ||||
|  | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|       // Test that the text in this step is correct | ||||
|       const avatarLocator = await page | ||||
|         .getByTestId('user-sidebar-toggle') | ||||
|         .locator('img') | ||||
|       const onboardingOverlayLocator = await page | ||||
|         .getByTestId('onboarding-content') | ||||
|         .locator('div') | ||||
|         .nth(1) | ||||
|  | ||||
|       // Expect the avatar to be visible and for the text to reference it | ||||
|       await expect(avatarLocator).toBeVisible() | ||||
|       await expect(onboardingOverlayLocator).toBeVisible() | ||||
|       await expect(onboardingOverlayLocator).toContainText('your avatar') | ||||
|  | ||||
|       // This is to force the avatar to 404. | ||||
|       // For our test image (only triggers locally. on CI, it's Kurt's / | ||||
|       // gravatar image ) | ||||
|       await page.route('/cat.jpg', async (route) => { | ||||
|         await route.fulfill({ | ||||
|           status: 404, | ||||
|           contentType: 'text/plain', | ||||
|           body: 'Not Found!', | ||||
|         }) | ||||
|       }) | ||||
|  | ||||
|       // 404 the CI avatar image | ||||
|       await page.route( | ||||
|         'https://lh3.googleusercontent.com/**', | ||||
|         async (route) => { | ||||
|           await route.fulfill({ | ||||
|             status: 404, | ||||
|             contentType: 'text/plain', | ||||
|             body: 'Not Found!', | ||||
|           }) | ||||
|         } | ||||
|       ) | ||||
|  | ||||
|       await page.reload({ waitUntil: 'domcontentloaded' }) | ||||
|  | ||||
|       // Now expect the text to be different | ||||
|       await expect(avatarLocator).not.toBeVisible() | ||||
|       await expect(onboardingOverlayLocator).toBeVisible() | ||||
|       await expect(onboardingOverlayLocator).toContainText('the menu button') | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test.fixme( | ||||
|     "Avatar text doesn't mention avatar when no avatar", | ||||
|     { | ||||
|       appSettings: { | ||||
|         app: { | ||||
|           onboardingStatus: 'incomplete', | ||||
|         }, | ||||
|       }, | ||||
|       cleanProjectDir: true, | ||||
|     }, | ||||
|     async ({ context, page, homePage }) => { | ||||
|       // Override beforeEach test setup | ||||
|       await context.addInitScript( | ||||
|         async ({ settingsKey, settings }) => { | ||||
|           localStorage.setItem(settingsKey, settings) | ||||
|           localStorage.setItem('FORCE_NO_IMAGE', 'FORCE_NO_IMAGE') | ||||
|         }, | ||||
|         { | ||||
|           settingsKey: TEST_SETTINGS_KEY, | ||||
|           settings: TOML.stringify({ | ||||
|             settings: TEST_SETTINGS_ONBOARDING_USER_MENU, | ||||
|           }), | ||||
|         } | ||||
|       ) | ||||
|  | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|       // Test that the text in this step is correct | ||||
|       const sidebar = page.getByTestId('user-sidebar-toggle') | ||||
|       const avatar = sidebar.locator('img') | ||||
|       const onboardingOverlayLocator = page | ||||
|         .getByTestId('onboarding-content') | ||||
|         .locator('div') | ||||
|         .nth(1) | ||||
|  | ||||
|       // Expect the avatar to be visible and for the text to reference it | ||||
|       await expect(avatar).not.toBeVisible() | ||||
|       await expect(onboardingOverlayLocator).toBeVisible() | ||||
|       await expect(onboardingOverlayLocator).toContainText('the menu button') | ||||
|  | ||||
|       // Test we mention what else is in this menu for https://github.com/KittyCAD/modeling-app/issues/2939 | ||||
|       // which doesn't deserver its own full test spun up | ||||
|       const userMenuFeatures = [ | ||||
|         'manage your account', | ||||
|         'report a bug', | ||||
|         'request a feature', | ||||
|         'sign out', | ||||
|       ] | ||||
|       for (const feature of userMenuFeatures) { | ||||
|         await expect(onboardingOverlayLocator).toContainText(feature) | ||||
|       } | ||||
|     } | ||||
|   ) | ||||
| }) | ||||
|  | ||||
| test( | ||||
|   'Restarting onboarding on desktop takes one attempt', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browser: _ }, testInfo) => { | ||||
|     const { electronApp, page } = await setupElectron({ | ||||
|       testInfo, | ||||
|       folderSetupFn: async (dir) => { | ||||
|         const routerTemplateDir = join(dir, 'router-template-slate') | ||||
|         await fsp.mkdir(routerTemplateDir, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           executorInputPath('router-template-slate.kcl'), | ||||
|           join(routerTemplateDir, 'main.kcl') | ||||
|         ) | ||||
|   { | ||||
|     appSettings: { | ||||
|       app: { | ||||
|         onboardingStatus: 'dismissed', | ||||
|       }, | ||||
|     }, | ||||
|     cleanProjectDir: true, | ||||
|   }, | ||||
|   async ({ context, page, homePage }, testInfo) => { | ||||
|     await context.folderSetupFn(async (dir) => { | ||||
|       const routerTemplateDir = join(dir, 'router-template-slate') | ||||
|       await fsp.mkdir(routerTemplateDir, { recursive: true }) | ||||
|       await fsp.copyFile( | ||||
|         executorInputPath('router-template-slate.kcl'), | ||||
|         join(routerTemplateDir, 'main.kcl') | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     // Our constants | ||||
| @ -418,9 +443,8 @@ test( | ||||
|     const restartOnboardingButton = page.getByRole('button', { | ||||
|       name: 'Reset onboarding', | ||||
|     }) | ||||
|     const restartConfirmationButton = page.getByRole('button', { | ||||
|       name: 'Make a new project', | ||||
|     }) | ||||
|     const nextButton = page.getByTestId('onboarding-next') | ||||
|  | ||||
|     const tutorialProjectIndicator = page | ||||
|       .getByTestId('project-sidebar-toggle') | ||||
|       .filter({ hasText: 'Tutorial Project 00' }) | ||||
| @ -439,7 +463,7 @@ test( | ||||
|     }) | ||||
|  | ||||
|     await test.step('Navigate into project', async () => { | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|       page.on('console', console.log) | ||||
|  | ||||
| @ -455,8 +479,8 @@ test( | ||||
|       await helpMenuButton.click() | ||||
|       await restartOnboardingButton.click() | ||||
|  | ||||
|       await expect(restartConfirmationButton).toBeVisible() | ||||
|       await restartConfirmationButton.click() | ||||
|       await nextButton.hover() | ||||
|       await nextButton.click() | ||||
|     }) | ||||
|  | ||||
|     await test.step('Confirm that the onboarding has restarted', async () => { | ||||
| @ -480,11 +504,9 @@ test( | ||||
|  | ||||
|       await restartOnboardingSettingsButton.click() | ||||
|       // Since the code is empty, we should not see the confirmation dialog | ||||
|       await expect(restartConfirmationButton).not.toBeVisible() | ||||
|       await expect(nextButton).not.toBeVisible() | ||||
|       await expect(tutorialProjectIndicator).toBeVisible() | ||||
|       await expect(tutorialModalText).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
|  | ||||
| @ -1,25 +1,49 @@ | ||||
| import { test, expect, AuthenticatedApp } from './fixtures/fixtureSetup' | ||||
| import { test, expect, Page } from './zoo-test' | ||||
| import { EditorFixture } from './fixtures/editorFixture' | ||||
| import { SceneFixture } from './fixtures/sceneFixture' | ||||
| import { ToolbarFixture } from './fixtures/toolbarFixture' | ||||
| import fs from 'node:fs/promises' | ||||
| import path from 'node:path' | ||||
| import { getUtils } from './test-utils' | ||||
|  | ||||
| // test file is for testing point an click code gen functionality that's not sketch mode related | ||||
|  | ||||
| test( | ||||
|   'verify extruding circle works', | ||||
|   { tag: ['@skipWin'] }, | ||||
|   async ({ app, cmdBar, editor, toolbar, scene }) => { | ||||
|     test.skip( | ||||
|       process.platform === 'win32', | ||||
|       'Fails on windows in CI, can not be replicated locally on windows.' | ||||
|     ) | ||||
|     const file = await app.getInputFile('test-circle-extrude.kcl') | ||||
|     await app.initialise(file) | ||||
|     const [clickCircle, moveToCircle] = scene.makeMouseHelpers(582, 217) | ||||
| test('verify extruding circle works', async ({ | ||||
|   context, | ||||
|   homePage, | ||||
|   cmdBar, | ||||
|   editor, | ||||
|   toolbar, | ||||
|   scene, | ||||
| }) => { | ||||
|   const file = await fs.readFile( | ||||
|     path.resolve( | ||||
|       __dirname, | ||||
|       '../../', | ||||
|       './src/wasm-lib/tests/executor/inputs/test-circle-extrude.kcl' | ||||
|     ), | ||||
|     'utf-8' | ||||
|   ) | ||||
|   await context.addInitScript((file) => { | ||||
|     localStorage.setItem('persistCode', file) | ||||
|   }, file) | ||||
|   await homePage.goToModelingScene() | ||||
|  | ||||
|     await test.step('because there is sweepable geometry, verify extrude is enable when nothing is selected', async () => { | ||||
|       await scene.clickNoWhere() | ||||
|       await expect(toolbar.extrudeButton).toBeEnabled() | ||||
|   const [clickCircle, moveToCircle] = scene.makeMouseHelpers(582, 217) | ||||
|  | ||||
|   await test.step('because there is sweepable geometry, verify extrude is enable when nothing is selected', async () => { | ||||
|     await scene.clickNoWhere() | ||||
|     await expect(toolbar.extrudeButton).toBeEnabled() | ||||
|   }) | ||||
|  | ||||
|   await test.step('check code model connection works and that button is still enable once circle is selected ', async () => { | ||||
|     await moveToCircle() | ||||
|     const circleSnippet = | ||||
|       'circle({ center = [318.33, 168.1], radius = 182.8 }, %)' | ||||
|     await editor.expectState({ | ||||
|       activeLines: ["constsketch002=startSketchOn('XZ')"], | ||||
|       highlightedCode: circleSnippet, | ||||
|       diagnostics: [], | ||||
|     }) | ||||
|  | ||||
|     await test.step('check code model connection works and that button is still enable once circle is selected ', async () => { | ||||
| @ -27,7 +51,7 @@ test( | ||||
|       const circleSnippet = | ||||
|         'circle({ center = [318.33, 168.1], radius = 182.8 }, %)' | ||||
|       await editor.expectState({ | ||||
|         activeLines: [], | ||||
|         activeLines: ["constsketch002=startSketchOn('XZ')"], | ||||
|         highlightedCode: circleSnippet, | ||||
|         diagnostics: [], | ||||
|       }) | ||||
| @ -40,39 +64,40 @@ test( | ||||
|       }) | ||||
|       await expect(toolbar.extrudeButton).toBeEnabled() | ||||
|     }) | ||||
|     await expect(toolbar.extrudeButton).toBeEnabled() | ||||
|   }) | ||||
|  | ||||
|     await test.step('do extrude flow and check extrude code is added to editor', async () => { | ||||
|       await toolbar.extrudeButton.click() | ||||
|   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', | ||||
|         currentArgValue: '5', | ||||
|         headerArguments: { Selection: '1 face', Distance: '' }, | ||||
|         highlightedHeaderArg: 'distance', | ||||
|         commandName: 'Extrude', | ||||
|       }) | ||||
|       await cmdBar.progressCmdBar() | ||||
|  | ||||
|       const expectString = 'extrude001 = extrude(5, sketch001)' | ||||
|       await editor.expectEditor.not.toContain(expectString) | ||||
|  | ||||
|       await cmdBar.expectState({ | ||||
|         stage: 'review', | ||||
|         headerArguments: { Selection: '1 face', Distance: '5' }, | ||||
|         commandName: 'Extrude', | ||||
|       }) | ||||
|       await cmdBar.progressCmdBar() | ||||
|  | ||||
|       await editor.expectEditor.toContain(expectString) | ||||
|     await cmdBar.expectState({ | ||||
|       stage: 'arguments', | ||||
|       currentArgKey: 'distance', | ||||
|       currentArgValue: '5', | ||||
|       headerArguments: { Selection: '1 face', Distance: '' }, | ||||
|       highlightedHeaderArg: 'distance', | ||||
|       commandName: 'Extrude', | ||||
|     }) | ||||
|   } | ||||
| ) | ||||
|     await cmdBar.progressCmdBar() | ||||
|  | ||||
|     const expectString = 'extrude001 = extrude(5, sketch001)' | ||||
|     await editor.expectEditor.not.toContain(expectString) | ||||
|  | ||||
|     await cmdBar.expectState({ | ||||
|       stage: 'review', | ||||
|       headerArguments: { Selection: '1 face', Distance: '5' }, | ||||
|       commandName: 'Extrude', | ||||
|     }) | ||||
|     await cmdBar.progressCmdBar() | ||||
|  | ||||
|     await editor.expectEditor.toContain(expectString) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| test.describe('verify sketch on chamfer works', () => { | ||||
|   const _sketchOnAChamfer = | ||||
|     ( | ||||
|       app: AuthenticatedApp, | ||||
|       page: Page, | ||||
|       editor: EditorFixture, | ||||
|       toolbar: ToolbarFixture, | ||||
|       scene: SceneFixture | ||||
| @ -124,7 +149,7 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|         await toolbar.startSketchPlaneSelection() | ||||
|         await clickChamfer() | ||||
|         // timeout wait for engine animation is unavoidable | ||||
|         await app.page.waitForTimeout(600) | ||||
|         await page.waitForTimeout(1000) | ||||
|         await editor.expectEditor.toContain(afterChamferSelectSnippet) | ||||
|       }) | ||||
|       await test.step('make sure a basic sketch can be added', async () => { | ||||
| @ -135,7 +160,9 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|           pixelDiff: 50, | ||||
|         }) | ||||
|         await rectangle2ndClick() | ||||
|         await editor.expectEditor.toContain(afterRectangle2ndClickSnippet) | ||||
|         await editor.expectEditor.toContain(afterRectangle2ndClickSnippet, { | ||||
|           shouldNormalise: true, | ||||
|         }) | ||||
|       }) | ||||
|  | ||||
|       await test.step('Clean up so that `_sketchOnAChamfer` util can be called again', async () => { | ||||
| @ -150,24 +177,35 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|         }) | ||||
|       }) | ||||
|     } | ||||
|   test( | ||||
|     'works on all edge selections and can break up multi edges in a chamfer array', | ||||
|     { tag: ['@skipWin'] }, | ||||
|     async ({ app, editor, toolbar, scene }) => { | ||||
|       test.skip( | ||||
|         process.platform === 'win32', | ||||
|         'Fails on windows in CI, can not be replicated locally on windows.' | ||||
|       ) | ||||
|       const file = await app.getInputFile('e2e-can-sketch-on-chamfer.kcl') | ||||
|       await app.initialise(file) | ||||
|   test('works on all edge selections and can break up multi edges in a chamfer array', async ({ | ||||
|     context, | ||||
|     page, | ||||
|     homePage, | ||||
|     editor, | ||||
|     toolbar, | ||||
|     scene, | ||||
|   }) => { | ||||
|     const file = await fs.readFile( | ||||
|       path.resolve( | ||||
|         __dirname, | ||||
|         '../../', | ||||
|         './src/wasm-lib/tests/executor/inputs/e2e-can-sketch-on-chamfer.kcl' | ||||
|       ), | ||||
|       'utf-8' | ||||
|     ) | ||||
|     await context.addInitScript((file) => { | ||||
|       localStorage.setItem('persistCode', file) | ||||
|     }, file) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|       const sketchOnAChamfer = _sketchOnAChamfer(app, editor, toolbar, scene) | ||||
|     const sketchOnAChamfer = _sketchOnAChamfer(page, editor, toolbar, scene) | ||||
|  | ||||
|       await sketchOnAChamfer({ | ||||
|         clickCoords: { x: 570, y: 220 }, | ||||
|         cameraPos: { x: 16020, y: -2000, z: 10500 }, | ||||
|         cameraTarget: { x: -150, y: -4500, z: -80 }, | ||||
|         beforeChamferSnippet: `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01) | ||||
|     await sketchOnAChamfer({ | ||||
|       clickCoords: { x: 570, y: 220 }, | ||||
|       cameraPos: { x: 16020, y: -2000, z: 10500 }, | ||||
|       cameraTarget: { x: -150, y: -4500, z: -80 }, | ||||
|       beforeChamferSnippet: `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01) | ||||
|       chamfer({length = 30,tags = [ | ||||
|       seg01, | ||||
|       getNextAdjacentEdge(yo), | ||||
| @ -175,10 +213,9 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|       getOppositeEdge(seg01) | ||||
|     ]}, %)`, | ||||
|  | ||||
|         afterChamferSelectSnippet: | ||||
|           'sketch002 = startSketchOn(extrude001, seg03)', | ||||
|         afterRectangle1stClickSnippet: 'startProfileAt([205.96, 254.59], %)', | ||||
|         afterRectangle2ndClickSnippet: `angledLine([0, 11.39], %, $rectangleSegmentA002) | ||||
|       afterChamferSelectSnippet: 'sketch002 = startSketchOn(extrude001, seg03)', | ||||
|       afterRectangle1stClickSnippet: 'startProfileAt([205.96, 254.59], %)', | ||||
|       afterRectangle2ndClickSnippet: `angledLine([0, 11.39], %, $rectangleSegmentA002) | ||||
|     |> angledLine([ | ||||
|          segAng(rectangleSegmentA002) - 90, | ||||
|          105.26 | ||||
| @ -189,13 +226,13 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|        ], %, $rectangleSegmentC001) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%)`, | ||||
|       }) | ||||
|     }) | ||||
|  | ||||
|       await sketchOnAChamfer({ | ||||
|         clickCoords: { x: 690, y: 250 }, | ||||
|         cameraPos: { x: 16020, y: -2000, z: 10500 }, | ||||
|         cameraTarget: { x: -150, y: -4500, z: -80 }, | ||||
|         beforeChamferSnippet: `angledLine([ | ||||
|     await sketchOnAChamfer({ | ||||
|       clickCoords: { x: 690, y: 250 }, | ||||
|       cameraPos: { x: 16020, y: -2000, z: 10500 }, | ||||
|       cameraTarget: { x: -150, y: -4500, z: -80 }, | ||||
|       beforeChamferSnippet: `angledLine([ | ||||
|          segAng(rectangleSegmentA001) - 90, | ||||
|          217.26 | ||||
|        ], %, $seg01)chamfer({ | ||||
| @ -207,10 +244,9 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|          ] | ||||
|        }, %)`, | ||||
|  | ||||
|         afterChamferSelectSnippet: | ||||
|           'sketch003 = startSketchOn(extrude001, seg04)', | ||||
|         afterRectangle1stClickSnippet: 'startProfileAt([-209.64, 255.28], %)', | ||||
|         afterRectangle2ndClickSnippet: `angledLine([0, 11.56], %, $rectangleSegmentA003) | ||||
|       afterChamferSelectSnippet: 'sketch003 = startSketchOn(extrude001, seg04)', | ||||
|       afterRectangle1stClickSnippet: 'startProfileAt([-209.64, 255.28], %)', | ||||
|       afterRectangle2ndClickSnippet: `angledLine([0, 11.56], %, $rectangleSegmentA003) | ||||
|     |> angledLine([ | ||||
|          segAng(rectangleSegmentA003) - 90, | ||||
|          106.84 | ||||
| @ -221,22 +257,21 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|        ], %, $rectangleSegmentC002) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%)`, | ||||
|       }) | ||||
|       await sketchOnAChamfer({ | ||||
|         clickCoords: { x: 677, y: 87 }, | ||||
|         cameraPos: { x: -6200, y: 1500, z: 6200 }, | ||||
|         cameraTarget: { x: 8300, y: 1100, z: 4800 }, | ||||
|         beforeChamferSnippet: `angledLine([0, 268.43], %, $rectangleSegmentA001)chamfer({ | ||||
|     }) | ||||
|     await sketchOnAChamfer({ | ||||
|       clickCoords: { x: 677, y: 87 }, | ||||
|       cameraPos: { x: -6200, y: 1500, z: 6200 }, | ||||
|       cameraTarget: { x: 8300, y: 1100, z: 4800 }, | ||||
|       beforeChamferSnippet: `angledLine([0, 268.43], %, $rectangleSegmentA001)chamfer({ | ||||
|          length = 30, | ||||
|          tags = [ | ||||
|            getNextAdjacentEdge(yo), | ||||
|            getNextAdjacentEdge(seg02) | ||||
|          ] | ||||
|        }, %)`, | ||||
|         afterChamferSelectSnippet: | ||||
|           'sketch003 = startSketchOn(extrude001, seg04)', | ||||
|         afterRectangle1stClickSnippet: 'startProfileAt([-209.64, 255.28], %)', | ||||
|         afterRectangle2ndClickSnippet: `angledLine([0, 11.56], %, $rectangleSegmentA003) | ||||
|       afterChamferSelectSnippet: 'sketch003 = startSketchOn(extrude001, seg04)', | ||||
|       afterRectangle1stClickSnippet: 'startProfileAt([75.8, 317.2], %)', | ||||
|       afterRectangle2ndClickSnippet: `angledLine([0, 11.56], %, $rectangleSegmentA003) | ||||
|     |> angledLine([ | ||||
|          segAng(rectangleSegmentA003) - 90, | ||||
|          106.84 | ||||
| @ -247,20 +282,19 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|        ], %, $rectangleSegmentC002) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%)`, | ||||
|       }) | ||||
|       /// last one | ||||
|       await sketchOnAChamfer({ | ||||
|         clickCoords: { x: 620, y: 300 }, | ||||
|         cameraPos: { x: -1100, y: -7700, z: 1600 }, | ||||
|         cameraTarget: { x: 1450, y: 670, z: 4000 }, | ||||
|         beforeChamferSnippet: `chamfer({ | ||||
|     }) | ||||
|     /// last one | ||||
|     await sketchOnAChamfer({ | ||||
|       clickCoords: { x: 620, y: 300 }, | ||||
|       cameraPos: { x: -1100, y: -7700, z: 1600 }, | ||||
|       cameraTarget: { x: 1450, y: 670, z: 4000 }, | ||||
|       beforeChamferSnippet: `chamfer({ | ||||
|          length = 30, | ||||
|          tags = [getNextAdjacentEdge(yo)] | ||||
|        }, %)`, | ||||
|         afterChamferSelectSnippet: | ||||
|           'sketch005 = startSketchOn(extrude001, seg06)', | ||||
|         afterRectangle1stClickSnippet: 'startProfileAt([-23.43, 19.69], %)', | ||||
|         afterRectangle2ndClickSnippet: `angledLine([0, 9.1], %, $rectangleSegmentA005) | ||||
|       afterChamferSelectSnippet: 'sketch005 = startSketchOn(extrude001, seg06)', | ||||
|       afterRectangle1stClickSnippet: 'startProfileAt([-23.43, 19.69], %)', | ||||
|       afterRectangle2ndClickSnippet: `angledLine([0, 9.1], %, $rectangleSegmentA005) | ||||
|  | ||||
|     |> angledLine([ | ||||
|          segAng(rectangleSegmentA005) - 90, | ||||
| @ -272,11 +306,11 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|        ], %, $rectangleSegmentC004) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%)`, | ||||
|       }) | ||||
|     }) | ||||
|  | ||||
|       await test.step('verify at the end of the test that final code is what is expected', async () => { | ||||
|         await editor.expectEditor.toContain( | ||||
|           `sketch001 = startSketchOn('XZ') | ||||
|     await test.step('verify at the end of the test that final code is what is expected', async () => { | ||||
|       await editor.expectEditor.toContain( | ||||
|         `sketch001 = startSketchOn('XZ') | ||||
|  | ||||
|       |> startProfileAt([75.8, 317.2], %) // [$startCapTag, $EndCapTag] | ||||
|       |> angledLine([0, 268.43], %, $rectangleSegmentA001) | ||||
| @ -305,7 +339,7 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|            tags = [getNextAdjacentEdge(yo)] | ||||
|          }, %, $seg06) | ||||
|     sketch005 = startSketchOn(extrude001, seg06) | ||||
|       |> startProfileAt([-23.43, 19.69], %) | ||||
|       |> startProfileAt([-23.43,19.69], %) | ||||
|       |> angledLine([0, 9.1], %, $rectangleSegmentA005) | ||||
|       |> angledLine([ | ||||
|            segAng(rectangleSegmentA005) - 90, | ||||
| @ -318,7 +352,7 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|       |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|       |> close(%) | ||||
|     sketch004 = startSketchOn(extrude001, seg05) | ||||
|       |> startProfileAt([82.57, 322.96], %) | ||||
|       |> startProfileAt([82.57,322.96], %) | ||||
|       |> angledLine([0, 11.16], %, $rectangleSegmentA004) | ||||
|       |> angledLine([ | ||||
|            segAng(rectangleSegmentA004) - 90, | ||||
| @ -331,7 +365,7 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|       |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|       |> close(%) | ||||
|     sketch003 = startSketchOn(extrude001, seg04) | ||||
|       |> startProfileAt([-209.64, 255.28], %) | ||||
|       |> startProfileAt([-209.64,255.28], %) | ||||
|       |> angledLine([0, 11.56], %, $rectangleSegmentA003) | ||||
|       |> angledLine([ | ||||
|            segAng(rectangleSegmentA003) - 90, | ||||
| @ -344,7 +378,7 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|       |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|       |> close(%) | ||||
|     sketch002 = startSketchOn(extrude001, seg03) | ||||
|       |> startProfileAt([205.96, 254.59], %) | ||||
|       |> startProfileAt([205.96,254.59], %) | ||||
|       |> angledLine([0, 11.39], %, $rectangleSegmentA002) | ||||
|       |> angledLine([ | ||||
|            segAng(rectangleSegmentA002) - 90, | ||||
| @ -357,43 +391,50 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|       |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|       |> close(%) | ||||
|     `, | ||||
|           { shouldNormalise: true } | ||||
|         ) | ||||
|       }) | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test( | ||||
|     'Works on chamfers that are non in a pipeExpression can break up multi edges in a chamfer array', | ||||
|     { tag: ['@skipWin'] }, | ||||
|     async ({ app, editor, toolbar, scene }) => { | ||||
|       test.skip( | ||||
|         process.platform === 'win32', | ||||
|         'Fails on windows in CI, can not be replicated locally on windows.' | ||||
|         { shouldNormalise: true } | ||||
|       ) | ||||
|       const file = await app.getInputFile( | ||||
|         'e2e-can-sketch-on-chamfer-no-pipeExpr.kcl' | ||||
|       ) | ||||
|       await app.initialise(file) | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|       const sketchOnAChamfer = _sketchOnAChamfer(app, editor, toolbar, scene) | ||||
|   test('Works on chamfers that are non in a pipeExpression can break up multi edges in a chamfer array', async ({ | ||||
|     context, | ||||
|     page, | ||||
|     homePage, | ||||
|     editor, | ||||
|     toolbar, | ||||
|     scene, | ||||
|   }) => { | ||||
|     const file = await fs.readFile( | ||||
|       path.resolve( | ||||
|         __dirname, | ||||
|         '../../', | ||||
|         './src/wasm-lib/tests/executor/inputs/e2e-can-sketch-on-chamfer-no-pipeExpr.kcl' | ||||
|       ), | ||||
|       'utf-8' | ||||
|     ) | ||||
|     await context.addInitScript((file) => { | ||||
|       localStorage.setItem('persistCode', file) | ||||
|     }, file) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|       await sketchOnAChamfer({ | ||||
|         clickCoords: { x: 570, y: 220 }, | ||||
|         cameraPos: { x: 16020, y: -2000, z: 10500 }, | ||||
|         cameraTarget: { x: -150, y: -4500, z: -80 }, | ||||
|         beforeChamferSnippet: `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01) | ||||
|     const sketchOnAChamfer = _sketchOnAChamfer(page, editor, toolbar, scene) | ||||
|  | ||||
|     await sketchOnAChamfer({ | ||||
|       clickCoords: { x: 570, y: 220 }, | ||||
|       cameraPos: { x: 16020, y: -2000, z: 10500 }, | ||||
|       cameraTarget: { x: -150, y: -4500, z: -80 }, | ||||
|       beforeChamferSnippet: `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01) | ||||
|       chamfer({length=30,tags=[ | ||||
|       seg01, | ||||
|       getNextAdjacentEdge(yo), | ||||
|       getNextAdjacentEdge(seg02), | ||||
|       getOppositeEdge(seg01) | ||||
|     ]}, extrude001)`, | ||||
|         beforeChamferSnippetEnd: '}, extrude001)', | ||||
|         afterChamferSelectSnippet: | ||||
|           'sketch002 = startSketchOn(extrude001, seg03)', | ||||
|         afterRectangle1stClickSnippet: 'startProfileAt([205.96, 254.59], %)', | ||||
|         afterRectangle2ndClickSnippet: `angledLine([0, 11.39], %, $rectangleSegmentA002) | ||||
|       beforeChamferSnippetEnd: '}, extrude001)', | ||||
|       afterChamferSelectSnippet: 'sketch002 = startSketchOn(extrude001, seg03)', | ||||
|       afterRectangle1stClickSnippet: 'startProfileAt([205.96, 254.59], %)', | ||||
|       afterRectangle2ndClickSnippet: `angledLine([0, 11.39], %, $rectangleSegmentA002) | ||||
|     |> angledLine([ | ||||
|          segAng(rectangleSegmentA002) - 90, | ||||
|          105.26 | ||||
| @ -404,9 +445,9 @@ test.describe('verify sketch on chamfer works', () => { | ||||
|        ], %, $rectangleSegmentC001) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%)`, | ||||
|       }) | ||||
|       await editor.expectEditor.toContain( | ||||
|         `sketch001 = startSketchOn('XZ') | ||||
|     }) | ||||
|     await editor.expectEditor.toContain( | ||||
|       `sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([75.8, 317.2], %) | ||||
|   |> angledLine([0, 268.43], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
| @ -446,50 +487,56 @@ sketch002 = startSketchOn(extrude001, seg03) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| `, | ||||
|         { shouldNormalise: true } | ||||
|       ) | ||||
|     } | ||||
|   ) | ||||
|       { shouldNormalise: true } | ||||
|     ) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| test(`Verify axis, origin, and horizontal snapping`, async ({ | ||||
|   app, | ||||
|   page, | ||||
|   homePage, | ||||
|   editor, | ||||
|   toolbar, | ||||
|   scene, | ||||
| }) => { | ||||
|   const viewPortSize = { width: 1200, height: 500 } | ||||
|  | ||||
|   await page.setBodyDimensions(viewPortSize) | ||||
|  | ||||
|   await homePage.goToModelingScene() | ||||
|  | ||||
|   // Constants and locators | ||||
|   // These are mappings from screenspace to KCL coordinates, | ||||
|   // until we merge in our coordinate system helpers | ||||
|   const xzPlane = [ | ||||
|     app.viewPortSize.width * 0.65, | ||||
|     app.viewPortSize.height * 0.3, | ||||
|     viewPortSize.width * 0.65, | ||||
|     viewPortSize.height * 0.3, | ||||
|   ] as const | ||||
|   const originSloppy = { | ||||
|     screen: [ | ||||
|       app.viewPortSize.width / 2 + 3, // 3px off the center of the screen | ||||
|       app.viewPortSize.height / 2, | ||||
|       viewPortSize.width / 2 + 3, // 3px off the center of the screen | ||||
|       viewPortSize.height / 2, | ||||
|     ], | ||||
|     kcl: [0, 0], | ||||
|   } as const | ||||
|   const xAxisSloppy = { | ||||
|     screen: [ | ||||
|       app.viewPortSize.width * 0.75, | ||||
|       app.viewPortSize.height / 2 - 3, // 3px off the X-axis | ||||
|       viewPortSize.width * 0.75, | ||||
|       viewPortSize.height / 2 - 3, // 3px off the X-axis | ||||
|     ], | ||||
|     kcl: [16.95, 0], | ||||
|     kcl: [20.34, 0], | ||||
|   } as const | ||||
|   const offYAxis = { | ||||
|     screen: [ | ||||
|       app.viewPortSize.width * 0.6, // Well off the Y-axis, out of snapping range | ||||
|       app.viewPortSize.height * 0.3, | ||||
|       viewPortSize.width * 0.6, // Well off the Y-axis, out of snapping range | ||||
|       viewPortSize.height * 0.3, | ||||
|     ], | ||||
|     kcl: [6.78, 6.78], | ||||
|     kcl: [8.14, 6.78], | ||||
|   } as const | ||||
|   const yAxisSloppy = { | ||||
|     screen: [ | ||||
|       app.viewPortSize.width / 2 + 5, // 5px off the Y-axis | ||||
|       app.viewPortSize.height * 0.3, | ||||
|       viewPortSize.width / 2 + 5, // 5px off the Y-axis | ||||
|       viewPortSize.height * 0.3, | ||||
|     ], | ||||
|     kcl: [0, 6.78], | ||||
|   } as const | ||||
| @ -510,15 +557,13 @@ test(`Verify axis, origin, and horizontal snapping`, async ({ | ||||
|     afterSegmentDraggedOnYAxis: `startProfileAt([${yAxisSloppy.kcl[0]}, ${yAxisSloppy.kcl[1]}], %)`, | ||||
|   } | ||||
|  | ||||
|   await app.initialise() | ||||
|  | ||||
|   await test.step(`Start a sketch on the XZ plane`, async () => { | ||||
|     await editor.closePane() | ||||
|     await toolbar.startSketchPlaneSelection() | ||||
|     await moveToXzPlane() | ||||
|     await clickOnXzPlane() | ||||
|     // timeout wait for engine animation is unavoidable | ||||
|     await app.page.waitForTimeout(600) | ||||
|     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 () => { | ||||
| @ -553,11 +598,15 @@ test(`Verify axis, origin, and horizontal snapping`, async ({ | ||||
| }) | ||||
|  | ||||
| test(`Verify user can double-click to edit a sketch`, async ({ | ||||
|   app, | ||||
|   context, | ||||
|   page, | ||||
|   homePage, | ||||
|   editor, | ||||
|   toolbar, | ||||
|   scene, | ||||
| }) => { | ||||
|   const u = await getUtils(page) | ||||
|  | ||||
|   const initialCode = `closedSketch = startSketchOn('XZ') | ||||
|   |> circle({ center = [8, 5], radius = 2 }, %) | ||||
| openSketch = startSketchOn('XY') | ||||
| @ -566,15 +615,24 @@ openSketch = startSketchOn('XY') | ||||
|   |> xLine(5, %) | ||||
|   |> tangentialArcTo([10, 0], %) | ||||
| ` | ||||
|   await app.initialise(initialCode) | ||||
|   const viewPortSize = { width: 1000, height: 500 } | ||||
|   await page.setBodyDimensions(viewPortSize) | ||||
|  | ||||
|   await context.addInitScript((code) => { | ||||
|     localStorage.setItem('persistCode', code) | ||||
|   }, initialCode) | ||||
|  | ||||
|   await homePage.goToModelingScene() | ||||
|   await u.waitForPageLoad() | ||||
|   await page.waitForTimeout(1000) | ||||
|  | ||||
|   const pointInsideCircle = { | ||||
|     x: app.viewPortSize.width * 0.63, | ||||
|     y: app.viewPortSize.height * 0.5, | ||||
|     x: viewPortSize.width * 0.63, | ||||
|     y: viewPortSize.height * 0.5, | ||||
|   } | ||||
|   const pointOnPathAfterSketching = { | ||||
|     x: app.viewPortSize.width * 0.58, | ||||
|     y: app.viewPortSize.height * 0.5, | ||||
|     x: viewPortSize.width * 0.65, | ||||
|     y: viewPortSize.height * 0.5, | ||||
|   } | ||||
|   // eslint-disable-next-line @typescript-eslint/no-unused-vars | ||||
|   const [_clickOpenPath, moveToOpenPath, dblClickOpenPath] = | ||||
| @ -607,41 +665,59 @@ openSketch = startSketchOn('XY') | ||||
|       diagnostics: [], | ||||
|     }) | ||||
|   }) | ||||
|   await page.waitForTimeout(1000) | ||||
|  | ||||
|   await exitSketch() | ||||
|   await page.waitForTimeout(1000) | ||||
|  | ||||
|   // Drag the sketch line out of the axis view which blocks the click | ||||
|   await page.dragAndDrop('#stream', '#stream', { | ||||
|     sourcePosition: { | ||||
|       x: viewPortSize.width * 0.7, | ||||
|       y: viewPortSize.height * 0.5, | ||||
|     }, | ||||
|     targetPosition: { | ||||
|       x: viewPortSize.width * 0.7, | ||||
|       y: viewPortSize.height * 0.4, | ||||
|     }, | ||||
|   }) | ||||
|  | ||||
|   await page.waitForTimeout(500) | ||||
|  | ||||
|   await test.step(`Double-click on the open sketch`, async () => { | ||||
|     await moveToOpenPath() | ||||
|     await scene.expectPixelColor([250, 250, 250], pointOnPathAfterSketching, 15) | ||||
|     // There is a full execution after exiting sketch that clears the scene. | ||||
|     await app.page.waitForTimeout(500) | ||||
|     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 app.page.waitForTimeout(500) | ||||
|     await page.waitForTimeout(500) | ||||
|     await editor.expectState({ | ||||
|       activeLines: [`|>xLine(5,%)`], | ||||
|       highlightedCode: 'xLine(5,%)', | ||||
|       activeLines: [`|>tangentialArcTo([10,0],%)`], | ||||
|       highlightedCode: 'tangentialArcTo([10,0],%)', | ||||
|       diagnostics: [], | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| test(`Offset plane point-and-click`, async ({ | ||||
|   app, | ||||
|   context, | ||||
|   page, | ||||
|   homePage, | ||||
|   scene, | ||||
|   editor, | ||||
|   toolbar, | ||||
|   cmdBar, | ||||
| }) => { | ||||
|   await app.initialise() | ||||
|  | ||||
|   // One dumb hardcoded screen pixel value | ||||
|   const testPoint = { x: 700, y: 150 } | ||||
|   const [clickOnXzPlane] = scene.makeMouseHelpers(testPoint.x, testPoint.y) | ||||
|   const expectedOutput = `plane001 = offsetPlane('XZ', 5)` | ||||
|  | ||||
|   await homePage.goToModelingScene() | ||||
|  | ||||
|   await test.step(`Look for the blue of the XZ plane`, async () => { | ||||
|     await scene.expectPixelColor([50, 51, 96], testPoint, 15) | ||||
|   }) | ||||
| @ -768,3 +844,179 @@ loftPointAndClickCases.forEach(({ shouldPreselect }) => { | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| const shellPointAndClickCapCases = [ | ||||
|   { shouldPreselect: true }, | ||||
|   { shouldPreselect: false }, | ||||
| ] | ||||
| shellPointAndClickCapCases.forEach(({ shouldPreselect }) => { | ||||
|   test(`Shell point-and-click cap (preselected sketches: ${shouldPreselect})`, async ({ | ||||
|     context, | ||||
|     page, | ||||
|     homePage, | ||||
|     scene, | ||||
|     editor, | ||||
|     toolbar, | ||||
|     cmdBar, | ||||
|   }) => { | ||||
|     const initialCode = `sketch001 = startSketchOn('XZ') | ||||
|     |> circle({ center = [0, 0], radius = 30 }, %) | ||||
|     extrude001 = extrude(30, sketch001) | ||||
|     ` | ||||
|     await context.addInitScript((initialCode) => { | ||||
|       localStorage.setItem('persistCode', initialCode) | ||||
|     }, initialCode) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // One dumb hardcoded screen pixel value | ||||
|     const testPoint = { x: 575, y: 200 } | ||||
|     const [clickOnCap] = scene.makeMouseHelpers(testPoint.x, testPoint.y) | ||||
|     const shellDeclaration = | ||||
|       "shell001 = shell({ faces = ['end'], thickness = 5 }, extrude001)" | ||||
|  | ||||
|     await test.step(`Look for the grey of the shape`, async () => { | ||||
|       await scene.expectPixelColor([127, 127, 127], testPoint, 15) | ||||
|     }) | ||||
|  | ||||
|     if (!shouldPreselect) { | ||||
|       await test.step(`Go through the command bar flow without preselected faces`, async () => { | ||||
|         await toolbar.shellButton.click() | ||||
|         await cmdBar.expectState({ | ||||
|           stage: 'arguments', | ||||
|           currentArgKey: 'selection', | ||||
|           currentArgValue: '', | ||||
|           headerArguments: { | ||||
|             Selection: '', | ||||
|             Thickness: '', | ||||
|           }, | ||||
|           highlightedHeaderArg: 'selection', | ||||
|           commandName: 'Shell', | ||||
|         }) | ||||
|         await clickOnCap() | ||||
|         await page.waitForTimeout(500) | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await cmdBar.expectState({ | ||||
|           stage: 'review', | ||||
|           headerArguments: { | ||||
|             Selection: '1 cap', | ||||
|             Thickness: '5', | ||||
|           }, | ||||
|           commandName: 'Shell', | ||||
|         }) | ||||
|         await cmdBar.progressCmdBar() | ||||
|       }) | ||||
|     } else { | ||||
|       await test.step(`Preselect the cap`, async () => { | ||||
|         await clickOnCap() | ||||
|         await page.waitForTimeout(500) | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Go through the command bar flow with a preselected face (cap)`, async () => { | ||||
|         await toolbar.shellButton.click() | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await cmdBar.progressCmdBar() | ||||
|         await cmdBar.expectState({ | ||||
|           stage: 'review', | ||||
|           headerArguments: { | ||||
|             Selection: '1 cap', | ||||
|             Thickness: '5', | ||||
|           }, | ||||
|           commandName: 'Shell', | ||||
|         }) | ||||
|         await cmdBar.progressCmdBar() | ||||
|       }) | ||||
|     } | ||||
|  | ||||
|     await test.step(`Confirm code is added to the editor, scene has changed`, async () => { | ||||
|       await editor.expectEditor.toContain(shellDeclaration) | ||||
|       await editor.expectState({ | ||||
|         diagnostics: [], | ||||
|         activeLines: [shellDeclaration], | ||||
|         highlightedCode: '', | ||||
|       }) | ||||
|       await scene.expectPixelColor([146, 146, 146], testPoint, 15) | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| test('Shell point-and-click wall', async ({ | ||||
|   context, | ||||
|   page, | ||||
|   homePage, | ||||
|   scene, | ||||
|   editor, | ||||
|   toolbar, | ||||
|   cmdBar, | ||||
| }) => { | ||||
|   const initialCode = `sketch001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-20, 20], %) | ||||
|   |> xLine(40, %) | ||||
|   |> yLine(-60, %) | ||||
|   |> xLine(-40, %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| extrude001 = extrude(40, sketch001) | ||||
|   ` | ||||
|   await context.addInitScript((initialCode) => { | ||||
|     localStorage.setItem('persistCode', initialCode) | ||||
|   }, initialCode) | ||||
|   await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|   await homePage.goToModelingScene() | ||||
|  | ||||
|   // One dumb hardcoded screen pixel value | ||||
|   const testPoint = { x: 580, y: 180 } | ||||
|   const [clickOnCap] = scene.makeMouseHelpers(testPoint.x, testPoint.y) | ||||
|   const [clickOnWall] = scene.makeMouseHelpers(testPoint.x, testPoint.y + 70) | ||||
|   const mutatedCode = 'xLine(-40, %, $seg01)' | ||||
|   const shellDeclaration = | ||||
|     "shell001 = shell({  faces = ['end', seg01],  thickness = 5}, extrude001)" | ||||
|   const formattedOutLastLine = '}, extrude001)' | ||||
|  | ||||
|   await test.step(`Look for the grey of the shape`, async () => { | ||||
|     await scene.expectPixelColor([99, 99, 99], testPoint, 15) | ||||
|   }) | ||||
|  | ||||
|   await test.step(`Go through the command bar flow, selecting a wall and keeping default thickness`, async () => { | ||||
|     await toolbar.shellButton.click() | ||||
|     await cmdBar.expectState({ | ||||
|       stage: 'arguments', | ||||
|       currentArgKey: 'selection', | ||||
|       currentArgValue: '', | ||||
|       headerArguments: { | ||||
|         Selection: '', | ||||
|         Thickness: '', | ||||
|       }, | ||||
|       highlightedHeaderArg: 'selection', | ||||
|       commandName: 'Shell', | ||||
|     }) | ||||
|     await clickOnCap() | ||||
|     await page.keyboard.down('Shift') | ||||
|     await clickOnWall() | ||||
|     await page.waitForTimeout(500) | ||||
|     await page.keyboard.up('Shift') | ||||
|     await cmdBar.progressCmdBar() | ||||
|     await cmdBar.progressCmdBar() | ||||
|     await cmdBar.expectState({ | ||||
|       stage: 'review', | ||||
|       headerArguments: { | ||||
|         Selection: '1 cap, 1 face', | ||||
|         Thickness: '5', | ||||
|       }, | ||||
|       commandName: 'Shell', | ||||
|     }) | ||||
|     await cmdBar.progressCmdBar() | ||||
|   }) | ||||
|  | ||||
|   await test.step(`Confirm code is added to the editor, scene has changed`, async () => { | ||||
|     await editor.expectEditor.toContain(mutatedCode) | ||||
|     await editor.expectEditor.toContain(shellDeclaration) | ||||
|     await editor.expectState({ | ||||
|       diagnostics: [], | ||||
|       activeLines: [formattedOutLastLine], | ||||
|       highlightedCode: '', | ||||
|     }) | ||||
|     await scene.expectPixelColor([49, 49, 49], testPoint, 15) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -1,46 +1,37 @@ | ||||
| import { test, expect, Page } from '@playwright/test' | ||||
| import { join } from 'path' | ||||
| import { test, expect, Page } from './zoo-test' | ||||
| import path from 'path' | ||||
| import * as fsp from 'fs/promises' | ||||
| import { | ||||
|   getUtils, | ||||
|   setup, | ||||
|   setupElectron, | ||||
|   tearDown, | ||||
|   executorInputPath, | ||||
| } from './test-utils' | ||||
| import { getUtils, executorInputPath } from './test-utils' | ||||
| import { TEST_CODE_TRIGGER_ENGINE_EXPORT_ERROR } from './storageStates' | ||||
| import { bracket } from 'lib/exampleKcl' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.describe('Regression tests', () => { | ||||
|   // bugs we found that don't fit neatly into other categories | ||||
|   test('bad model has inline error #3251', async ({ page }) => { | ||||
|   test('bad model has inline error #3251', async ({ | ||||
|     context, | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     // because the model has `line([0,0]..` it is valid code, but the model is invalid | ||||
|     // regression test for https://github.com/KittyCAD/modeling-app/issues/3251 | ||||
|     // Since the bad model also found as issue with the artifact graph, which in tern blocked the editor diognostics | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|     await context.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch2 = startSketchOn("XY") | ||||
| sketch001 = startSketchAt([-0, -0]) | ||||
|   |> line([0, 0], %) | ||||
|   |> line([-4.84, -5.29], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%)` | ||||
|   sketch001 = startSketchAt([-0, -0]) | ||||
|     |> line([0, 0], %) | ||||
|     |> line([-4.84, -5.29], %) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%)` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     // error in guter | ||||
|     await expect(page.locator('.cm-lint-marker-error')).toBeVisible() | ||||
| @ -56,6 +47,7 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|   }) | ||||
|   test('user should not have to press down twice in cmdbar', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     // because the model has `line([0,0]..` it is valid code, but the model is invalid | ||||
|     // regression test for https://github.com/KittyCAD/modeling-app/issues/3251 | ||||
| @ -64,26 +56,38 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch2 = startSketchOn("XY") | ||||
| sketch001 = startSketchAt([-0, -0]) | ||||
|   |> line([0, 0], %) | ||||
|   |> line([-4.84, -5.29], %) | ||||
|         `sketch001 = startSketchOn('XY') | ||||
|   |> startProfileAt([82.33, 238.21], %) | ||||
|   |> angledLine([0, 288.63], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        197.97 | ||||
|      ], %, $rectangleSegmentB001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $rectangleSegmentC001) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%)` | ||||
|   |> close(%) | ||||
| extrude001 = extrude(50, sketch001) | ||||
| ` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await page.goto('/') | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     await test.step('Check arrow down works', async () => { | ||||
|       await page.getByTestId('command-bar-open-button').hover() | ||||
|       await page.getByTestId('command-bar-open-button').click() | ||||
|  | ||||
|       await page | ||||
|         .getByRole('option', { name: 'floppy disk arrow Export' }) | ||||
|         .click() | ||||
|       const floppy = page.getByRole('option', { | ||||
|         name: 'floppy disk arrow Export', | ||||
|       }) | ||||
|  | ||||
|       await floppy.click() | ||||
|  | ||||
|       // press arrow down key twice | ||||
|       await page.keyboard.press('ArrowDown') | ||||
| @ -115,21 +119,22 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|       ) | ||||
|     }) | ||||
|   }) | ||||
|   test('executes on load', async ({ page }) => { | ||||
|   test('executes on load', async ({ page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('-XZ') | ||||
|     |> startProfileAt([-6.95, 4.98], %) | ||||
|     |> line([25.1, 0.41], %) | ||||
|     |> line([0.73, -14.93], %) | ||||
|     |> line([-23.44, 0.52], %)` | ||||
|   |> startProfileAt([-6.95, 4.98], %) | ||||
|   |> line([25.1, 0.41], %) | ||||
|   |> line([0.73, -14.93], %) | ||||
|   |> line([-23.44, 0.52], %)` | ||||
|       ) | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     // expand variables section | ||||
|     const variablesTabButton = page.getByTestId('variables-pane-button') | ||||
| @ -148,14 +153,15 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|     ).toBeVisible() | ||||
|   }) | ||||
|  | ||||
|   test('re-executes', async ({ page }) => { | ||||
|   test('re-executes', async ({ page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem('persistCode', `myVar = 5`) | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     const variablesTabButton = page.getByTestId('variables-pane-button') | ||||
|     await variablesTabButton.click() | ||||
| @ -174,32 +180,33 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|       page.locator('.pretty-json-container >> text=myVar:67') | ||||
|     ).toBeVisible() | ||||
|   }) | ||||
|   test('ProgramMemory can be serialised', async ({ page }) => { | ||||
|   test('ProgramMemory can be serialised', async ({ page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `part = startSketchOn('XY') | ||||
|     |> startProfileAt([0, 0], %) | ||||
|     |> line([0, 1], %) | ||||
|     |> line([1, 0], %) | ||||
|     |> line([0, -1], %) | ||||
|     |> close(%) | ||||
|     |> extrude(1, %) | ||||
|     |> patternLinear3d({ | ||||
|           axis: [1, 0, 1], | ||||
|           repetitions: 3, | ||||
|           distance: 6 | ||||
|         }, %)` | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([0, 1], %) | ||||
|   |> line([1, 0], %) | ||||
|   |> line([0, -1], %) | ||||
|   |> close(%) | ||||
|   |> extrude(1, %) | ||||
|   |> patternLinear3d({ | ||||
|         axis: [1, 0, 1], | ||||
|         repetitions: 3, | ||||
|         distance: 6 | ||||
|       }, %)` | ||||
|       ) | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     const messages: string[] = [] | ||||
|  | ||||
|     // Listen for all console events and push the message text to an array | ||||
|     page.on('console', (message) => messages.push(message.text())) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     // wait for execution done | ||||
|     await u.openDebugPanel() | ||||
| @ -212,19 +219,26 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|       }) | ||||
|     }) | ||||
|   }) | ||||
|   test('ensure the Zoo logo is not a link in browser app', async ({ page }) => { | ||||
|  | ||||
|   // Not relevant to us anymore, or at least for the time being. | ||||
|   test.skip('ensure the Zoo logo is not a link in browser app', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     const zooLogo = page.locator('[data-testid="app-logo"]') | ||||
|     // Make sure it's not a link | ||||
|     await expect(zooLogo).not.toHaveAttribute('href') | ||||
|   }) | ||||
|  | ||||
|   test( | ||||
|     'Position _ Is Out Of Range... regression test', | ||||
|     { tag: ['@skipWin'] }, | ||||
|     async ({ page }) => { | ||||
|     async ({ context, page, homePage }) => { | ||||
|       // SKip on windows, its being weird. | ||||
|       test.skip( | ||||
|         process.platform === 'win32', | ||||
| @ -233,25 +247,26 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|  | ||||
|       const u = await getUtils(page) | ||||
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await page.addInitScript(async () => { | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await context.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `exampleSketch = startSketchOn("XZ") | ||||
|     |> startProfileAt([0, 0], %) | ||||
|     |> angledLine({ angle: 50, length: 45 }, %) | ||||
|     |> yLineTo(0, %) | ||||
|     |> close(%) | ||||
|     |> | ||||
|  | ||||
|   example = extrude(5, exampleSketch) | ||||
|   shell({ faces: ['end'], thickness: 0.25 }, exampleSketch)` | ||||
|       |> startProfileAt([0, 0], %) | ||||
|       |> angledLine({ angle: 50, length: 45 }, %) | ||||
|       |> yLineTo(0, %) | ||||
|       |> close(%) | ||||
|       |> | ||||
|    | ||||
|     example = extrude(5, exampleSketch) | ||||
|     shell({ faces: ['end'], thickness: 0.25 }, exampleSketch)` | ||||
|         ) | ||||
|       }) | ||||
|  | ||||
|       await expect(async () => { | ||||
|         await page.goto('/') | ||||
|         await homePage.goToModelingScene() | ||||
|         await u.waitForPageLoad() | ||||
|  | ||||
|         // error in guter | ||||
|         await expect(page.locator('.cm-lint-marker-error')).toBeVisible({ | ||||
|           timeout: 1_000, | ||||
| @ -293,12 +308,12 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|  | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toContainText(`exampleSketch = startSketchOn("XZ") | ||||
|     |> startProfileAt([0, 0], %) | ||||
|     |> angledLine({ angle: 50, length: 45 }, %) | ||||
|     |> yLineTo(0, %) | ||||
|     |> close(%) | ||||
|  | ||||
|     thing: "blah"`) | ||||
|       |> startProfileAt([0, 0], %) | ||||
|       |> angledLine({ angle: 50, length: 45 }, %) | ||||
|       |> yLineTo(0, %) | ||||
|       |> close(%) | ||||
|    | ||||
|       thing: "blah"`) | ||||
|  | ||||
|       await expect(page.locator('.cm-lint-marker-error')).toBeVisible() | ||||
|     } | ||||
| @ -306,6 +321,7 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|  | ||||
|   test('when engine fails export we handle the failure and alert the user', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript( | ||||
| @ -316,9 +332,10 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|       { code: TEST_CODE_TRIGGER_ENGINE_EXPORT_ERROR } | ||||
|     ) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     // wait for execution done | ||||
|     await u.openDebugPanel() | ||||
| @ -374,7 +391,6 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|  | ||||
|     // wait for execution done | ||||
|     await u.openDebugPanel() | ||||
|     await u.clearCommandLogs() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
| @ -408,7 +424,7 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|   test( | ||||
|     'ensure you can not export while an export is already going', | ||||
|     { tag: ['@skipLinux', '@skipWin'] }, | ||||
|     async ({ page }) => { | ||||
|     async ({ page, homePage }) => { | ||||
|       // This is being weird on ubuntu and windows. | ||||
|       test.skip( | ||||
|         // eslint-disable-next-line jest/valid-title | ||||
| @ -428,9 +444,10 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|           } | ||||
|         ) | ||||
|  | ||||
|         await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|         await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|         await u.waitForAuthSkipAppStart() | ||||
|         await homePage.goToModelingScene() | ||||
|         await u.waitForPageLoad() | ||||
|  | ||||
|         // wait for execution done | ||||
|         await u.openDebugPanel() | ||||
| @ -500,20 +517,17 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|   test( | ||||
|     `Network health indicator only appears in modeling view`, | ||||
|     { tag: '@electron' }, | ||||
|     async ({ browserName: _ }, testInfo) => { | ||||
|       const { electronApp, page } = await setupElectron({ | ||||
|         testInfo, | ||||
|         folderSetupFn: async (dir) => { | ||||
|           const bracketDir = join(dir, 'bracket') | ||||
|           await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|           await fsp.copyFile( | ||||
|             executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|             join(bracketDir, 'main.kcl') | ||||
|           ) | ||||
|         }, | ||||
|     async ({ context, page }, testInfo) => { | ||||
|       await context.folderSetupFn(async (dir) => { | ||||
|         const bracketDir = path.join(dir, 'bracket') | ||||
|         await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|           path.join(bracketDir, 'main.kcl') | ||||
|         ) | ||||
|       }) | ||||
|  | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       const u = await getUtils(page) | ||||
|  | ||||
|       // Locators | ||||
| @ -539,18 +553,17 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|         await u.waitForPageLoad() | ||||
|         await expect(networkHealthIndicator).toContainText('Connected') | ||||
|       }) | ||||
|  | ||||
|       await electronApp.close() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test(`View gizmo stays visible even when zoomed out all the way`, async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     // Constants and locators | ||||
|     const planeColor: [number, number, number] = [161, 220, 155] | ||||
|     const planeColor: [number, number, number] = [170, 220, 170] | ||||
|     const bgColor: [number, number, number] = [27, 27, 27] | ||||
|     const middlePixelIsColor = async (color: [number, number, number]) => { | ||||
|       return u.getGreatestPixDiff({ x: 600, y: 250 }, color) | ||||
| @ -561,8 +574,9 @@ sketch001 = startSketchAt([-0, -0]) | ||||
|       await page.addInitScript(async () => { | ||||
|         localStorage.setItem('persistCode', '') | ||||
|       }) | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await u.waitForPageLoad() | ||||
|       await u.closeKclCodePanel() | ||||
|     }) | ||||
|  | ||||
|  | ||||
| @ -47,7 +47,11 @@ test.beforeEach(async ({ page }) => { | ||||
|  | ||||
| test.setTimeout(60_000) | ||||
|  | ||||
| test( | ||||
| // We test this end to end already - getting this to work on web just to take | ||||
| // a snapshot of it feels weird. I'd rather our regular tests fail. | ||||
| // The primary failure is doExport now relies on the filesystem. We can follow | ||||
| // up with another PR if we want this back. | ||||
| test.skip( | ||||
|   'exports of each format should work', | ||||
|   { tag: ['@snapshot', '@skipWin', '@skipMacos'] }, | ||||
|   async ({ page, context }) => { | ||||
| @ -950,7 +954,75 @@ test( | ||||
|  | ||||
| test.describe('Grid visibility', { tag: '@snapshot' }, () => { | ||||
|   // FIXME: Skip on macos its being weird. | ||||
|   test.skip(process.platform === 'darwin', 'Skip on macos') | ||||
|   // test.skip(process.platform === 'darwin', 'Skip on macos') | ||||
|  | ||||
|   test('Grid turned off to on via command bar', async ({ page }) => { | ||||
|     const u = await getUtils(page) | ||||
|     const stream = page.getByTestId('stream') | ||||
|     const mask = [ | ||||
|       page.locator('#app-header'), | ||||
|       page.locator('#sidebar-top-ribbon'), | ||||
|       page.locator('#sidebar-bottom-ribbon'), | ||||
|     ] | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.goto('/') | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|     // wait for execution done | ||||
|     await expect( | ||||
|       page.locator('[data-message-type="execution-done"]') | ||||
|     ).toHaveCount(1) | ||||
|     await u.closeDebugPanel() | ||||
|     await u.closeKclCodePanel() | ||||
|     // TODO: Find a way to truly know that the objects have finished | ||||
|     // rendering, because an execution-done message is not sufficient. | ||||
|     await page.waitForTimeout(1000) | ||||
|  | ||||
|     // Open the command bar. | ||||
|     await page | ||||
|       .getByRole('button', { name: 'Commands', exact: false }) | ||||
|       .or(page.getByRole('button', { name: '⌘K' })) | ||||
|       .click() | ||||
|     const commandName = 'show scale grid' | ||||
|     const commandOption = page.getByRole('option', { | ||||
|       name: commandName, | ||||
|       exact: false, | ||||
|     }) | ||||
|     const cmdSearchBar = page.getByPlaceholder('Search commands') | ||||
|     // This selector changes after we set the setting | ||||
|     await cmdSearchBar.fill(commandName) | ||||
|     await expect(commandOption).toBeVisible() | ||||
|     await commandOption.click() | ||||
|  | ||||
|     const toggleInput = page.getByPlaceholder('Off') | ||||
|     await expect(toggleInput).toBeVisible() | ||||
|     await expect(toggleInput).toBeFocused() | ||||
|  | ||||
|     // Select On | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await expect(page.getByRole('option', { name: 'Off' })).toHaveAttribute( | ||||
|       'data-headlessui-state', | ||||
|       'active selected' | ||||
|     ) | ||||
|     await page.keyboard.press('ArrowUp') | ||||
|     await expect(page.getByRole('option', { name: 'On' })).toHaveAttribute( | ||||
|       'data-headlessui-state', | ||||
|       'active' | ||||
|     ) | ||||
|     await page.keyboard.press('Enter') | ||||
|  | ||||
|     // Check the toast appeared | ||||
|     await expect( | ||||
|       page.getByText(`Set show scale grid to "true" as a user default`) | ||||
|     ).toBeVisible() | ||||
|  | ||||
|     await expect(stream).toHaveScreenshot({ | ||||
|       maxDiffPixels: 100, | ||||
|       mask, | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   test('Grid turned off', async ({ page }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
| After Width: | Height: | Size: 52 KiB | 
| After Width: | Height: | Size: 54 KiB | 
| Before Width: | Height: | Size: 37 KiB After Width: | Height: | Size: 37 KiB | 
| Before Width: | Height: | Size: 40 KiB After Width: | Height: | Size: 40 KiB | 
| Before Width: | Height: | Size: 40 KiB After Width: | Height: | Size: 40 KiB | 
| Before Width: | Height: | Size: 38 KiB After Width: | Height: | Size: 38 KiB | 
| Before Width: | Height: | Size: 42 KiB After Width: | Height: | Size: 42 KiB | 
| Before Width: | Height: | Size: 36 KiB After Width: | Height: | Size: 36 KiB | 
| @ -109,242 +109,21 @@ keychain = startSketchOn("XY") | ||||
|   |> close(%) | ||||
|   |> extrude(thickness, %) | ||||
|  | ||||
| // generated from  /home/paultag/Downloads/zma-logomark.svg | ||||
| fn svg = (surface, origin, depth) => { | ||||
|   let a0 = surface |> startProfileAt([origin[0] + 45.430427, origin[1] + -14.627736], %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0, 0.764157 ], | ||||
|     control2: [ 0, 1.528314 ], | ||||
|     to: [ 0, 2.292469 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -3.03202, 0 ], | ||||
|     control2: [ -6.064039, 0 ], | ||||
|     to: [ -9.09606, 0 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0, -1.077657 ], | ||||
|     control2: [ 0, -2.155312 ], | ||||
|     to: [ 0, -3.232969 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 2.741805, 0 ], | ||||
|     control2: [ 5.483613, 0 ], | ||||
|     to: [ 8.225417, 0 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -2.740682, -2.961815 ], | ||||
|     control2: [ -5.490342, -5.925794 ], | ||||
|     to: [ -8.225417, -8.886255 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0, -0.723995 ], | ||||
|     control2: [ 0, -1.447988 ], | ||||
|     to: [ 0, -2.171981 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0.712124, 0.05061 ], | ||||
|     control2: [ 1.511636, -0.09877 ], | ||||
|     to: [ 2.172096, 0.07005 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0.68573, 0.740811 ], | ||||
|     control2: [ 1.371459, 1.481622 ], | ||||
|     to: [ 2.057187, 2.222436 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0, -0.76416 ], | ||||
|     control2: [ 0, -1.52832 ], | ||||
|     to: [ 0, -2.29248 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 3.032013, 0 ], | ||||
|     control2: [ 6.064026, 0 ], | ||||
|     to: [ 9.096038, 0 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0, 1.077657 ], | ||||
|     control2: [ 0, 2.155314 ], | ||||
|     to: [ 0, 3.232973 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -2.741312, 0 ], | ||||
|     control2: [ -5.482623, 0 ], | ||||
|     to: [ -8.223936, 0 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 2.741313, 2.961108 ], | ||||
|     control2: [ 5.482624, 5.922216 ], | ||||
|     to: [ 8.223936, 8.883325 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0, 0.724968 ], | ||||
|     control2: [ 0, 1.449938 ], | ||||
|     to: [ 0, 2.174907 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -0.712656, -0.05145 ], | ||||
|     control2: [ -1.512554, 0.09643 ], | ||||
|     to: [ -2.173592, -0.07298 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -0.685222, -0.739834 ], | ||||
|     control2: [ -1.370445, -1.479669 ], | ||||
|     to: [ -2.055669, -2.219505 ] | ||||
|    }, %) | ||||
|     |> close(%) | ||||
|     |> extrude(depth, %) | ||||
| keychain1 = startSketchOn("XY") | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> lineTo([width, 0], %) | ||||
|   |> lineTo([width, height], %) | ||||
|   |> lineTo([0, height], %) | ||||
|   |> close(%) | ||||
|   |> extrude(thickness, %) | ||||
|  | ||||
|   let a1 = surface |> startProfileAt([origin[0] + 57.920488, origin[1] + -15.244943], %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -2.78904, 0.106635 ], | ||||
|     control2: [ -5.052548, -2.969529 ], | ||||
|     to: [ -4.055141, -5.598369 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0.841523, -0.918736 ], | ||||
|     control2: [ 0.439412, -1.541892 ], | ||||
|     to: [ -0.368488, -2.214378 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -0.418245, -0.448461 ], | ||||
|     control2: [ -0.836489, -0.896922 ], | ||||
|     to: [ -1.254732, -1.345384 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -2.76806, 2.995359 ], | ||||
|     control2: [ -2.32667, 8.18409 ], | ||||
|     to: [ 0.897655, 10.678932 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 2.562822, 2.186098 ], | ||||
|     control2: [ 6.605111, 2.28043 ], | ||||
|     to: [ 9.271202, 0.226476 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -0.743744, -0.797465 ], | ||||
|     control2: [ -1.487487, -1.594932 ], | ||||
|     to: [ -2.231232, -2.392397 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -0.672938, 0.421422 ], | ||||
|     control2: [ -1.465362, 0.646946 ], | ||||
|     to: [ -2.259264, 0.64512 ] | ||||
|    }, %) | ||||
|     |> close(%) | ||||
|     |> extrude(depth, %) | ||||
|  | ||||
|   let a2 = surface |> startProfileAt([origin[0] + 62.19406300000001, origin[1] + -19.500698999999997], %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0.302938, 1.281141 ], | ||||
|     control2: [ -1.53575, 2.434288 ], | ||||
|     to: [ -0.10908, 3.279477 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0.504637, 0.54145 ], | ||||
|     control2: [ 1.009273, 1.082899 ], | ||||
|     to: [ 1.513909, 1.624348 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 2.767778, -2.995425 ], | ||||
|     control2: [ 2.327135, -8.184384 ], | ||||
|     to: [ -0.897661, -10.679047 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -2.562947, -2.186022 ], | ||||
|     control2: [ -6.604089, -2.279606 ], | ||||
|     to: [ -9.271196, -0.227813 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0.744231, 0.797952 ], | ||||
|     control2: [ 1.488461, 1.595904 ], | ||||
|     to: [ 2.232692, 2.393856 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 2.302377, -1.564629 ], | ||||
|     control2: [ 5.793126, -0.15358 ], | ||||
|     to: [ 6.396577, 2.547372 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0.08981, 0.346302 ], | ||||
|     control2: [ 0.134865, 0.704078 ], | ||||
|     to: [ 0.13476, 1.061807 ] | ||||
|    }, %) | ||||
|     |> close(%) | ||||
|     |> extrude(depth, %) | ||||
|  | ||||
|   let a3 = surface |> startProfileAt([origin[0] + 74.124866, origin[1] + -15.244943], %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -2.78904, 0.106635 ], | ||||
|     control2: [ -5.052549, -2.969529 ], | ||||
|     to: [ -4.055142, -5.598369 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0.841527, -0.918738 ], | ||||
|     control2: [ 0.43941, -1.541892 ], | ||||
|     to: [ -0.368497, -2.214367 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -0.418254, -0.448466 ], | ||||
|     control2: [ -0.836507, -0.896931 ], | ||||
|     to: [ -1.254761, -1.345395 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -2.768019, 2.995371 ], | ||||
|     control2: [ -2.326624, 8.184088 ], | ||||
|     to: [ 0.897678, 10.678932 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 2.56289, 2.186191 ], | ||||
|     control2: [ 6.60516, 2.280307 ], | ||||
|     to: [ 9.271371, 0.226476 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -0.743808, -0.797465 ], | ||||
|     control2: [ -1.487616, -1.594932 ], | ||||
|     to: [ -2.231424, -2.392397 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -0.672916, 0.421433 ], | ||||
|     control2: [ -1.465344, 0.646926 ], | ||||
|     to: [ -2.259225, 0.64512 ] | ||||
|    }, %) | ||||
|     |> close(%) | ||||
|     |> extrude(depth, %) | ||||
|  | ||||
|   let a4 = surface |> startProfileAt([origin[0] + 77.57333899999998, origin[1] + -16.989262999999998], %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0.743298, 0.797463 ], | ||||
|     control2: [ 1.486592, 1.594926 ], | ||||
|     to: [ 2.229888, 2.392389 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 2.767827, -2.995393 ], | ||||
|     control2: [ 2.327103, -8.184396 ], | ||||
|     to: [ -0.897672, -10.679047 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ -2.562939, -2.186037 ], | ||||
|     control2: [ -6.604077, -2.279589 ], | ||||
|     to: [ -9.271185, -0.227813 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0.744243, 0.797952 ], | ||||
|     control2: [ 1.488486, 1.595904 ], | ||||
|     to: [ 2.232729, 2.393856 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 2.302394, -1.564623 ], | ||||
|     control2: [ 5.793201, -0.153598 ], | ||||
|     to: [ 6.396692, 2.547372 ] | ||||
|    }, %) | ||||
|     |> bezierCurve({ | ||||
|     control1: [ 0.32074, 1.215468 ], | ||||
|     control2: [ 0.06159, 2.564765 ], | ||||
|     to: [ -0.690452, 3.573243 ] | ||||
|    }, %) | ||||
|     |> close(%) | ||||
|     |> extrude(depth, %) | ||||
| keychain2 = startSketchOn("XY") | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> lineTo([width, 0], %) | ||||
|   |> lineTo([width, height], %) | ||||
|   |> lineTo([0, height], %) | ||||
|   |> close(%) | ||||
|   |> extrude(thickness, %) | ||||
|  | ||||
| box = startSketchOn('XY') | ||||
|   |> startProfileAt([0, 0], %) | ||||
| @ -354,7 +133,7 @@ box = startSketchOn('XY') | ||||
|   |> close(%) | ||||
|   |> extrude(10, %) | ||||
|  | ||||
|   sketch001 = startSketchOn(box, revolveAxis) | ||||
| sketch001 = startSketchOn(box, revolveAxis) | ||||
|   |> startProfileAt([5, 10], %) | ||||
|   |> line([0, -10], %) | ||||
|   |> line([2, 0], %) | ||||
| @ -364,18 +143,12 @@ box = startSketchOn('XY') | ||||
|   axis: revolveAxis, | ||||
|   angle: 90 | ||||
|   }, %) | ||||
|   return 0 | ||||
| } | ||||
|  | ||||
| sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([0.0, 0.0], %) | ||||
|   |> xLine(0.0, %) | ||||
|   |> close(%) | ||||
|  | ||||
|  | ||||
| svg(startSketchOn(keychain, 'end'), [-33, 32], -thickness) | ||||
|  | ||||
| startSketchOn(keychain, 'end') | ||||
|   |> circle({ center: [ | ||||
|        width / 2, | ||||
|        height - (keychainHoleSize + 1.5) | ||||
|      ], radius: keychainHoleSize }, %) | ||||
|   |> extrude(-thickness, %)` | ||||
| ` | ||||
|  | ||||
| export const TEST_CODE_TRIGGER_ENGINE_EXPORT_ERROR = `thing = 1` | ||||
|  | ||||
| @ -1,29 +1,16 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
|  | ||||
| import { commonPoints, getUtils, setup, tearDown } from './test-utils' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
| import { commonPoints, getUtils } from './test-utils' | ||||
|  | ||||
| test.describe('Test network and connection issues', () => { | ||||
|   test('simulate network down and network little widget', async ({ | ||||
|     page, | ||||
|     browserName, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     // TODO: Don't skip Mac for these. After `window.tearDown` is working in Safari, these should work on webkit | ||||
|     test.skip( | ||||
|       browserName === 'webkit', | ||||
|       'Skip on Safari until `window.tearDown` is working there' | ||||
|     ) | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     const networkToggle = page.getByTestId('network-toggle') | ||||
|  | ||||
| @ -62,7 +49,7 @@ test.describe('Test network and connection issues', () => { | ||||
|     }) | ||||
|  | ||||
|     // Expect the network to be down | ||||
|     await expect(networkToggle).toContainText('Offline') | ||||
|     await expect(networkToggle).toContainText('Problem') | ||||
|  | ||||
|     // Click the network widget | ||||
|     await networkWidget.click() | ||||
| @ -93,26 +80,19 @@ test.describe('Test network and connection issues', () => { | ||||
|  | ||||
|   test('Engine disconnect & reconnect in sketch mode', async ({ | ||||
|     page, | ||||
|     browserName, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     // TODO: Don't skip Mac for these. After `window.tearDown` is working in Safari, these should work on webkit | ||||
|     test.skip( | ||||
|       browserName === 'webkit', | ||||
|       'Skip on Safari until `window.tearDown` is working there' | ||||
|     ) | ||||
|     const networkToggle = page.getByTestId('network-toggle') | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|     const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|  | ||||
|     await expect( | ||||
|       page.getByRole('button', { name: 'Start Sketch' }) | ||||
|     ).not.toBeDisabled({ timeout: 15000 }) | ||||
|  | ||||
|     // click on "Start Sketch" button | ||||
|     await u.clearCommandLogs() | ||||
|     await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||
| @ -132,7 +112,7 @@ test.describe('Test network and connection issues', () => { | ||||
|     await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt(${commonPoints.startAt}, %)`) | ||||
|   |> startProfileAt(${commonPoints.startAt}, %)`) | ||||
|     await page.waitForTimeout(100) | ||||
|  | ||||
|     await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) | ||||
| @ -140,8 +120,8 @@ test.describe('Test network and connection issues', () => { | ||||
|  | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt(${commonPoints.startAt}, %) | ||||
|     |> xLine(${commonPoints.num1}, %)`) | ||||
|   |> startProfileAt(${commonPoints.startAt}, %) | ||||
|   |> xLine(${commonPoints.num1}, %)`) | ||||
|  | ||||
|     // Expect the network to be up | ||||
|     await expect(networkToggle).toContainText('Connected') | ||||
| @ -156,7 +136,7 @@ test.describe('Test network and connection issues', () => { | ||||
|     }) | ||||
|  | ||||
|     // Expect the network to be down | ||||
|     await expect(networkToggle).toContainText('Offline') | ||||
|     await expect(networkToggle).toContainText('Problem') | ||||
|  | ||||
|     // Ensure we are not in sketch mode | ||||
|     await expect( | ||||
|  | ||||
| @ -1,22 +1,20 @@ | ||||
| import { | ||||
|   expect, | ||||
|   Page, | ||||
|   Download, | ||||
|   BrowserContext, | ||||
|   TestInfo, | ||||
|   _electron as electron, | ||||
|   Locator, | ||||
|   test, | ||||
| } from '@playwright/test' | ||||
| import { test, Page } from './zoo-test' | ||||
| import { EngineCommand } from 'lang/std/artifactGraph' | ||||
| import fsp from 'fs/promises' | ||||
| import fsSync from 'fs' | ||||
| import { join } from 'path' | ||||
| import path from 'path' | ||||
| import pixelMatch from 'pixelmatch' | ||||
| import { PNG } from 'pngjs' | ||||
| import { Protocol } from 'playwright-core/types/protocol' | ||||
| import type { Models } from '@kittycad/lib' | ||||
| import { APP_NAME, COOKIE_NAME } from 'lib/constants' | ||||
| import { COOKIE_NAME } from 'lib/constants' | ||||
| import { secrets } from './secrets' | ||||
| import { | ||||
|   TEST_SETTINGS_KEY, | ||||
| @ -30,6 +28,134 @@ import { isErrorWhitelisted } from './lib/console-error-whitelist' | ||||
| import { isArray } from 'lib/utils' | ||||
| import { reportRejection } from 'lib/trap' | ||||
|  | ||||
| // The below is copied from playwright-core because it exports none of them :( | ||||
| import { Env, BrowserContextOptions } from 'playwright-core' | ||||
| import type * as channels from '@protocol/channels' | ||||
|  | ||||
| // Copied from playwright-core | ||||
| function envObjectToArray(env: Env): { name: string; value: string }[] { | ||||
|   const result: { name: string; value: string }[] = [] | ||||
|   for (const name in env) { | ||||
|     if (!Object.is(env[name], undefined)) | ||||
|       result.push({ name, value: String(env[name]) }) | ||||
|   } | ||||
|   return result | ||||
| } | ||||
|  | ||||
| // Copied from playwright-core | ||||
| export async function toClientCertificatesProtocol( | ||||
|   certs?: BrowserContextOptions['clientCertificates'] | ||||
| ): Promise<channels.PlaywrightNewRequestParams['clientCertificates']> { | ||||
|   if (!certs) return undefined | ||||
|  | ||||
|   const bufferizeContent = async ( | ||||
|     value?: Buffer, | ||||
|     path?: string | ||||
|   ): Promise<Buffer | undefined> => { | ||||
|     if (value) return value | ||||
|     if (path) return await fs.promises.readFile(path) | ||||
|   } | ||||
|  | ||||
|   return await Promise.all( | ||||
|     certs.map(async (cert) => ({ | ||||
|       origin: cert.origin, | ||||
|       cert: await bufferizeContent(cert.cert, cert.certPath), | ||||
|       key: await bufferizeContent(cert.key, cert.keyPath), | ||||
|       pfx: await bufferizeContent(cert.pfx, cert.pfxPath), | ||||
|       passphrase: cert.passphrase, | ||||
|     })) | ||||
|   ) | ||||
| } | ||||
|  | ||||
| // Copied from playwright-core | ||||
| function toAcceptDownloadsProtocol(acceptDownloads?: boolean) { | ||||
|   if (acceptDownloads === undefined) return undefined | ||||
|   if (acceptDownloads) return 'accept' | ||||
|   return 'deny' | ||||
| } | ||||
|  | ||||
| // Copied from playwright-core | ||||
| function prepareRecordHarOptions( | ||||
|   options: BrowserContextOptions['recordHar'] | ||||
| ): channels.RecordHarOptions | undefined { | ||||
|   if (!options) return | ||||
|   return { | ||||
|     path: options.path, | ||||
|     content: options.content || (options.omitContent ? 'omit' : undefined), | ||||
|     urlGlob: isString(options.urlFilter) ? options.urlFilter : undefined, | ||||
|     urlRegexSource: isRegExp(options.urlFilter) | ||||
|       ? options.urlFilter.source | ||||
|       : undefined, | ||||
|     urlRegexFlags: isRegExp(options.urlFilter) | ||||
|       ? options.urlFilter.flags | ||||
|       : undefined, | ||||
|     mode: options.mode, | ||||
|   } | ||||
| } | ||||
|  | ||||
| // Copied from playwright-core | ||||
| async function prepareStorageState( | ||||
|   options: BrowserContextOptions | ||||
| ): Promise<channels.BrowserNewContextParams['storageState']> { | ||||
|   if (typeof options.storageState !== 'string') return options.storageState | ||||
|   try { | ||||
|     return JSON.parse(await fs.promises.readFile(options.storageState, 'utf8')) | ||||
|   } catch (e) { | ||||
|     rewriteErrorMessage( | ||||
|       e, | ||||
|       `Error reading storage state from ${options.storageState}:\n` + e.message | ||||
|     ) | ||||
|     throw e | ||||
|   } | ||||
| } | ||||
|  | ||||
| // Copied from playwright-core | ||||
| async function prepareBrowserContextParams( | ||||
|   options: BrowserContextOptions | ||||
| ): Promise<channels.BrowserNewContextParams> { | ||||
|   if (options.videoSize && !options.videosPath) | ||||
|     throw new Error(`"videoSize" option requires "videosPath" to be specified`) | ||||
|   if (options.extraHTTPHeaders) | ||||
|     network.validateHeaders(options.extraHTTPHeaders) | ||||
|   const contextParams: channels.BrowserNewContextParams = { | ||||
|     ...options, | ||||
|     viewport: options.viewport === null ? undefined : options.viewport, | ||||
|     noDefaultViewport: options.viewport === null, | ||||
|     extraHTTPHeaders: options.extraHTTPHeaders | ||||
|       ? headersObjectToArray(options.extraHTTPHeaders) | ||||
|       : undefined, | ||||
|     storageState: await prepareStorageState(options), | ||||
|     serviceWorkers: options.serviceWorkers, | ||||
|     recordHar: prepareRecordHarOptions(options.recordHar), | ||||
|     colorScheme: | ||||
|       options.colorScheme === null ? 'no-override' : options.colorScheme, | ||||
|     reducedMotion: | ||||
|       options.reducedMotion === null ? 'no-override' : options.reducedMotion, | ||||
|     forcedColors: | ||||
|       options.forcedColors === null ? 'no-override' : options.forcedColors, | ||||
|     acceptDownloads: toAcceptDownloadsProtocol(options.acceptDownloads), | ||||
|     clientCertificates: await toClientCertificatesProtocol( | ||||
|       options.clientCertificates | ||||
|     ), | ||||
|   } | ||||
|   if (!contextParams.recordVideo && options.videosPath) { | ||||
|     contextParams.recordVideo = { | ||||
|       dir: options.videosPath, | ||||
|       size: options.videoSize, | ||||
|     } | ||||
|   } | ||||
|   if (contextParams.recordVideo && contextParams.recordVideo.dir) | ||||
|     contextParams.recordVideo.dir = path.resolve( | ||||
|       process.cwd(), | ||||
|       contextParams.recordVideo.dir | ||||
|     ) | ||||
|   return contextParams | ||||
| } | ||||
|  | ||||
| const toNormalizedCode = (text: string) => { | ||||
|   return text.replace(/\s+/g, '') | ||||
| } | ||||
|  | ||||
| type TestColor = [number, number, number] | ||||
| export const TEST_COLORS = { | ||||
|   WHITE: [249, 249, 249] as TestColor, | ||||
| @ -98,11 +224,16 @@ async function removeCurrentCode(page: Page) { | ||||
| } | ||||
|  | ||||
| export async function sendCustomCmd(page: Page, cmd: EngineCommand) { | ||||
|   await page.getByTestId('custom-cmd-input').fill(JSON.stringify(cmd)) | ||||
|   const json = JSON.stringify(cmd) | ||||
|   await page.getByTestId('custom-cmd-input').fill(json) | ||||
|   await expect(page.getByTestId('custom-cmd-input')).toHaveValue(json) | ||||
|   await page.getByTestId('custom-cmd-send-button').scrollIntoViewIfNeeded() | ||||
|   await page.getByTestId('custom-cmd-send-button').click() | ||||
| } | ||||
|  | ||||
| async function clearCommandLogs(page: Page) { | ||||
|   await page.getByTestId('custom-cmd-input').fill('') | ||||
|   await page.getByTestId('clear-commands').scrollIntoViewIfNeeded() | ||||
|   await page.getByTestId('clear-commands').click() | ||||
| } | ||||
|  | ||||
| @ -150,6 +281,19 @@ export async function closePane(page: Page, testId: string) { | ||||
|  | ||||
| async function openKclCodePanel(page: Page) { | ||||
|   await openPane(page, 'code-pane-button') | ||||
|  | ||||
|   // Code Mirror lazy loads text! Wowza! Let's force-load the text for tests. | ||||
|   await page.evaluate(() => { | ||||
|     // editorManager is available on the window object. | ||||
|     //@ts-ignore this is in an entirely different context that tsc can't see. | ||||
|     editorManager._editorView.dispatch({ | ||||
|       selection: { | ||||
|         //@ts-ignore this is in an entirely different context that tsc can't see. | ||||
|         anchor: editorManager._editorView.docView.length, | ||||
|       }, | ||||
|       scrollIntoView: true, | ||||
|     }) | ||||
|   }) | ||||
| } | ||||
|  | ||||
| async function closeKclCodePanel(page: Page) { | ||||
| @ -165,6 +309,9 @@ async function closeKclCodePanel(page: Page) { | ||||
|  | ||||
| async function openDebugPanel(page: Page) { | ||||
|   await openPane(page, 'debug-pane-button') | ||||
|  | ||||
|   // The debug pane needs time to load everything. | ||||
|   await page.waitForTimeout(3000) | ||||
| } | ||||
|  | ||||
| export async function closeDebugPanel(page: Page) { | ||||
| @ -412,6 +559,10 @@ export async function getUtils(page: Page, test_?: typeof test) { | ||||
|         .boundingBox({ timeout: 5_000 }) | ||||
|         .then((box) => ({ ...box, x: box?.x || 0, y: box?.y || 0 })), | ||||
|     codeLocator: page.locator('.cm-content'), | ||||
|     crushKclCodeIntoOneLineAndThenMaybeSome: async () => { | ||||
|       const code = await page.locator('.cm-content').innerText() | ||||
|       return code.replaceAll(' ', '').replaceAll('\n', '') | ||||
|     }, | ||||
|     normalisedEditorCode: async () => { | ||||
|       const code = await page.locator('.cm-content').innerText() | ||||
|       return normaliseKclNumbers(code) | ||||
| @ -482,13 +633,18 @@ export async function getUtils(page: Page, test_?: typeof test) { | ||||
|       ) | ||||
|     }, | ||||
|  | ||||
|     toNormalizedCode: (text: string) => { | ||||
|       return text.replace(/\s+/g, '') | ||||
|     toNormalizedCode(text: string) { | ||||
|       return toNormalizedCode(text) | ||||
|     }, | ||||
|  | ||||
|     editorTextMatches: async (code: string) => { | ||||
|     async editorTextMatches(code: string) { | ||||
|       const editor = page.locator(editorSelector) | ||||
|       return expect(editor).toHaveText(code, { useInnerText: true }) | ||||
|       return expect | ||||
|         .poll(async () => { | ||||
|           const text = await editor.textContent() | ||||
|           return toNormalizedCode(text ?? '') | ||||
|         }) | ||||
|         .toContain(toNormalizedCode(code)) | ||||
|     }, | ||||
|  | ||||
|     pasteCodeInEditor: async (code: string) => { | ||||
| @ -514,7 +670,7 @@ export async function getUtils(page: Page, test_?: typeof test) { | ||||
|           page.getByRole('button', { name: 'Start Sketch' }) | ||||
|         ).not.toBeDisabled() | ||||
|         await page.getByTestId('create-file-button').click() | ||||
|         await page.getByTestId('file-rename-field').fill(name) | ||||
|         await page.getByTestId('tree-input-field').fill(name) | ||||
|         await page.keyboard.press('Enter') | ||||
|       }) | ||||
|     }, | ||||
| @ -674,6 +830,34 @@ export const makeTemplate: ( | ||||
|   } | ||||
| } | ||||
|  | ||||
| const PLAYWRIGHT_DOWNLOAD_DIR = 'downloads-during-playwright' | ||||
|  | ||||
| export const getPlaywrightDownloadDir = (page: Page) => { | ||||
|   return path.resolve(page.dir, PLAYWRIGHT_DOWNLOAD_DIR) | ||||
| } | ||||
|  | ||||
| const moveDownloadedFileTo = async (page: Page, toLocation: string) => { | ||||
|   await fsp.mkdir(path.dirname(toLocation), { recursive: true }) | ||||
|  | ||||
|   const downloadDir = getPlaywrightDownloadDir(page) | ||||
|  | ||||
|   // Expect there to be at least one file | ||||
|   await expect | ||||
|     .poll(async () => { | ||||
|       const files = await fsp.readdir(downloadDir) | ||||
|       return files.length | ||||
|     }) | ||||
|     .toBeGreaterThan(0) | ||||
|  | ||||
|   // Go through the downloads dir and move files to new location | ||||
|   const files = await fsp.readdir(downloadDir) | ||||
|  | ||||
|   // Assumption: only ever one file here. | ||||
|   for (let file of files) { | ||||
|     await fsp.rename(path.resolve(downloadDir, file), toLocation) | ||||
|   } | ||||
| } | ||||
|  | ||||
| export interface Paths { | ||||
|   modelPath: string | ||||
|   imagePath: string | ||||
| @ -686,7 +870,8 @@ export const doExport = async ( | ||||
|   exportFrom: 'dropdown' | 'sidebarButton' | 'commandBar' = 'dropdown' | ||||
| ): Promise<Paths> => { | ||||
|   if (exportFrom === 'dropdown') { | ||||
|     await page.getByRole('button', { name: APP_NAME }).click() | ||||
|     await page.getByTestId('project-sidebar-toggle').click() | ||||
|  | ||||
|     const exportMenuButton = page.getByRole('button', { | ||||
|       name: 'Export current part', | ||||
|     }) | ||||
| @ -727,25 +912,12 @@ export const doExport = async ( | ||||
|   } | ||||
|   await expect(page.getByText('Confirm Export')).toBeVisible() | ||||
|  | ||||
|   const getPromiseAndResolve = () => { | ||||
|     let resolve: any = () => {} | ||||
|     const promise = new Promise<Download>((r) => { | ||||
|       resolve = r | ||||
|     }) | ||||
|     return [promise, resolve] | ||||
|   } | ||||
|  | ||||
|   const [downloadPromise1, downloadResolve1] = getPromiseAndResolve() | ||||
|   let downloadCnt = 0 | ||||
|  | ||||
|   if (exportFrom === 'dropdown') | ||||
|     page.on('download', async (download) => { | ||||
|       if (downloadCnt === 0) { | ||||
|         downloadResolve1(download) | ||||
|       } | ||||
|       downloadCnt++ | ||||
|     }) | ||||
|   await page.getByRole('button', { name: 'Submit command' }).click() | ||||
|  | ||||
|   // This usually happens immediately after. If we're too slow we don't | ||||
|   // catch it. | ||||
|   await expect(page.getByText('Exported successfully')).toBeVisible() | ||||
|  | ||||
|   if (exportFrom === 'sidebarButton' || exportFrom === 'commandBar') { | ||||
|     return { | ||||
|       modelPath: '', | ||||
| @ -755,15 +927,12 @@ export const doExport = async ( | ||||
|   } | ||||
|  | ||||
|   // Handle download | ||||
|   const download = await downloadPromise1 | ||||
|   const downloadLocationer = (extra = '', isImage = false) => | ||||
|     `./e2e/playwright/export-snapshots/${output.type}-${ | ||||
|       'storage' in output ? output.storage : '' | ||||
|     }${extra}.${isImage ? 'png' : output.type}` | ||||
|   const downloadLocation = downloadLocationer() | ||||
|  | ||||
|   await download.saveAs(downloadLocation) | ||||
|  | ||||
|   if (output.type === 'step') { | ||||
|     // stable timestamps for step files | ||||
|     const fileContents = await fsp.readFile(downloadLocation, 'utf-8') | ||||
| @ -772,6 +941,12 @@ export const doExport = async ( | ||||
|       '1970-01-01T00:00:00.0+00:00' | ||||
|     ) | ||||
|     await fsp.writeFile(downloadLocation, newFileContents) | ||||
|   } else { | ||||
|     // By default all files are downloaded to the same place in playwright | ||||
|     // (declared in src/lib/exportSave) | ||||
|     // To remain consistent with our old web tests, we want to move some downloads | ||||
|     // (images) to another directory. | ||||
|     await moveDownloadedFileTo(page, downloadLocation) | ||||
|   } | ||||
|  | ||||
|   return { | ||||
| @ -798,6 +973,8 @@ export async function tearDown(page: Page, testInfo: TestInfo) { | ||||
|   // It seems it's best to give the browser about 3s to close things | ||||
|   // It's not super reliable but we have no real other choice for now | ||||
|   await page.waitForTimeout(3000) | ||||
|  | ||||
|   await testInfo.tronApp?.close() | ||||
| } | ||||
|  | ||||
| // settingsOverrides may need to be augmented to take more generic items, | ||||
| @ -808,12 +985,24 @@ export async function setup( | ||||
|   testInfo?: TestInfo | ||||
| ) { | ||||
|   await context.addInitScript( | ||||
|     async ({ token, settingsKey, settings, IS_PLAYWRIGHT_KEY }) => { | ||||
|     async ({ | ||||
|       token, | ||||
|       settingsKey, | ||||
|       settings, | ||||
|       IS_PLAYWRIGHT_KEY, | ||||
|       PLAYWRIGHT_TEST_DIR, | ||||
|       PERSIST_MODELING_CONTEXT, | ||||
|     }) => { | ||||
|       localStorage.clear() | ||||
|       localStorage.setItem('TOKEN_PERSIST_KEY', token) | ||||
|       localStorage.setItem('persistCode', ``) | ||||
|       localStorage.setItem( | ||||
|         PERSIST_MODELING_CONTEXT, | ||||
|         JSON.stringify({ openPanes: ['code'] }) | ||||
|       ) | ||||
|       localStorage.setItem(settingsKey, settings) | ||||
|       localStorage.setItem(IS_PLAYWRIGHT_KEY, 'true') | ||||
|       localStorage.setItem('PLAYWRIGHT_TEST_DIR', PLAYWRIGHT_TEST_DIR) | ||||
|     }, | ||||
|     { | ||||
|       token: secrets.token, | ||||
| @ -830,6 +1019,8 @@ export async function setup( | ||||
|         } as Partial<SaveSettingsPayload>, | ||||
|       }), | ||||
|       IS_PLAYWRIGHT_KEY, | ||||
|       PLAYWRIGHT_TEST_DIR: TEST_SETTINGS.app.projectDirectory, | ||||
|       PERSIST_MODELING_CONTEXT, | ||||
|     } | ||||
|   ) | ||||
|  | ||||
| @ -848,12 +1039,15 @@ export async function setup( | ||||
|   await page.emulateMedia({ reducedMotion: 'reduce' }) | ||||
|  | ||||
|   // Trigger a navigation, since loading file:// doesn't. | ||||
|   await page.reload() | ||||
|   // await page.reload() | ||||
| } | ||||
|  | ||||
| let electronApp = undefined | ||||
| let context = undefined | ||||
| let page = undefined | ||||
|  | ||||
| export async function setupElectron({ | ||||
|   testInfo, | ||||
|   folderSetupFn, | ||||
|   cleanProjectDir = true, | ||||
|   appSettings, | ||||
| }: { | ||||
| @ -876,7 +1070,7 @@ export async function setupElectron({ | ||||
|     await fsp.mkdir(projectDirName) | ||||
|   } | ||||
|  | ||||
|   const electronApp = await electron.launch({ | ||||
|   const options = { | ||||
|     args: ['.', '--no-sandbox'], | ||||
|     env: { | ||||
|       ...process.env, | ||||
| @ -886,14 +1080,22 @@ export async function setupElectron({ | ||||
|     ...(process.env.ELECTRON_OVERRIDE_DIST_PATH | ||||
|       ? { executablePath: process.env.ELECTRON_OVERRIDE_DIST_PATH + 'electron' } | ||||
|       : {}), | ||||
|   }) | ||||
|   const context = electronApp.context() | ||||
|   const page = await electronApp.firstWindow() | ||||
|   context.on('console', console.log) | ||||
|   page.on('console', console.log) | ||||
|   } | ||||
|  | ||||
|   // Do this once and then reuse window on subsequent calls. | ||||
|   if (!electronApp) { | ||||
|     electronApp = await electron.launch(options) | ||||
|   } | ||||
|  | ||||
|   if (!context || !page) { | ||||
|     context = electronApp.context() | ||||
|     page = await electronApp.firstWindow() | ||||
|     context.on('console', console.log) | ||||
|     page.on('console', console.log) | ||||
|   } | ||||
|  | ||||
|   if (cleanProjectDir) { | ||||
|     const tempSettingsFilePath = join(projectDirName, SETTINGS_FILE_NAME) | ||||
|     const tempSettingsFilePath = path.join(projectDirName, SETTINGS_FILE_NAME) | ||||
|     const settingsOverrides = TOML.stringify( | ||||
|       appSettings | ||||
|         ? { | ||||
| @ -920,11 +1122,7 @@ export async function setupElectron({ | ||||
|     await fsp.writeFile(tempSettingsFilePath, settingsOverrides) | ||||
|   } | ||||
|  | ||||
|   await folderSetupFn?.(projectDirName) | ||||
|  | ||||
|   await setup(context, page) | ||||
|  | ||||
|   return { electronApp, page, dir: projectDirName } | ||||
|   return { electronApp, page, context, dir: projectDirName, options } | ||||
| } | ||||
|  | ||||
| function failOnConsoleErrors(page: Page, testInfo?: TestInfo) { | ||||
| @ -1010,7 +1208,7 @@ export async function createProject({ | ||||
| } | ||||
|  | ||||
| export function executorInputPath(fileName: string): string { | ||||
|   return join('src', 'wasm-lib', 'tests', 'executor', 'inputs', fileName) | ||||
|   return path.join('src', 'wasm-lib', 'tests', 'executor', 'inputs', fileName) | ||||
| } | ||||
|  | ||||
| export async function doAndWaitForImageDiff( | ||||
| @ -1101,3 +1299,12 @@ export function getPixelRGBs(page: Page) { | ||||
|     }) | ||||
|   } | ||||
| } | ||||
|  | ||||
| export async function pollEditorLinesSelectedLength(page: Page, lines: number) { | ||||
|   return expect | ||||
|     .poll(async () => { | ||||
|       const lines = await page.locator('.cm-activeLine').all() | ||||
|       return lines.length | ||||
|     }) | ||||
|     .toBe(lines) | ||||
| } | ||||
|  | ||||
| @ -1,23 +1,14 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
| import { EngineCommand } from 'lang/std/artifactGraph' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import { getUtils, setup, tearDown } from './test-utils' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
| import { getUtils } from './test-utils' | ||||
|  | ||||
| test.describe('Testing Camera Movement', () => { | ||||
|   test('Can move camera reliably', async ({ page, context }) => { | ||||
|     test.skip(process.platform === 'darwin', 'Can move camera reliably') | ||||
|   test('Can move camera reliably', async ({ page, context, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.openAndClearDebugPanel() | ||||
|     await u.closeKclCodePanel() | ||||
|  | ||||
| @ -183,6 +174,7 @@ test.describe('Testing Camera Movement', () => { | ||||
|  | ||||
|   test('Zoom should be consistent when exiting or entering sketches', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     // start new sketch pan and zoom before exiting, when exiting the sketch should stay in the same place | ||||
|     // than zoom and pan outside of sketch mode and enter again and it should not change from where it is | ||||
| @ -190,9 +182,9 @@ test.describe('Testing Camera Movement', () => { | ||||
|  | ||||
|     test.skip(process.platform !== 'darwin', 'Zoom should be consistent') | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.openDebugPanel() | ||||
|  | ||||
|     await expect( | ||||
| @ -344,7 +336,10 @@ test.describe('Testing Camera Movement', () => { | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   test(`Zoom by scroll should not fire while orbiting`, async ({ page }) => { | ||||
|   test(`Zoom by scroll should not fire while orbiting`, async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     /** | ||||
|      * Currently we only allow zooming by scroll when no other camera movement is happening, | ||||
|      * set within cameraMouseDragGuards in cameraControls.ts, | ||||
| @ -383,7 +378,7 @@ test.describe('Testing Camera Movement', () => { | ||||
|     const expectedOrbitCamZPosition = 64.0 | ||||
|  | ||||
|     await test.step(`Test setup`, async () => { | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await homePage.goToModelingScene() | ||||
|       await u.closeKclCodePanel() | ||||
|       // This test requires the mouse controls to be set to Solidworks | ||||
|       await u.openDebugPanel() | ||||
|  | ||||
| @ -1,35 +1,32 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
|  | ||||
| import { getUtils, setup, tearDown, TEST_COLORS } from './test-utils' | ||||
| import { | ||||
|   getUtils, | ||||
|   TEST_COLORS, | ||||
|   pollEditorLinesSelectedLength, | ||||
| } from './test-utils' | ||||
| import { XOR } from 'lib/utils' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.describe('Testing constraints', () => { | ||||
|   test('Can constrain line length', async ({ page }) => { | ||||
|   test('Can constrain line length', async ({ page, homePage }) => { | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XY') | ||||
|     |> startProfileAt([-10, -10], %) | ||||
|     |> line([20, 0], %) | ||||
|     |> line([0, 20], %) | ||||
|     |> xLine(-20, %) | ||||
|   ` | ||||
|   |> startProfileAt([-10, -10], %) | ||||
|   |> line([20, 0], %) | ||||
|   |> line([0, 20], %) | ||||
|   |> xLine(-20, %) | ||||
|     ` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     const u = await getUtils(page) | ||||
|     const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
| @ -41,7 +38,9 @@ test.describe('Testing constraints', () => { | ||||
|  | ||||
|     // enter sketch again | ||||
|     await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|     await page.waitForTimeout(500) // wait for animation | ||||
|  | ||||
|     // Wait for overlays to populate | ||||
|     await page.waitForTimeout(1000) | ||||
|  | ||||
|     const startXPx = 500 | ||||
|     await page.mouse.move(startXPx + PUR * 15, 250 - PUR * 10) | ||||
| @ -63,42 +62,48 @@ test.describe('Testing constraints', () => { | ||||
|       page.getByRole('button', { name: 'Exit Sketch' }) | ||||
|     ).toBeVisible() | ||||
|  | ||||
|     await page.waitForTimeout(500) // wait for animation | ||||
|     await page.waitForTimeout(2500) // wait for animation | ||||
|  | ||||
|     // Exit sketch | ||||
|     await page.mouse.move(startXPx + PUR * 15, 250 - PUR * 10) | ||||
|     await page.keyboard.press('Escape') | ||||
|     await expect( | ||||
|       page.getByRole('button', { name: 'Exit Sketch' }) | ||||
|     ).not.toBeVisible() | ||||
|     await expect | ||||
|       .poll(async () => { | ||||
|         await page.keyboard.press('Escape', { delay: 500 }) | ||||
|         return page.getByRole('button', { name: 'Exit Sketch' }).isVisible() | ||||
|       }) | ||||
|       .toBe(true) | ||||
|   }) | ||||
|   test(`Remove constraints`, async ({ page }) => { | ||||
|   test(`Remove constraints`, async ({ page, homePage }) => { | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `yo = 79 | ||||
| part001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> line([74.36, 130.4], %, $seg01) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> angledLine([segAng(seg01), yo], %) | ||||
|   |> line([41.19, 58.97 + 5], %) | ||||
| part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 120], %) | ||||
|   |> xLine(-385.34, %, $seg_what) | ||||
|   |> yLine(-170.06, %) | ||||
|   |> xLine(segLen(seg_what), %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|   part001 = startSketchOn('XZ') | ||||
|     |> startProfileAt([-7.54, -26.74], %) | ||||
|     |> line([74.36, 130.4], %, $seg01) | ||||
|     |> line([78.92, -120.11], %) | ||||
|     |> angledLine([segAng(seg01), yo], %) | ||||
|     |> line([41.19, 58.97 + 5], %) | ||||
|   part002 = startSketchOn('XZ') | ||||
|     |> startProfileAt([299.05, 120], %) | ||||
|     |> xLine(-385.34, %, $seg_what) | ||||
|     |> yLine(-170.06, %) | ||||
|     |> xLine(segLen(seg_what), %) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|       ) | ||||
|     }) | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     await page.getByText('line([74.36, 130.4], %, $seg01)').click() | ||||
|     await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|  | ||||
|     // Wait for overlays to populate | ||||
|     await page.waitForTimeout(1000) | ||||
|  | ||||
|     const line3 = await u.getSegmentBodyCoords(`[data-overlay-index="${2}"]`) | ||||
|  | ||||
|     await page.mouse.click(line3.x, line3.y) | ||||
| @ -111,8 +116,8 @@ part002 = startSketchOn('XZ') | ||||
|     await page.getByRole('button', { name: 'remove constraints' }).click() | ||||
|  | ||||
|     await page.getByText('line([39.13, 68.63], %)').click() | ||||
|     await pollEditorLinesSelectedLength(page, 1) | ||||
|     const activeLinesContent = await page.locator('.cm-activeLine').all() | ||||
|     await expect(activeLinesContent).toHaveLength(1) | ||||
|     await expect(activeLinesContent[0]).toHaveText('|> line([39.13, 68.63], %)') | ||||
|  | ||||
|     // checking the count of the overlays is a good proxy check that the client sketch scene is in a good state | ||||
| @ -130,43 +135,75 @@ part002 = startSketchOn('XZ') | ||||
|       }, | ||||
|     ] as const | ||||
|     for (const { testName, offset } of cases) { | ||||
|       test(`${testName}`, async ({ page }) => { | ||||
|       test(`${testName}`, async ({ page, homePage }) => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
| part001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> line([74.36, 130.4], %, $seg01) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> angledLine([segAng(seg01), 78.33], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
| part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
|   |> yLine(-264.06, %) | ||||
|   |> xLine(segLen(seg_what), %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|       part001 = startSketchOn('XZ') | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line([74.36, 130.4], %, $seg01) | ||||
|         |> line([78.92, -120.11], %) | ||||
|         |> angledLine([segAng(seg01), 78.33], %) | ||||
|         |> line([51.19, 48.97], %) | ||||
|       part002 = startSketchOn('XZ') | ||||
|         |> startProfileAt([299.05, 231.45], %) | ||||
|         |> xLine(-425.34, %, $seg_what) | ||||
|         |> yLine(-264.06, %) | ||||
|         |> xLine(segLen(seg_what), %) | ||||
|         |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|           ) | ||||
|  | ||||
|           const isChecked = await createNewVariableCheckbox.isChecked() | ||||
|           const addVariable = testName === 'Add variable' | ||||
|           XOR(isChecked, addVariable) && // XOR because no need to click the checkbox if the state is already correct | ||||
|             (await createNewVariableCheckbox.click()) | ||||
|  | ||||
|           await page | ||||
|             .getByRole('button', { name: 'Add constraining value' }) | ||||
|             .click() | ||||
|  | ||||
|           // Wait for the codemod to take effect | ||||
|           await expect(page.locator('.cm-content')).toContainText(`angle: -57,`) | ||||
|           await expect(page.locator('.cm-content')).toContainText( | ||||
|             `offset: ${offset},` | ||||
|           ) | ||||
|  | ||||
|           await pollEditorLinesSelectedLength(page, 2) | ||||
|           const activeLinesContent = await page.locator('.cm-activeLine').all() | ||||
|           await expect(activeLinesContent[0]).toHaveText( | ||||
|             `|> line([74.36, 130.4], %, $seg01)` | ||||
|           ) | ||||
|           await expect(activeLinesContent[1]).toHaveText(`}, %)`) | ||||
|  | ||||
|           // checking the count of the overlays is a good proxy check that the client sketch scene is in a good state | ||||
|           await expect(page.getByTestId('segment-overlay')).toHaveCount(4) | ||||
|         }) | ||||
|         const u = await getUtils(page) | ||||
|         await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|         await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|         await u.waitForAuthSkipAppStart() | ||||
|         await homePage.goToModelingScene() | ||||
|         await u.waitForPageLoad() | ||||
|  | ||||
|         await page.getByText('line([74.36, 130.4], %, $seg01)').click() | ||||
|         await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|  | ||||
|         // Give time for overlays to populate | ||||
|         await page.waitForTimeout(1000) | ||||
|  | ||||
|         const [line1, line3] = await Promise.all([ | ||||
|           u.getSegmentBodyCoords(`[data-overlay-index="${0}"]`), | ||||
|           u.getSegmentBodyCoords(`[data-overlay-index="${2}"]`), | ||||
|         ]) | ||||
|  | ||||
|         await page.mouse.click(line1.x, line1.y) | ||||
|         await page.keyboard.down('Shift') | ||||
|         await page.mouse.click(line3.x, line3.y) | ||||
|         await page.waitForTimeout(100) // this wait is needed for webkit - not sure why | ||||
|         await page.keyboard.up('Shift') | ||||
|         await page.keyboard.down('Shift') | ||||
|         await page.waitForTimeout(100) | ||||
|         await page.mouse.click(line3.x, line3.y) | ||||
|         await page.waitForTimeout(100) | ||||
|         await page.keyboard.up('Shift') | ||||
|         await page.waitForTimeout(100) | ||||
|         await page | ||||
|           .getByRole('button', { | ||||
|             name: 'Length: open menu', | ||||
| @ -194,6 +231,7 @@ part002 = startSketchOn('XZ') | ||||
|           `offset = ${offset},` | ||||
|         ) | ||||
|  | ||||
|         await pollEditorLinesSelectedLength(page, 2) | ||||
|         const activeLinesContent = await page.locator('.cm-activeLine').all() | ||||
|         await expect(activeLinesContent[0]).toHaveText( | ||||
|           `|> line([74.36, 130.4], %, $seg01)` | ||||
| @ -229,33 +267,37 @@ part002 = startSketchOn('XZ') | ||||
|       }, | ||||
|     ] as const | ||||
|     for (const { testName, value, constraint } of cases) { | ||||
|       test(`${constraint} - ${testName}`, async ({ page }) => { | ||||
|       test(`${constraint} - ${testName}`, async ({ page, homePage }) => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
| part001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
| part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
|   |> yLine(-264.06, %) | ||||
|   |> xLine(segLen(seg_what), %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|       part001 = startSketchOn('XZ') | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line([74.36, 130.4], %) | ||||
|         |> line([78.92, -120.11], %) | ||||
|         |> line([9.16, 77.79], %) | ||||
|         |> line([51.19, 48.97], %) | ||||
|       part002 = startSketchOn('XZ') | ||||
|         |> startProfileAt([299.05, 231.45], %) | ||||
|         |> xLine(-425.34, %, $seg_what) | ||||
|         |> yLine(-264.06, %) | ||||
|         |> xLine(segLen(seg_what), %) | ||||
|         |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|           ) | ||||
|         }) | ||||
|         const u = await getUtils(page) | ||||
|         await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|         await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|         await u.waitForAuthSkipAppStart() | ||||
|         await homePage.goToModelingScene() | ||||
|         await u.waitForPageLoad() | ||||
|  | ||||
|         await page.getByText('line([74.36, 130.4], %)').click() | ||||
|         await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|  | ||||
|         // Wait for overlays to populate | ||||
|         await page.waitForTimeout(1000) | ||||
|  | ||||
|         const [line1, line3] = await Promise.all([ | ||||
|           u.getSegmentBodyCoords(`[data-overlay-index="${0}"]`), | ||||
|           u.getSegmentBodyCoords(`[data-overlay-index="${2}"]`), | ||||
| @ -335,33 +377,37 @@ part002 = startSketchOn('XZ') | ||||
|       }, | ||||
|     ] as const | ||||
|     for (const { testName, addVariable, value, constraint } of cases) { | ||||
|       test(`${constraint} - ${testName}`, async ({ page }) => { | ||||
|       test(`${constraint} - ${testName}`, async ({ page, homePage }) => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
| part001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
| part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
|   |> yLine(-264.06, %) | ||||
|   |> xLine(segLen(seg_what), %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|       part001 = startSketchOn('XZ') | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line([74.36, 130.4], %) | ||||
|         |> line([78.92, -120.11], %) | ||||
|         |> line([9.16, 77.79], %) | ||||
|         |> line([51.19, 48.97], %) | ||||
|       part002 = startSketchOn('XZ') | ||||
|         |> startProfileAt([299.05, 231.45], %) | ||||
|         |> xLine(-425.34, %, $seg_what) | ||||
|         |> yLine(-264.06, %) | ||||
|         |> xLine(segLen(seg_what), %) | ||||
|         |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|           ) | ||||
|         }) | ||||
|         const u = await getUtils(page) | ||||
|         await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|         await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|         await u.waitForAuthSkipAppStart() | ||||
|         await homePage.goToModelingScene() | ||||
|         await u.waitForPageLoad() | ||||
|  | ||||
|         await page.getByText('line([74.36, 130.4], %)').click() | ||||
|         await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|  | ||||
|         // Wait for overlays to populate | ||||
|         await page.waitForTimeout(1000) | ||||
|  | ||||
|         const [line3] = await Promise.all([ | ||||
|           u.getSegmentBodyCoords(`[data-overlay-index="${2}"]`), | ||||
|         ]) | ||||
| @ -372,9 +418,11 @@ part002 = startSketchOn('XZ') | ||||
|           await page.mouse.click(900, 250) | ||||
|         } | ||||
|         await page.keyboard.down('Shift') | ||||
|         await page.waitForTimeout(100) | ||||
|         await page.mouse.click(line3.x, line3.y) | ||||
|         await page.waitForTimeout(100) // this wait is needed for webkit - not sure why | ||||
|         await page.waitForTimeout(100) | ||||
|         await page.keyboard.up('Shift') | ||||
|         await page.waitForTimeout(100) | ||||
|         await page | ||||
|           .getByRole('button', { | ||||
|             name: 'Length: open menu', | ||||
| @ -442,33 +490,37 @@ part002 = startSketchOn('XZ') | ||||
|       }, | ||||
|     ] as const | ||||
|     for (const { testName, addVariable, value, axisSelect } of cases) { | ||||
|       test(`${testName}`, async ({ page }) => { | ||||
|       test(`${testName}`, async ({ page, homePage }) => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
| part001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
| part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
|   |> yLine(-264.06, %) | ||||
|   |> xLine(segLen(seg_what), %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|       part001 = startSketchOn('XZ') | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line([74.36, 130.4], %) | ||||
|         |> line([78.92, -120.11], %) | ||||
|         |> line([9.16, 77.79], %) | ||||
|         |> line([51.19, 48.97], %) | ||||
|       part002 = startSketchOn('XZ') | ||||
|         |> startProfileAt([299.05, 231.45], %) | ||||
|         |> xLine(-425.34, %, $seg_what) | ||||
|         |> yLine(-264.06, %) | ||||
|         |> xLine(segLen(seg_what), %) | ||||
|         |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|           ) | ||||
|         }) | ||||
|         const u = await getUtils(page) | ||||
|         await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|         await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|         await u.waitForAuthSkipAppStart() | ||||
|         await homePage.goToModelingScene() | ||||
|         await u.waitForPageLoad() | ||||
|  | ||||
|         await page.getByText('line([74.36, 130.4], %)').click() | ||||
|         await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|  | ||||
|         // Wait for overlays to populate | ||||
|         await page.waitForTimeout(1000) | ||||
|  | ||||
|         const [line1, line3] = await Promise.all([ | ||||
|           u.getSegmentBodyCoords(`[data-overlay-index="${0}"]`), | ||||
|           u.getSegmentBodyCoords(`[data-overlay-index="${2}"]`), | ||||
| @ -524,7 +576,7 @@ part002 = startSketchOn('XZ') | ||||
|       }) | ||||
|     } | ||||
|   }) | ||||
|   test.describe('Test Angle/Length constraint single selection', () => { | ||||
|   test.describe('Test Angle constraint single selection', () => { | ||||
|     const cases = [ | ||||
|       { | ||||
|         testName: 'Angle - Add variable', | ||||
| @ -538,47 +590,39 @@ part002 = startSketchOn('XZ') | ||||
|         constraint: 'angle', | ||||
|         value: '83, 78.33', | ||||
|       }, | ||||
|       { | ||||
|         testName: 'Length - Add variable', | ||||
|         addVariable: true, | ||||
|         constraint: 'length', | ||||
|         value: '83, length001', | ||||
|       }, | ||||
|       { | ||||
|         testName: 'Length - No variable', | ||||
|         addVariable: false, | ||||
|         constraint: 'length', | ||||
|         value: '83, 78.33', | ||||
|       }, | ||||
|     ] as const | ||||
|     for (const { testName, addVariable, value, constraint } of cases) { | ||||
|       test(`${testName}`, async ({ page }) => { | ||||
|       test(`${testName}`, async ({ page, homePage }) => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
| part001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
| part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
|   |> yLine(-264.06, %) | ||||
|   |> xLine(segLen(seg_what), %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|       part001 = startSketchOn('XZ') | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line([74.36, 130.4], %) | ||||
|         |> line([78.92, -120.11], %) | ||||
|         |> line([9.16, 77.79], %) | ||||
|         |> line([51.19, 48.97], %) | ||||
|       part002 = startSketchOn('XZ') | ||||
|         |> startProfileAt([299.05, 231.45], %) | ||||
|         |> xLine(-425.34, %, $seg_what) | ||||
|         |> yLine(-264.06, %) | ||||
|         |> xLine(segLen(seg_what), %) | ||||
|         |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|           ) | ||||
|         }) | ||||
|         const u = await getUtils(page) | ||||
|         await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|         await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|         await u.waitForAuthSkipAppStart() | ||||
|         await homePage.goToModelingScene() | ||||
|         await u.waitForPageLoad() | ||||
|  | ||||
|         await page.getByText('line([74.36, 130.4], %)').click() | ||||
|         await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|  | ||||
|         // Wait for overlays to populate | ||||
|         await page.waitForTimeout(1000) | ||||
|  | ||||
|         const line3 = await u.getSegmentBodyCoords( | ||||
|           `[data-overlay-index="${2}"]` | ||||
|         ) | ||||
| @ -608,6 +652,90 @@ part002 = startSketchOn('XZ') | ||||
|       }) | ||||
|     } | ||||
|   }) | ||||
|   test.describe('Test Length constraint single selection', () => { | ||||
|     const cases = [ | ||||
|       { | ||||
|         testName: 'Length - Add variable', | ||||
|         addVariable: true, | ||||
|         constraint: 'length', | ||||
|         value: '83, length001', | ||||
|       }, | ||||
|       { | ||||
|         testName: 'Length - No variable', | ||||
|         addVariable: false, | ||||
|         constraint: 'length', | ||||
|         value: '83, 78.33', | ||||
|       }, | ||||
|     ] as const | ||||
|     for (const { testName, addVariable, value, constraint } of cases) { | ||||
|       test(`${testName}`, async ({ context, homePage, page }) => { | ||||
|         // constants and locators | ||||
|         const cmdBarKclInput = page | ||||
|           .getByTestId('cmd-bar-arg-value') | ||||
|           .getByRole('textbox') | ||||
|         const cmdBarKclVariableNameInput = | ||||
|           page.getByPlaceholder('Variable name') | ||||
|         const cmdBarSubmitButton = page.getByRole('button', { | ||||
|           name: 'arrow right Continue', | ||||
|         }) | ||||
|  | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
| part001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
| part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
|   |> yLine(-264.06, %) | ||||
|   |> xLine(segLen(seg_what), %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|           ) | ||||
|         }) | ||||
|         const u = await getUtils(page) | ||||
|         await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|         await homePage.goToModelingScene() | ||||
|  | ||||
|         await page.getByText('line([74.36, 130.4], %)').click() | ||||
|         await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|  | ||||
|         const line3 = await u.getSegmentBodyCoords( | ||||
|           `[data-overlay-index="${2}"]` | ||||
|         ) | ||||
|  | ||||
|         await page.mouse.click(line3.x, line3.y) | ||||
|         await page | ||||
|           .getByRole('button', { | ||||
|             name: 'Length: open menu', | ||||
|           }) | ||||
|           .click() | ||||
|         await page.getByTestId('dropdown-constraint-' + constraint).click() | ||||
|  | ||||
|         if (!addVariable) { | ||||
|           await test.step(`Clear the variable input`, async () => { | ||||
|             await cmdBarKclVariableNameInput.clear() | ||||
|             await cmdBarKclVariableNameInput.press('Backspace') | ||||
|           }) | ||||
|         } | ||||
|         await expect(cmdBarKclInput).toHaveText('78.33') | ||||
|         await cmdBarSubmitButton.click() | ||||
|  | ||||
|         const changedCode = `|> angledLine([${value}], %)` | ||||
|         await expect(page.locator('.cm-content')).toContainText(changedCode) | ||||
|         // checking active assures the cursor is where it should be | ||||
|         await expect(page.locator('.cm-activeLine')).toHaveText(changedCode) | ||||
|  | ||||
|         // checking the count of the overlays is a good proxy check that the client sketch scene is in a good state | ||||
|         await expect(page.getByTestId('segment-overlay')).toHaveCount(4) | ||||
|       }) | ||||
|     } | ||||
|   }) | ||||
|   test.describe('Many segments - no modal constraints', () => { | ||||
|     const cases = [ | ||||
|       { | ||||
| @ -628,33 +756,37 @@ part002 = startSketchOn('XZ') | ||||
|       }, | ||||
|     ] as const | ||||
|     for (const { codeAfter, constraintName } of cases) { | ||||
|       test(`${constraintName}`, async ({ page }) => { | ||||
|       test(`${constraintName}`, async ({ page, homePage }) => { | ||||
|         await page.addInitScript(async (customCode) => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
| part001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
| part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
|   |> yLine(-264.06, %) | ||||
|   |> xLine(segLen(seg_what), %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|       part001 = startSketchOn('XZ') | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line([74.36, 130.4], %) | ||||
|         |> line([78.92, -120.11], %) | ||||
|         |> line([9.16, 77.79], %) | ||||
|         |> line([51.19, 48.97], %) | ||||
|       part002 = startSketchOn('XZ') | ||||
|         |> startProfileAt([299.05, 231.45], %) | ||||
|         |> xLine(-425.34, %, $seg_what) | ||||
|         |> yLine(-264.06, %) | ||||
|         |> xLine(segLen(seg_what), %) | ||||
|         |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|           ) | ||||
|         }) | ||||
|         const u = await getUtils(page) | ||||
|         await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|         await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|         await u.waitForAuthSkipAppStart() | ||||
|         await homePage.goToModelingScene() | ||||
|         await u.waitForPageLoad() | ||||
|  | ||||
|         await page.getByText('line([74.36, 130.4], %)').click() | ||||
|         await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|  | ||||
|         // Wait for overlays to populate | ||||
|         await page.waitForTimeout(1000) | ||||
|  | ||||
|         const line1 = await u.getSegmentBodyCoords( | ||||
|           `[data-overlay-index="${0}"]` | ||||
|         ) | ||||
| @ -673,8 +805,8 @@ part002 = startSketchOn('XZ') | ||||
|         await page.keyboard.up('Shift') | ||||
|  | ||||
|         // check actives lines | ||||
|         await pollEditorLinesSelectedLength(page, codeAfter.length) | ||||
|         const activeLinesContent = await page.locator('.cm-activeLine').all() | ||||
|         await expect(activeLinesContent).toHaveLength(codeAfter.length) | ||||
|  | ||||
|         const constraintMenuButton = page.getByRole('button', { | ||||
|           name: 'Length: open menu', | ||||
| @ -725,32 +857,36 @@ part002 = startSketchOn('XZ') | ||||
|       }, | ||||
|     ] as const | ||||
|     for (const { codeAfter, constraintName } of cases) { | ||||
|       test(`${constraintName}`, async ({ page }) => { | ||||
|       test(`${constraintName}`, async ({ page, homePage }) => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
| part001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
| part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
|   |> yLine(-264.06, %) | ||||
|   |> xLine(segLen(seg_what), %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|       part001 = startSketchOn('XZ') | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line([74.36, 130.4], %) | ||||
|         |> line([78.92, -120.11], %) | ||||
|         |> line([9.16, 77.79], %) | ||||
|       part002 = startSketchOn('XZ') | ||||
|         |> startProfileAt([299.05, 231.45], %) | ||||
|         |> xLine(-425.34, %, $seg_what) | ||||
|         |> yLine(-264.06, %) | ||||
|         |> xLine(segLen(seg_what), %) | ||||
|         |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|           ) | ||||
|         }) | ||||
|         const u = await getUtils(page) | ||||
|         await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|         await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|         await u.waitForAuthSkipAppStart() | ||||
|         await homePage.goToModelingScene() | ||||
|         await u.waitForPageLoad() | ||||
|  | ||||
|         await page.getByText('line([74.36, 130.4], %)').click() | ||||
|         await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|  | ||||
|         // Wait for overlays to populate | ||||
|         await page.waitForTimeout(1000) | ||||
|  | ||||
|         const line1 = await u.getBoundingBox(`[data-overlay-index="${0}"]`) | ||||
|         const line3 = await u.getBoundingBox(`[data-overlay-index="${2}"]`) | ||||
|  | ||||
| @ -777,8 +913,8 @@ part002 = startSketchOn('XZ') | ||||
|         // check there are still 2 cursors (they should stay on the same lines as before constraint was applied) | ||||
|         await expect(page.locator('.cm-cursor')).toHaveCount(2) | ||||
|         // check actives lines | ||||
|         await pollEditorLinesSelectedLength(page, 2) | ||||
|         const activeLinesContent = await page.locator('.cm-activeLine').all() | ||||
|         await expect(activeLinesContent).toHaveLength(2) | ||||
|  | ||||
|         // check both cursors are where they should be after constraint is applied | ||||
|         await expect(activeLinesContent[0]).toHaveText( | ||||
| @ -802,40 +938,47 @@ part002 = startSketchOn('XZ') | ||||
|       }, | ||||
|     ] as const | ||||
|     for (const { codeAfter, constraintName, axisClick } of cases) { | ||||
|       test(`${constraintName}`, async ({ page }) => { | ||||
|       test(`${constraintName}`, async ({ page, homePage }) => { | ||||
|         await page.addInitScript(async () => { | ||||
|           localStorage.setItem( | ||||
|             'persistCode', | ||||
|             `yo = 5 | ||||
| part001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
| part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
|   |> yLine(-264.06, %) | ||||
|   |> xLine(segLen(seg_what), %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|       part001 = startSketchOn('XZ') | ||||
|         |> startProfileAt([-7.54, -26.74], %) | ||||
|         |> line([74.36, 130.4], %) | ||||
|         |> line([78.92, -120.11], %) | ||||
|         |> line([9.16, 77.79], %) | ||||
|       part002 = startSketchOn('XZ') | ||||
|         |> startProfileAt([299.05, 231.45], %) | ||||
|         |> xLine(-425.34, %, $seg_what) | ||||
|         |> yLine(-264.06, %) | ||||
|         |> xLine(segLen(seg_what), %) | ||||
|         |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|           ) | ||||
|         }) | ||||
|         const u = await getUtils(page) | ||||
|         await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|         await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|         await u.waitForAuthSkipAppStart() | ||||
|         await homePage.goToModelingScene() | ||||
|         await u.waitForPageLoad() | ||||
|  | ||||
|         await page.getByText('line([74.36, 130.4], %)').click() | ||||
|         await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|  | ||||
|         // Wait for overlays to populate | ||||
|         await page.waitForTimeout(1000) | ||||
|  | ||||
|         const line3 = await u.getBoundingBox(`[data-overlay-index="${2}"]`) | ||||
|  | ||||
|         // select segment and axis by holding down shift | ||||
|         await page.mouse.click(line3.x - 3, line3.y + 20) | ||||
|         await page.waitForTimeout(100) | ||||
|         await page.keyboard.down('Shift') | ||||
|         await page.waitForTimeout(100) | ||||
|         await page.mouse.click(axisClick.x, axisClick.y) | ||||
|         await page.waitForTimeout(100) | ||||
|         await page.keyboard.up('Shift') | ||||
|         await page.waitForTimeout(100) | ||||
|         const constraintMenuButton = page.getByRole('button', { | ||||
|           name: 'Length: open menu', | ||||
|         }) | ||||
| @ -857,21 +1000,23 @@ part002 = startSketchOn('XZ') | ||||
|  | ||||
|   test('Horizontally constrained line remains selected after applying constraint', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     test.setTimeout(70_000) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XY') | ||||
|   |> startProfileAt([-1.05, -1.07], %) | ||||
|   |> line([3.79, 2.68], %, $seg01) | ||||
|   |> line([3.13, -2.4], %)` | ||||
|     |> startProfileAt([-1.05, -1.07], %) | ||||
|     |> line([3.79, 2.68], %, $seg01) | ||||
|     |> line([3.13, -2.4], %)` | ||||
|       ) | ||||
|     }) | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     await page.getByText('line([3.79, 2.68], %, $seg01)').click() | ||||
|     await expect(page.getByRole('button', { name: 'Edit Sketch' })).toBeEnabled( | ||||
| @ -879,6 +1024,9 @@ part002 = startSketchOn('XZ') | ||||
|     ) | ||||
|     await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|  | ||||
|     // Wait for overlays to populate | ||||
|     await page.waitForTimeout(1000) | ||||
|  | ||||
|     await page.waitForTimeout(100) | ||||
|     const lineBefore = await u.getSegmentBodyCoords( | ||||
|       `[data-overlay-index="1"]`, | ||||
| @ -899,11 +1047,17 @@ part002 = startSketchOn('XZ') | ||||
|         name: 'Length: open menu', | ||||
|       }) | ||||
|       .click() | ||||
|     await page.waitForTimeout(500) | ||||
|     await page.getByRole('button', { name: 'Horizontal', exact: true }).click() | ||||
|     await page.waitForTimeout(500) | ||||
|  | ||||
|     await pollEditorLinesSelectedLength(page, 1) | ||||
|     let activeLinesContent = await page.locator('.cm-activeLine').all() | ||||
|     await expect(activeLinesContent[0]).toHaveText(`|> xLine(3.13, %)`) | ||||
|  | ||||
|     // Wait for code editor to settle. | ||||
|     await page.waitForTimeout(2000) | ||||
|  | ||||
|     // If the overlay-angle is updated the THREE.js scene is in a good state | ||||
|     await expect( | ||||
|       await page.locator('[data-overlay-index="1"]') | ||||
| @ -913,11 +1067,17 @@ part002 = startSketchOn('XZ') | ||||
|       `[data-overlay-index="1"]`, | ||||
|       0 | ||||
|     ) | ||||
|     expect( | ||||
|       await u.getGreatestPixDiff(lineAfter, TEST_COLORS.BLUE) | ||||
|     ).toBeLessThan(3) | ||||
|  | ||||
|     await page.waitForTimeout(300) | ||||
|     const linebb = await u.getBoundingBox('[data-overlay-index="1"]') | ||||
|     await page.mouse.move(linebb.x, linebb.y, { steps: 25 }) | ||||
|     await page.mouse.click(linebb.x, linebb.y) | ||||
|  | ||||
|     await expect | ||||
|       .poll(async () => await u.getGreatestPixDiff(lineAfter, TEST_COLORS.BLUE)) | ||||
|       .toBeLessThan(3) | ||||
|  | ||||
|     await page.waitForTimeout(500) | ||||
|  | ||||
|     await page | ||||
|       .getByRole('button', { | ||||
|         name: 'Length: open menu', | ||||
| @ -931,6 +1091,7 @@ part002 = startSketchOn('XZ') | ||||
|     await page.getByLabel('length Value').fill('10') | ||||
|     await page.getByRole('button', { name: 'Add constraining value' }).click() | ||||
|  | ||||
|     await pollEditorLinesSelectedLength(page, 1) | ||||
|     activeLinesContent = await page.locator('.cm-activeLine').all() | ||||
|     await expect(activeLinesContent[0]).toHaveText(`|> xLine(length001, %)`) | ||||
|  | ||||
|  | ||||
| @ -1,18 +1,9 @@ | ||||
| import { _test, _expect } from './playwright-deprecated' | ||||
| import { test } from './fixtures/fixtureSetup' | ||||
| import { getUtils, setup, tearDown } from './test-utils' | ||||
| import { test, expect } from './zoo-test' | ||||
| import { getUtils } from './test-utils' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import { TEST_CODE_GIZMO } from './storageStates' | ||||
|  | ||||
| _test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| _test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| _test.describe('Testing Gizmo', () => { | ||||
| test.describe('Testing Gizmo', () => { | ||||
|   const cases = [ | ||||
|     { | ||||
|       testDescription: 'top view', | ||||
| @ -57,14 +48,17 @@ _test.describe('Testing Gizmo', () => { | ||||
|     expectedCameraTarget, | ||||
|     testDescription, | ||||
|   } of cases) { | ||||
|     _test(`check ${testDescription}`, async ({ page, browserName }) => { | ||||
|     test(`check ${testDescription}`, async ({ page, homePage }) => { | ||||
|       const u = await getUtils(page) | ||||
|       await page.addInitScript((TEST_CODE_GIZMO) => { | ||||
|         localStorage.setItem('persistCode', TEST_CODE_GIZMO) | ||||
|       }, TEST_CODE_GIZMO) | ||||
|       await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|  | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|       await homePage.goToModelingScene() | ||||
|       await u.waitForPageLoad() | ||||
|  | ||||
|       await page.waitForTimeout(100) | ||||
|       // wait for execution done | ||||
|       await u.openDebugPanel() | ||||
| @ -117,30 +111,30 @@ _test.describe('Testing Gizmo', () => { | ||||
|  | ||||
|       await Promise.all([ | ||||
|         // position | ||||
|         _expect(page.getByTestId('cam-x-position')).toHaveValue( | ||||
|         expect(page.getByTestId('cam-x-position')).toHaveValue( | ||||
|           expectedCameraPosition.x.toString() | ||||
|         ), | ||||
|         _expect(page.getByTestId('cam-y-position')).toHaveValue( | ||||
|         expect(page.getByTestId('cam-y-position')).toHaveValue( | ||||
|           expectedCameraPosition.y.toString() | ||||
|         ), | ||||
|         _expect(page.getByTestId('cam-z-position')).toHaveValue( | ||||
|         expect(page.getByTestId('cam-z-position')).toHaveValue( | ||||
|           expectedCameraPosition.z.toString() | ||||
|         ), | ||||
|         // target | ||||
|         _expect(page.getByTestId('cam-x-target')).toHaveValue( | ||||
|         expect(page.getByTestId('cam-x-target')).toHaveValue( | ||||
|           expectedCameraTarget.x.toString() | ||||
|         ), | ||||
|         _expect(page.getByTestId('cam-y-target')).toHaveValue( | ||||
|         expect(page.getByTestId('cam-y-target')).toHaveValue( | ||||
|           expectedCameraTarget.y.toString() | ||||
|         ), | ||||
|         _expect(page.getByTestId('cam-z-target')).toHaveValue( | ||||
|         expect(page.getByTestId('cam-z-target')).toHaveValue( | ||||
|           expectedCameraTarget.z.toString() | ||||
|         ), | ||||
|       ]) | ||||
|     }) | ||||
|   } | ||||
|  | ||||
|   _test('Context menu and popover menu', async ({ page }) => { | ||||
|   test('Context menu and popover menu', async ({ page, homePage }) => { | ||||
|     const testCase = { | ||||
|       testDescription: 'Right view', | ||||
|       expectedCameraPosition: { x: 5660.02, y: -152, z: 26 }, | ||||
| @ -152,9 +146,9 @@ _test.describe('Testing Gizmo', () => { | ||||
|     await page.addInitScript((TEST_CODE_GIZMO) => { | ||||
|       localStorage.setItem('persistCode', TEST_CODE_GIZMO) | ||||
|     }, TEST_CODE_GIZMO) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await page.waitForTimeout(100) | ||||
|     // wait for execution done | ||||
|     await u.openDebugPanel() | ||||
| @ -196,7 +190,7 @@ _test.describe('Testing Gizmo', () => { | ||||
|     const buttonToTest = page.getByRole('button', { | ||||
|       name: testCase.testDescription, | ||||
|     }) | ||||
|     await _expect(buttonToTest).toBeVisible() | ||||
|     await expect(buttonToTest).toBeVisible() | ||||
|     await buttonToTest.click() | ||||
|  | ||||
|     // Now assert we've moved to the correct view | ||||
| @ -215,23 +209,23 @@ _test.describe('Testing Gizmo', () => { | ||||
|  | ||||
|     await Promise.all([ | ||||
|       // position | ||||
|       _expect(page.getByTestId('cam-x-position')).toHaveValue( | ||||
|       expect(page.getByTestId('cam-x-position')).toHaveValue( | ||||
|         testCase.expectedCameraPosition.x.toString() | ||||
|       ), | ||||
|       _expect(page.getByTestId('cam-y-position')).toHaveValue( | ||||
|       expect(page.getByTestId('cam-y-position')).toHaveValue( | ||||
|         testCase.expectedCameraPosition.y.toString() | ||||
|       ), | ||||
|       _expect(page.getByTestId('cam-z-position')).toHaveValue( | ||||
|       expect(page.getByTestId('cam-z-position')).toHaveValue( | ||||
|         testCase.expectedCameraPosition.z.toString() | ||||
|       ), | ||||
|       // target | ||||
|       _expect(page.getByTestId('cam-x-target')).toHaveValue( | ||||
|       expect(page.getByTestId('cam-x-target')).toHaveValue( | ||||
|         testCase.expectedCameraTarget.x.toString() | ||||
|       ), | ||||
|       _expect(page.getByTestId('cam-y-target')).toHaveValue( | ||||
|       expect(page.getByTestId('cam-y-target')).toHaveValue( | ||||
|         testCase.expectedCameraTarget.y.toString() | ||||
|       ), | ||||
|       _expect(page.getByTestId('cam-z-target')).toHaveValue( | ||||
|       expect(page.getByTestId('cam-z-target')).toHaveValue( | ||||
|         testCase.expectedCameraTarget.z.toString() | ||||
|       ), | ||||
|     ]) | ||||
| @ -242,32 +236,59 @@ _test.describe('Testing Gizmo', () => { | ||||
|     const gizmoPopoverButton = page.getByRole('button', { | ||||
|       name: 'view settings', | ||||
|     }) | ||||
|     await _expect(gizmoPopoverButton).toBeVisible() | ||||
|     await expect(gizmoPopoverButton).toBeVisible() | ||||
|     await gizmoPopoverButton.click() | ||||
|     await _expect(buttonToTest).toBeVisible() | ||||
|     await expect(buttonToTest).toBeVisible() | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| test.describe(`Testing gizmo, fixture-based`, () => { | ||||
|   test('Center on selection from menu', async ({ | ||||
|     app, | ||||
|     context, | ||||
|     page, | ||||
|     homePage, | ||||
|     cmdBar, | ||||
|     editor, | ||||
|     toolbar, | ||||
|     scene, | ||||
|   }) => { | ||||
|     test.skip( | ||||
|       process.platform === 'win32', | ||||
|       'Fails on windows in CI, can not be replicated locally on windows.' | ||||
|     ) | ||||
|     await context.addInitScript(() => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         ` | ||||
|         const sketch002 = startSketchOn('XZ') | ||||
|           |> startProfileAt([-108.83, -57.48], %) | ||||
|           |> angledLine([0, 105.13], %, $rectangleSegmentA001) | ||||
|           |> angledLine([ | ||||
|                segAng(rectangleSegmentA001) - 90, | ||||
|                77.9 | ||||
|              ], %) | ||||
|           |> angledLine([ | ||||
|                segAng(rectangleSegmentA001), | ||||
|                -segLen(rectangleSegmentA001) | ||||
|              ], %) | ||||
|           |> close(%) | ||||
|         const sketch001 = startSketchOn('XZ') | ||||
|           |> circle({ | ||||
|                center: [818.33, 168.1], | ||||
|                radius: 182.8 | ||||
|              }, %) | ||||
|           |> extrude(50, %) | ||||
|       ` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await homePage.goToModelingScene() | ||||
|     const u = await getUtils(page) | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     await test.step(`Setup`, async () => { | ||||
|       const file = await app.getInputFile('test-circle-extrude.kcl') | ||||
|       await app.initialise(file) | ||||
|       await scene.expectState({ | ||||
|         camera: { | ||||
|           position: [4982.21, -23865.37, 13810.64], | ||||
|           target: [4982.21, 0, 2737.1], | ||||
|           position: [11912.6, -39586.98, 21391.21], | ||||
|           target: [11912.6, -635, 3317.49], | ||||
|         }, | ||||
|       }) | ||||
|     }) | ||||
| @ -275,7 +296,7 @@ test.describe(`Testing gizmo, fixture-based`, () => { | ||||
|  | ||||
|     await test.step(`Select an edge of this circle`, async () => { | ||||
|       const circleSnippet = | ||||
|         'circle({ center = [318.33, 168.1], radius = 182.8 }, %)' | ||||
|         'circle({ center: [818.33, 168.1], radius: 182.8 }, %)' | ||||
|       await moveToCircle() | ||||
|       await clickCircle() | ||||
|       await editor.expectState({ | ||||
| @ -292,8 +313,8 @@ test.describe(`Testing gizmo, fixture-based`, () => { | ||||
|     await test.step(`Verify the camera moved`, async () => { | ||||
|       await scene.expectState({ | ||||
|         camera: { | ||||
|           position: [0, -23865.37, 11073.53], | ||||
|           target: [0, 0, 0], | ||||
|           position: [20785.58, -40221.98, 22343.46], | ||||
|           target: [20785.58, -1270, 4269.74], | ||||
|         }, | ||||
|       }) | ||||
|     }) | ||||
|  | ||||
| @ -1,18 +1,8 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { getUtils, setup, tearDown } from './test-utils' | ||||
| import { TEST_SETTINGS, TEST_SETTINGS_KEY } from './storageStates' | ||||
| import * as TOML from '@iarna/toml' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
| import { test, expect } from './zoo-test' | ||||
| import { getUtils } from './test-utils' | ||||
|  | ||||
| test.describe('Test toggling perspective', () => { | ||||
|   test('via command palette and toggle', async ({ page }) => { | ||||
|   test.fixme('via command palette and toggle', async ({ page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     // Locators and constants | ||||
| @ -20,7 +10,7 @@ test.describe('Test toggling perspective', () => { | ||||
|     const screenHeight = 500 | ||||
|     const checkedScreenLocation = { | ||||
|       x: screenWidth * 0.71, | ||||
|       y: screenHeight * 0.4, | ||||
|       y: screenHeight * 0.2, | ||||
|     } | ||||
|     const backgroundColor: [number, number, number] = [29, 29, 29] | ||||
|     const xzPlaneColor: [number, number, number] = [82, 55, 96] | ||||
| @ -40,8 +30,8 @@ test.describe('Test toggling perspective', () => { | ||||
|     }) | ||||
|  | ||||
|     await test.step('Setup', async () => { | ||||
|       await page.setViewportSize({ width: screenWidth, height: screenHeight }) | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await page.setBodyDimensions({ width: screenWidth, height: screenHeight }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await u.closeKclCodePanel() | ||||
|       await expect | ||||
|         .poll(async () => locationToHaveColor(backgroundColor), { | ||||
| @ -52,11 +42,17 @@ test.describe('Test toggling perspective', () => { | ||||
|       await expect(projectionToggle).toHaveAttribute('aria-checked', 'true') | ||||
|     }) | ||||
|  | ||||
|     // Extremely wild note: flicking between ortho and persp actually changes | ||||
|     // the orientation of the axis/camera. How can you see this? Well toggle it, | ||||
|     // then refresh. You'll see it doesn't match what we left. | ||||
|     await test.step('Switch to ortho via command palette', async () => { | ||||
|       await commandPaletteButton.click() | ||||
|       await page.waitForTimeout(1000) | ||||
|       await commandOption.click() | ||||
|       await page.waitForTimeout(1000) | ||||
|       await orthoOption.click() | ||||
|       await expect(commandToast).toBeVisible() | ||||
|       await expect(commandToast).not.toBeVisible() | ||||
|       await expect | ||||
|         .poll(async () => locationToHaveColor(xzPlaneColor), { | ||||
|           timeout: 5000, | ||||
| @ -67,27 +63,9 @@ test.describe('Test toggling perspective', () => { | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Refresh the page and ensure the stream is loaded in ortho`, async () => { | ||||
|       // In playwright web, the settings set while testing are not persisted because | ||||
|       // the `addInitScript` within `setup` is re-run on page reload | ||||
|       await page.addInitScript( | ||||
|         ({ settingsKey, settings }) => { | ||||
|           localStorage.setItem(settingsKey, settings) | ||||
|         }, | ||||
|         { | ||||
|           settingsKey: TEST_SETTINGS_KEY, | ||||
|           settings: TOML.stringify({ | ||||
|             settings: { | ||||
|               ...TEST_SETTINGS, | ||||
|               modeling: { | ||||
|                 ...TEST_SETTINGS.modeling, | ||||
|                 cameraProjection: 'orthographic', | ||||
|               }, | ||||
|             }, | ||||
|           }), | ||||
|         } | ||||
|       ) | ||||
|       await page.reload() | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await page.waitForTimeout(1000) | ||||
|       await u.closeKclCodePanel() | ||||
|       await expect | ||||
|         .poll(async () => locationToHaveColor(xzPlaneColor), { | ||||
|           timeout: 5000, | ||||
|  | ||||
| @ -1,35 +1,30 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { getUtils, setup, setupElectron, tearDown } from './test-utils' | ||||
| import { test, expect } from './zoo-test' | ||||
| import { getUtils } from './test-utils' | ||||
| import { bracket } from 'lib/exampleKcl' | ||||
| import * as fsp from 'fs/promises' | ||||
| import { join } from 'path' | ||||
| import { FILE_EXT } from 'lib/constants' | ||||
| import { UnitLength_type } from '@kittycad/lib/dist/types/src/models' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.describe('Testing in-app sample loading', () => { | ||||
|   /** | ||||
|    * Note this test implicitly depends on the KCL sample "car-wheel.kcl", | ||||
|    * its title, and its units settings. https://github.com/KittyCAD/kcl-samples/blob/main/car-wheel/car-wheel.kcl | ||||
|    */ | ||||
|   test('Web: should overwrite current code, cannot create new file', async ({ | ||||
|     editor, | ||||
|     context, | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     await test.step(`Test setup`, async () => { | ||||
|       await page.addInitScript((code) => { | ||||
|       await context.addInitScript((code) => { | ||||
|         window.localStorage.setItem('persistCode', code) | ||||
|       }, bracket) | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|     }) | ||||
|  | ||||
|     // Locators and constants | ||||
| @ -54,13 +49,13 @@ test.describe('Testing in-app sample loading', () => { | ||||
|       }) | ||||
|     const warningText = page.getByText('Overwrite current file and units?') | ||||
|     const confirmButton = page.getByRole('button', { name: 'Submit command' }) | ||||
|     const codeLocator = page.locator('.cm-content') | ||||
|     const unitsToast = (unit: UnitLength_type) => | ||||
|       page.getByText(`Set default unit to "${unit}" for this project`) | ||||
|  | ||||
|     await test.step(`Precondition: check the initial code`, async () => { | ||||
|       await u.openKclCodePanel() | ||||
|       await expect(codeLocator).toContainText(bracket.split('\n')[0]) | ||||
|       await editor.scrollToText(bracket.split('\n')[0]) | ||||
|       await editor.expectEditor.toContain(bracket.split('\n')[0]) | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Load a KCL sample with the command palette`, async () => { | ||||
| @ -73,7 +68,7 @@ test.describe('Testing in-app sample loading', () => { | ||||
|       await expect(warningText).toBeVisible() | ||||
|       await confirmButton.click() | ||||
|  | ||||
|       await expect(codeLocator).toContainText('// ' + newSample.title) | ||||
|       await editor.expectEditor.toContain('// ' + newSample.title) | ||||
|       await expect(unitsToast('in')).toBeVisible() | ||||
|     }) | ||||
|   }) | ||||
| @ -86,16 +81,13 @@ test.describe('Testing in-app sample loading', () => { | ||||
|   test( | ||||
|     'Desktop: should create new file by default, optionally overwrite', | ||||
|     { tag: '@electron' }, | ||||
|     async ({ browserName: _ }, testInfo) => { | ||||
|       const { electronApp, page, dir } = await setupElectron({ | ||||
|         testInfo, | ||||
|         folderSetupFn: async (dir) => { | ||||
|           const bracketDir = join(dir, 'bracket') | ||||
|           await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|           await fsp.writeFile(join(bracketDir, 'main.kcl'), bracket, { | ||||
|             encoding: 'utf-8', | ||||
|           }) | ||||
|         }, | ||||
|     async ({ editor, context, page }, testInfo) => { | ||||
|       const { dir } = await context.folderSetupFn(async (dir) => { | ||||
|         const bracketDir = join(dir, 'bracket') | ||||
|         await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|         await fsp.writeFile(join(bracketDir, 'main.kcl'), bracket, { | ||||
|           encoding: 'utf-8', | ||||
|         }) | ||||
|       }) | ||||
|       const u = await getUtils(page) | ||||
|  | ||||
| @ -134,19 +126,19 @@ test.describe('Testing in-app sample loading', () => { | ||||
|         page.getByRole('listitem').filter({ | ||||
|           has: page.getByRole('button', { name }), | ||||
|         }) | ||||
|       const codeLocator = page.locator('.cm-content') | ||||
|       const unitsToast = (unit: UnitLength_type) => | ||||
|         page.getByText(`Set default unit to "${unit}" for this project`) | ||||
|  | ||||
|       await test.step(`Test setup`, async () => { | ||||
|         await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|         await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|         await projectCard.click() | ||||
|         await u.waitForPageLoad() | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Precondition: check the initial code`, async () => { | ||||
|         await u.openKclCodePanel() | ||||
|         await expect(codeLocator).toContainText(bracket.split('\n')[0]) | ||||
|         await editor.scrollToText(bracket.split('\n')[0]) | ||||
|         await editor.expectEditor.toContain(bracket.split('\n')[0]) | ||||
|         await u.openFilePanel() | ||||
|  | ||||
|         await expect(projectMenuButton).toContainText('main.kcl') | ||||
| @ -163,7 +155,7 @@ test.describe('Testing in-app sample loading', () => { | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Ensure we made and opened a new file`, async () => { | ||||
|         await expect(codeLocator).toContainText('// ' + sampleOne.title) | ||||
|         await editor.expectEditor.toContain('// ' + sampleOne.title) | ||||
|         await expect(newlyCreatedFile(sampleOne.file)).toBeVisible() | ||||
|         await expect(projectMenuButton).toContainText(sampleOne.file) | ||||
|         await expect(unitsToast('in')).toBeVisible() | ||||
| @ -182,7 +174,7 @@ test.describe('Testing in-app sample loading', () => { | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Ensure we overwrote the current file without navigating`, async () => { | ||||
|         await expect(codeLocator).toContainText('// ' + sampleTwo.title) | ||||
|         await editor.expectEditor.toContain('// ' + sampleTwo.title) | ||||
|         await test.step(`Check actual file contents`, async () => { | ||||
|           await expect | ||||
|             .poll(async () => { | ||||
| @ -198,8 +190,6 @@ test.describe('Testing in-app sample loading', () => { | ||||
|         await expect(projectMenuButton).toContainText(sampleOne.file) | ||||
|         await expect(unitsToast('mm')).toBeVisible() | ||||
|       }) | ||||
|  | ||||
|       await electronApp.close() | ||||
|     } | ||||
|   ) | ||||
| }) | ||||
|  | ||||
| @ -1,24 +1,16 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
|  | ||||
| import { commonPoints, getUtils, setup, tearDown } from './test-utils' | ||||
| import { commonPoints, getUtils } from './test-utils' | ||||
| import { Coords2d } from 'lang/std/sketch' | ||||
| import { KCL_DEFAULT_LENGTH } from 'lib/constants' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.describe('Testing selections', () => { | ||||
|   test.setTimeout(90_000) | ||||
|   test( | ||||
|     'Selections work on fresh and edited sketch', | ||||
|     { tag: ['@skipWin'] }, | ||||
|     async ({ page }) => { | ||||
|     async ({ page, homePage }) => { | ||||
|       // Skip on windows its being weird. | ||||
|       test.skip(process.platform === 'win32', 'Skip on windows') | ||||
|  | ||||
| @ -27,9 +19,9 @@ test.describe('Testing selections', () => { | ||||
|       // source ranges are wrong, hovers won't work | ||||
|       const u = await getUtils(page) | ||||
|       const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await homePage.goToModelingScene() | ||||
|       await u.openDebugPanel() | ||||
|  | ||||
|       const yAxisClick = () => | ||||
| @ -79,31 +71,31 @@ test.describe('Testing selections', () => { | ||||
|       await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt(${commonPoints.startAt}, %)`) | ||||
|       |> startProfileAt(${commonPoints.startAt}, %)`) | ||||
|  | ||||
|       await page.waitForTimeout(100) | ||||
|       await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) | ||||
|  | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt(${commonPoints.startAt}, %) | ||||
|     |> xLine(${commonPoints.num1}, %)`) | ||||
|       |> startProfileAt(${commonPoints.startAt}, %) | ||||
|       |> xLine(${commonPoints.num1}, %)`) | ||||
|  | ||||
|       await page.waitForTimeout(100) | ||||
|       await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 20) | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt(${commonPoints.startAt}, %) | ||||
|     |> xLine(${commonPoints.num1}, %) | ||||
|     |> yLine(${commonPoints.num1 + 0.01}, %)`) | ||||
|       |> startProfileAt(${commonPoints.startAt}, %) | ||||
|       |> xLine(${commonPoints.num1}, %) | ||||
|       |> yLine(${commonPoints.num1 + 0.01}, %)`) | ||||
|       await page.waitForTimeout(100) | ||||
|       await page.mouse.click(startXPx, 500 - PUR * 20) | ||||
|       await expect(page.locator('.cm-content')) | ||||
|         .toHaveText(`sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt(${commonPoints.startAt}, %) | ||||
|     |> xLine(${commonPoints.num1}, %) | ||||
|     |> yLine(${commonPoints.num1 + 0.01}, %) | ||||
|     |> xLine(${commonPoints.num2 * -1}, %)`) | ||||
|       |> startProfileAt(${commonPoints.startAt}, %) | ||||
|       |> xLine(${commonPoints.num1}, %) | ||||
|       |> yLine(${commonPoints.num1 + 0.01}, %) | ||||
|       |> xLine(${commonPoints.num2 * -1}, %)`) | ||||
|  | ||||
|       // deselect line tool | ||||
|       await page.getByRole('button', { name: 'line Line', exact: true }).click() | ||||
| @ -264,78 +256,78 @@ test.describe('Testing selections', () => { | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test('Solids should be select and deletable', async ({ page }) => { | ||||
|   test('Solids should be select and deletable', async ({ page, homePage }) => { | ||||
|     test.setTimeout(90_000) | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XZ') | ||||
|         |> startProfileAt([-79.26, 95.04], %) | ||||
|         |> line([112.54, 127.64], %, $seg02) | ||||
|         |> line([170.36, -121.61], %, $seg01) | ||||
|         |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|         |> close(%) | ||||
| extrude001 = extrude(50, sketch001) | ||||
| sketch005 = startSketchOn(extrude001, 'END') | ||||
|   |> startProfileAt([23.24, 136.52], %) | ||||
|   |> line([-8.44, 36.61], %) | ||||
|   |> line([49.4, 2.05], %) | ||||
|   |> line([29.69, -46.95], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| sketch003 = startSketchOn(extrude001, seg01) | ||||
|   |> startProfileAt([21.23, 17.81], %) | ||||
|   |> line([51.97, 21.32], %) | ||||
|   |> line([4.07, -22.75], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| sketch002 = startSketchOn(extrude001, seg02) | ||||
|   |> startProfileAt([-100.54, 16.99], %) | ||||
|   |> line([0, 20.03], %) | ||||
|   |> line([62.61, 0], %, $seg03) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| extrude002 = extrude(50, sketch002) | ||||
| sketch004 = startSketchOn(extrude002, seg03) | ||||
|   |> startProfileAt([57.07, 134.77], %) | ||||
|   |> line([-4.72, 22.84], %) | ||||
|   |> line([28.8, 6.71], %) | ||||
|   |> line([9.19, -25.33], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| extrude003 = extrude(20, sketch004) | ||||
| pipeLength = 40 | ||||
| pipeSmallDia = 10 | ||||
| pipeLargeDia = 20 | ||||
| thickness = 0.5 | ||||
| part009 = startSketchOn('XY') | ||||
|   |> startProfileAt([pipeLargeDia - (thickness / 2), 38], %) | ||||
|   |> line([thickness, 0], %) | ||||
|   |> line([0, -1], %) | ||||
|   |> angledLineToX({ | ||||
|        angle = 60, | ||||
|        to = pipeSmallDia + thickness | ||||
|      }, %) | ||||
|   |> line([0, -pipeLength], %) | ||||
|   |> angledLineToX({ | ||||
|        angle = -60, | ||||
|        to = pipeLargeDia + thickness | ||||
|      }, %) | ||||
|   |> line([0, -1], %) | ||||
|   |> line([-thickness, 0], %) | ||||
|   |> line([0, 1], %) | ||||
|   |> angledLineToX({ angle = 120, to = pipeSmallDia }, %) | ||||
|   |> line([0, pipeLength], %) | ||||
|   |> angledLineToX({ angle = 60, to = pipeLargeDia }, %) | ||||
|   |> close(%) | ||||
| rev = revolve({ axis = 'y' }, part009) | ||||
| ` | ||||
|       |> startProfileAt([-79.26, 95.04], %) | ||||
|       |> line([112.54, 127.64], %, $seg02) | ||||
|       |> line([170.36, -121.61], %, $seg01) | ||||
|       |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|       |> close(%) | ||||
|   extrude001 = extrude(50, sketch001) | ||||
|   sketch005 = startSketchOn(extrude001, 'END') | ||||
|     |> startProfileAt([23.24, 136.52], %) | ||||
|     |> line([-8.44, 36.61], %) | ||||
|     |> line([49.4, 2.05], %) | ||||
|     |> line([29.69, -46.95], %) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|   sketch003 = startSketchOn(extrude001, seg01) | ||||
|     |> startProfileAt([21.23, 17.81], %) | ||||
|     |> line([51.97, 21.32], %) | ||||
|     |> line([4.07, -22.75], %) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|   sketch002 = startSketchOn(extrude001, seg02) | ||||
|     |> startProfileAt([-100.54, 16.99], %) | ||||
|     |> line([0, 20.03], %) | ||||
|     |> line([62.61, 0], %, $seg03) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|   extrude002 = extrude(50, sketch002) | ||||
|   sketch004 = startSketchOn(extrude002, seg03) | ||||
|     |> startProfileAt([57.07, 134.77], %) | ||||
|     |> line([-4.72, 22.84], %) | ||||
|     |> line([28.8, 6.71], %) | ||||
|     |> line([9.19, -25.33], %) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|   extrude003 = extrude(20, sketch004) | ||||
|   pipeLength = 40 | ||||
|   pipeSmallDia = 10 | ||||
|   pipeLargeDia = 20 | ||||
|   thickness = 0.5 | ||||
|   part009 = startSketchOn('XY') | ||||
|     |> startProfileAt([pipeLargeDia - (thickness / 2), 38], %) | ||||
|     |> line([thickness, 0], %) | ||||
|     |> line([0, -1], %) | ||||
|     |> angledLineToX({ | ||||
|      angle = 60, | ||||
|      to = pipeSmallDia + thickness | ||||
|    }, %) | ||||
|     |> line([0, -pipeLength], %) | ||||
|     |> angledLineToX({ | ||||
|      angle = -60, | ||||
|      to = pipeLargeDia + thickness | ||||
|    }, %) | ||||
|     |> line([0, -1], %) | ||||
|     |> line([-thickness, 0], %) | ||||
|     |> line([0, 1], %) | ||||
|     |> angledLineToX({ angle = 120, to = pipeSmallDia }, %) | ||||
|     |> line([0, pipeLength], %) | ||||
|     |> angledLineToX({ angle = 60, to = pipeLargeDia }, %) | ||||
|     |> close(%) | ||||
|   rev = revolve({ axis: 'y' }, part009) | ||||
|   ` | ||||
|       ) | ||||
|     }, KCL_DEFAULT_LENGTH) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.goto('/') | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
| @ -395,29 +387,29 @@ rev = revolve({ axis = 'y' }, part009) | ||||
|       `extrude001 = extrude(50, sketch001)` | ||||
|     ) | ||||
|     await expect(u.codeLocator).toContainText(`sketch005 = startSketchOn({ | ||||
|        plane = { | ||||
|          origin = { x = 0, y = -50, z = 0 }, | ||||
|          x_axis = { x = 1, y = 0, z = 0 }, | ||||
|          y_axis = { x = 0, y = 0, z = 1 }, | ||||
|          z_axis = { x = 0, y = -1, z = 0 } | ||||
|        } | ||||
|      })`) | ||||
|      plane = { | ||||
|        origin = { x = 0, y = -50, z = 0 }, | ||||
|        x_axis = { x = 1, y = 0, z = 0 }, | ||||
|        y_axis = { x = 0, y = 0, z = 1 }, | ||||
|        z_axis = { x = 0, y = -1, z = 0 } | ||||
|      } | ||||
|    })`) | ||||
|     await expect(u.codeLocator).toContainText(`sketch003 = startSketchOn({ | ||||
|        plane = { | ||||
|          origin = { x = 116.53, y = 0, z = 163.25 }, | ||||
|          x_axis = { x = -0.81, y = 0, z = 0.58 }, | ||||
|          y_axis = { x = 0, y = -1, z = 0 }, | ||||
|          z_axis = { x = 0.58, y = 0, z = 0.81 } | ||||
|        } | ||||
|      })`) | ||||
|      plane = { | ||||
|        origin = { x = 116.53, y = 0, z = 163.25 }, | ||||
|        x_axis = { x = -0.81, y = 0, z = 0.58 }, | ||||
|        y_axis = { x = 0, y = -1, z = 0 }, | ||||
|        z_axis = { x = 0.58, y = 0, z = 0.81 } | ||||
|      } | ||||
|    })`) | ||||
|     await expect(u.codeLocator).toContainText(`sketch002 = startSketchOn({ | ||||
|        plane = { | ||||
|          origin = { x = -91.74, y = 0, z = 80.89 }, | ||||
|          x_axis = { x = -0.66, y = 0, z = -0.75 }, | ||||
|          y_axis = { x = 0, y = -1, z = 0 }, | ||||
|          z_axis = { x = -0.75, y = 0, z = 0.66 } | ||||
|        } | ||||
|      })`) | ||||
|      plane = { | ||||
|        origin = { x = -91.74, y = 0, z = 80.89 }, | ||||
|        x_axis = { x = -0.66, y = 0, z = -0.75 }, | ||||
|        y_axis = { x = 0, y = -1, z = 0 }, | ||||
|        z_axis = { x = -0.75, y = 0, z = 0.66 } | ||||
|      } | ||||
|    })`) | ||||
|  | ||||
|     // DELETE SOLID 2D | ||||
|     await page.mouse.click(solid2d.x, solid2d.y) | ||||
| @ -433,31 +425,32 @@ rev = revolve({ axis = 'y' }, part009) | ||||
|   }) | ||||
|   test("Deleting solid that the AST mod can't handle results in a toast message", async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-79.26, 95.04], %) | ||||
|   |> line([112.54, 127.64], %, $seg02) | ||||
|   |> line([170.36, -121.61], %, $seg01) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| extrude001 = extrude(50, sketch001) | ||||
| launderExtrudeThroughVar = extrude001 | ||||
| sketch002 = startSketchOn(launderExtrudeThroughVar, seg02) | ||||
|   |> startProfileAt([-100.54, 16.99], %) | ||||
|   |> line([0, 20.03], %) | ||||
|   |> line([62.61, 0], %, $seg03) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| ` | ||||
|     |> startProfileAt([-79.26, 95.04], %) | ||||
|     |> line([112.54, 127.64], %, $seg02) | ||||
|     |> line([170.36, -121.61], %, $seg01) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|   extrude001 = extrude(50, sketch001) | ||||
|   launderExtrudeThroughVar = extrude001 | ||||
|   sketch002 = startSketchOn(launderExtrudeThroughVar, seg02) | ||||
|     |> startProfileAt([-100.54, 16.99], %) | ||||
|     |> line([0, 20.03], %) | ||||
|     |> line([62.61, 0], %, $seg03) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|   ` | ||||
|       ) | ||||
|     }, KCL_DEFAULT_LENGTH) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.goto('/') | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]', 10_000) | ||||
| @ -497,37 +490,38 @@ sketch002 = startSketchOn(launderExtrudeThroughVar, seg02) | ||||
|   }) | ||||
|   test('Hovering over 3d features highlights code, clicking puts the cursor in the right place and sends selection id to engine', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async (KCL_DEFAULT_LENGTH) => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `part001 = startSketchOn('XZ') | ||||
|     |> startProfileAt([20, 0], %) | ||||
|     |> line([7.13, 4 + 0], %) | ||||
|     |> angledLine({ angle = 3 + 0, length = 3.14 + 0 }, %) | ||||
|     |> lineTo([20.14 + 0, -0.14 + 0], %) | ||||
|     |> xLineTo(29 + 0, %) | ||||
|     |> yLine(-3.14 + 0, %, $a) | ||||
|     |> xLine(1.63, %) | ||||
|     |> angledLineOfXLength({ angle = 3 + 0, length = 3.14 }, %) | ||||
|     |> angledLineOfYLength({ angle = 30, length = 3 + 0 }, %) | ||||
|     |> angledLineToX({ angle = 22.14 + 0, to = 12 }, %) | ||||
|     |> angledLineToY({ angle = 30, to = 11.14 }, %) | ||||
|     |> angledLineThatIntersects({ | ||||
|           angle = 3.14, | ||||
|           intersectTag = a, | ||||
|           offset = 0 | ||||
|         }, %) | ||||
|     |> tangentialArcTo([13.14 + 0, 13.14], %) | ||||
|     |> close(%) | ||||
|     |> extrude(5 + 7, %) | ||||
|   ` | ||||
|   |> startProfileAt([20, 0], %) | ||||
|   |> line([7.13, 4 + 0], %) | ||||
|   |> angledLine({ angle = 3 + 0, length = 3.14 + 0 }, %) | ||||
|   |> lineTo([20.14 + 0, -0.14 + 0], %) | ||||
|   |> xLineTo(29 + 0, %) | ||||
|   |> yLine(-3.14 + 0, %, $a) | ||||
|   |> xLine(1.63, %) | ||||
|   |> angledLineOfXLength({ angle = 3 + 0, length = 3.14 }, %) | ||||
|   |> angledLineOfYLength({ angle = 30, length = 3 + 0 }, %) | ||||
|   |> angledLineToX({ angle = 22.14 + 0, to = 12 }, %) | ||||
|   |> angledLineToY({ angle = 30, to = 11.14 }, %) | ||||
|   |> angledLineThatIntersects({ | ||||
|         angle = 3.14, | ||||
|         intersectTag = a, | ||||
|         offset = 0 | ||||
|       }, %) | ||||
|   |> tangentialArcTo([13.14 + 0, 13.14], %) | ||||
|   |> close(%) | ||||
|   |> extrude(5 + 7, %) | ||||
|     ` | ||||
|       ) | ||||
|     }, KCL_DEFAULT_LENGTH) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // wait for execution done | ||||
|     await u.openDebugPanel() | ||||
| @ -650,7 +644,7 @@ sketch002 = startSketchOn(launderExtrudeThroughVar, seg02) | ||||
|     await checkCodeAtHoverPosition( | ||||
|       'flatExtrusionFace', | ||||
|       flatExtrusionFace, | ||||
|       `angledLineThatIntersects({angle=3.14,intersectTag=a,offset=0},%)extrude(5+7,%)`, | ||||
|       `angledLineThatIntersects({angle:3.14,intersectTag:a,offset:0},%)extrude(5+7,%)`, | ||||
|       '}, %)' | ||||
|     ) | ||||
|  | ||||
| @ -707,19 +701,19 @@ sketch002 = startSketchOn(launderExtrudeThroughVar, seg02) | ||||
|     await checkCodeAtHoverPosition( | ||||
|       'straightSegmentEdge', | ||||
|       straightSegmentEdge, | ||||
|       `angledLineToY({angle=30,to=11.14},%)`, | ||||
|       'angledLineToY({ angle = 30, to = 11.14 }, %)' | ||||
|       `angledLineToY({angle:30,to:11.14},%)`, | ||||
|       'angledLineToY({ angle: 30, to: 11.14 }, %)' | ||||
|     ) | ||||
|     await checkCodeAtHoverPosition( | ||||
|       'straightSegmentOppositeEdge', | ||||
|       straightSegmentOppositeEdge, | ||||
|       `angledLineToY({angle=30,to=11.14},%)`, | ||||
|       'angledLineToY({ angle = 30, to = 11.14 }, %)' | ||||
|       `angledLineToY({angle:30,to:11.14},%)`, | ||||
|       'angledLineToY({ angle: 30, to: 11.14 }, %)' | ||||
|     ) | ||||
|     await checkCodeAtHoverPosition( | ||||
|       'straightSegmentAdjacentEdge', | ||||
|       straightSegmentAdjacentEdge, | ||||
|       `angledLineThatIntersects({angle=3.14,intersectTag=a,offset=0},%)`, | ||||
|       `angledLineThatIntersects({angle:3.14,intersectTag:a,offset:0},%)`, | ||||
|       '}, %)' | ||||
|     ) | ||||
|  | ||||
| @ -727,29 +721,29 @@ sketch002 = startSketchOn(launderExtrudeThroughVar, seg02) | ||||
|  | ||||
|     await u.removeCurrentCode() | ||||
|     await u.codeLocator.fill(`sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([75.8, 317.2], %) // [$startCapTag, $EndCapTag] | ||||
|   |> angledLine([0, 268.43], %, $rectangleSegmentA001) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001) - 90, | ||||
|        217.26 | ||||
|      ], %, $seg01) | ||||
|   |> angledLine([ | ||||
|        segAng(rectangleSegmentA001), | ||||
|        -segLen(rectangleSegmentA001) | ||||
|      ], %, $yo) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %, $seg02) | ||||
|   |> close(%) | ||||
| extrude001 = extrude(100, sketch001) | ||||
|   |> chamfer({ | ||||
|        length = 30, | ||||
|        tags = [ | ||||
|          seg01, | ||||
|          getNextAdjacentEdge(yo), | ||||
|          getNextAdjacentEdge(seg02), | ||||
|          getOppositeEdge(seg01) | ||||
|        ] | ||||
|      }, %) | ||||
| `) | ||||
|     |> startProfileAt([75.8, 317.2], %) // [$startCapTag, $EndCapTag] | ||||
|     |> angledLine([0, 268.43], %, $rectangleSegmentA001) | ||||
|     |> angledLine([ | ||||
|      segAng(rectangleSegmentA001) - 90, | ||||
|      217.26 | ||||
|    ], %, $seg01) | ||||
|     |> angledLine([ | ||||
|      segAng(rectangleSegmentA001), | ||||
|      -segLen(rectangleSegmentA001) | ||||
|    ], %, $yo) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %, $seg02) | ||||
|     |> close(%) | ||||
|   extrude001 = extrude(100, sketch001) | ||||
|     |> chamfer({ | ||||
|      length = 30, | ||||
|      tags = [ | ||||
|        seg01, | ||||
|        getNextAdjacentEdge(yo), | ||||
|        getNextAdjacentEdge(seg02), | ||||
|        getOppositeEdge(seg01) | ||||
|      ] | ||||
|    }, %) | ||||
|   `) | ||||
|     await expect( | ||||
|       page.getByTestId('model-state-indicator-execution-done') | ||||
|     ).toBeVisible() | ||||
| @ -786,14 +780,14 @@ extrude001 = extrude(100, sketch001) | ||||
|     await checkCodeAtHoverPosition( | ||||
|       'oppositeChamfer', | ||||
|       oppositeChamfer, | ||||
|       `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01)chamfer({length=30,tags=[seg01,getNextAdjacentEdge(yo),getNextAdjacentEdge(seg02),getOppositeEdge(seg01)]},%)`, | ||||
|       `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01)chamfer({length:30,tags:[seg01,getNextAdjacentEdge(yo),getNextAdjacentEdge(seg02),getOppositeEdge(seg01)]},%)`, | ||||
|       '}, %)' | ||||
|     ) | ||||
|  | ||||
|     await checkCodeAtHoverPosition( | ||||
|       'baseChamfer', | ||||
|       baseChamfer, | ||||
|       `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01)chamfer({length=30,tags=[seg01,getNextAdjacentEdge(yo),getNextAdjacentEdge(seg02),getOppositeEdge(seg01)]},%)`, | ||||
|       `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01)chamfer({length:30,tags:[seg01,getNextAdjacentEdge(yo),getNextAdjacentEdge(seg02),getOppositeEdge(seg01)]},%)`, | ||||
|       '}, %)' | ||||
|     ) | ||||
|  | ||||
| @ -824,52 +818,58 @@ extrude001 = extrude(100, sketch001) | ||||
|     await checkCodeAtHoverPosition( | ||||
|       'adjacentChamfer1', | ||||
|       adjacentChamfer1, | ||||
|       `lineTo([profileStartX(%),profileStartY(%)],%,$seg02)chamfer({length=30,tags=[seg01,getNextAdjacentEdge(yo),getNextAdjacentEdge(seg02),getOppositeEdge(seg01)]},%)`, | ||||
|       `lineTo([profileStartX(%),profileStartY(%)],%,$seg02)chamfer({length:30,tags:[seg01,getNextAdjacentEdge(yo),getNextAdjacentEdge(seg02),getOppositeEdge(seg01)]},%)`, | ||||
|       '}, %)' | ||||
|     ) | ||||
|  | ||||
|     await checkCodeAtHoverPosition( | ||||
|       'adjacentChamfer2', | ||||
|       adjacentChamfer2, | ||||
|       `angledLine([segAng(rectangleSegmentA001),-segLen(rectangleSegmentA001)],%,$yo)chamfer({length=30,tags=[seg01,getNextAdjacentEdge(yo),getNextAdjacentEdge(seg02),getOppositeEdge(seg01)]},%)`, | ||||
|       `angledLine([segAng(rectangleSegmentA001),-segLen(rectangleSegmentA001)],%,$yo)chamfer({length:30,tags:[seg01,getNextAdjacentEdge(yo),getNextAdjacentEdge(seg02),getOppositeEdge(seg01)]},%)`, | ||||
|       '}, %)' | ||||
|     ) | ||||
|   }) | ||||
|   test("Extrude button should be disabled if there's no extrudable geometry when nothing is selected", async ({ | ||||
|     page, | ||||
|     editor, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([3.29, 7.86], %) | ||||
|   |> line([2.48, 2.44], %) | ||||
|   |> line([2.66, 1.17], %) | ||||
|   |> line([3.75, 0.46], %) | ||||
|   |> line([4.99, -0.46], %, $seg01) | ||||
|   |> line([3.3, -2.12], %) | ||||
|   |> line([2.16, -3.33], %) | ||||
|   |> line([0.85, -3.08], %) | ||||
|   |> line([-0.18, -3.36], %) | ||||
|   |> line([-3.86, -2.73], %) | ||||
|   |> line([-17.67, 0.85], %) | ||||
|   |> close(%) | ||||
| extrude001 = extrude(10, sketch001) | ||||
|   ` | ||||
|     |> startProfileAt([3.29, 7.86], %) | ||||
|     |> line([2.48, 2.44], %) | ||||
|     |> line([2.66, 1.17], %) | ||||
|     |> line([3.75, 0.46], %) | ||||
|     |> line([4.99, -0.46], %, $seg01) | ||||
|     |> line([3.3, -2.12], %) | ||||
|     |> line([2.16, -3.33], %) | ||||
|     |> line([0.85, -3.08], %) | ||||
|     |> line([-0.18, -3.36], %) | ||||
|     |> line([-3.86, -2.73], %) | ||||
|     |> line([-17.67, 0.85], %) | ||||
|     |> close(%) | ||||
|   extrude001 = extrude(10, sketch001) | ||||
|     ` | ||||
|       ) | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     // wait for execution done | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
|     const selectUnExtrudable = () => | ||||
|       page.getByText(`line([4.99, -0.46], %, $seg01)`).click() | ||||
|     const selectUnExtrudable = async () => { | ||||
|       await editor.scrollToText(`line([4.99, -0.46], %, $seg01)`) | ||||
|       await page.getByText(`line([4.99, -0.46], %, $seg01)`).click() | ||||
|     } | ||||
|     const clickEmpty = () => page.mouse.click(700, 460) | ||||
|     await selectUnExtrudable() | ||||
|     // expect extrude button to be disabled | ||||
| @ -879,17 +879,18 @@ extrude001 = extrude(10, sketch001) | ||||
|  | ||||
|     // expect active line to contain nothing | ||||
|     await expect(page.locator('.cm-activeLine')).toHaveText('') | ||||
|  | ||||
|     // and extrude to still be disabled | ||||
|     await expect(page.getByRole('button', { name: 'Extrude' })).toBeDisabled() | ||||
|  | ||||
|     const codeToAdd = `${await u.codeLocator.allInnerTexts()} | ||||
| sketch002 = startSketchOn(extrude001, $seg01) | ||||
|   |> startProfileAt([-12.94, 6.6], %) | ||||
|   |> line([2.45, -0.2], %) | ||||
|   |> line([-2, -1.25], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| ` | ||||
|   sketch002 = startSketchOn(extrude001, $seg01) | ||||
|     |> startProfileAt([-12.94, 6.6], %) | ||||
|     |> line([2.45, -0.2], %) | ||||
|     |> line([-2, -1.25], %) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|   ` | ||||
|     await u.codeLocator.fill(codeToAdd) | ||||
|  | ||||
|     await selectUnExtrudable() | ||||
| @ -904,23 +905,23 @@ sketch002 = startSketchOn(extrude001, $seg01) | ||||
|     ).not.toBeDisabled() | ||||
|   }) | ||||
|  | ||||
|   test('Fillet button states test', async ({ page }) => { | ||||
|   test('Fillet button states test', async ({ page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-5, -5], %) | ||||
|   |> line([0, 10], %) | ||||
|   |> line([10, 0], %) | ||||
|   |> line([0, -10], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%)` | ||||
|     |> startProfileAt([-5, -5], %) | ||||
|     |> line([0, 10], %) | ||||
|     |> line([10, 0], %) | ||||
|     |> line([0, -10], %) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%)` | ||||
|       ) | ||||
|     }) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
| @ -937,7 +938,7 @@ sketch002 = startSketchOn(extrude001, $seg01) | ||||
|  | ||||
|     // test fillet button with the body in the scene | ||||
|     const codeToAdd = `${await u.codeLocator.allInnerTexts()} | ||||
| extrude001 = extrude(10, sketch001)` | ||||
|   extrude001 = extrude(10, sketch001)` | ||||
|     await u.codeLocator.clear() | ||||
|     await u.codeLocator.fill(codeToAdd) | ||||
|     await selectSegment() | ||||
| @ -958,6 +959,7 @@ extrude001 = extrude(10, sketch001)` | ||||
|  | ||||
|   test('Testing selections (and hovers) work on sketches when NOT in sketch mode', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const cases = [ | ||||
|       { | ||||
| @ -978,21 +980,21 @@ extrude001 = extrude(10, sketch001)` | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `yo = 79 | ||||
| part001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-7.54, -26.74], %) | ||||
|   |> ${cases[0].expectedCode} | ||||
|   |> line([-3.19, -138.43], %) | ||||
|   |> ${cases[1].expectedCode} | ||||
|   |> line([41.19, 28.97 + 5], %) | ||||
|   |> ${cases[2].expectedCode}` | ||||
|   part001 = startSketchOn('XZ') | ||||
|     |> startProfileAt([-7.54, -26.74], %) | ||||
|     |> ${cases[0].expectedCode} | ||||
|     |> line([-3.19, -138.43], %) | ||||
|     |> ${cases[1].expectedCode} | ||||
|     |> line([41.19, 28.97 + 5], %) | ||||
|     |> ${cases[2].expectedCode}` | ||||
|         ) | ||||
|       }, | ||||
|       { cases } | ||||
|     ) | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.openAndClearDebugPanel() | ||||
|  | ||||
|     await u.sendCustomCmd({ | ||||
| @ -1025,24 +1027,25 @@ part001 = startSketchOn('XZ') | ||||
|   }) | ||||
|   test("Hovering and selection of extruded faces works, and is not overridden shortly after user's click", async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     await page.addInitScript(async () => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-79.26, 95.04], %) | ||||
|   |> line([112.54, 127.64], %) | ||||
|   |> line([170.36, -121.61], %, $seg01) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| extrude001 = extrude(50, sketch001) | ||||
|         ` | ||||
|     |> startProfileAt([-79.26, 95.04], %) | ||||
|     |> line([112.54, 127.64], %) | ||||
|     |> line([170.36, -121.61], %, $seg01) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|   extrude001 = extrude(50, sketch001) | ||||
|       ` | ||||
|       ) | ||||
|     }) | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.openAndClearDebugPanel() | ||||
|  | ||||
|     await u.sendCustomCmd({ | ||||
| @ -1125,6 +1128,7 @@ extrude001 = extrude(50, sketch001) | ||||
|   }) | ||||
|   test("Various pipe expressions should and shouldn't allow edit and or extrude", async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     const selectionsSnippets = { | ||||
| @ -1143,46 +1147,46 @@ extrude001 = extrude(50, sketch001) | ||||
|         localStorage.setItem( | ||||
|           'persistCode', | ||||
|           `part001 = startSketchOn('XZ') | ||||
|     ${extrudeAndEditBlocked} | ||||
|     |> line([25.96, 2.93], %) | ||||
|     |> line([5.25, -5.72], %) | ||||
|     |> line([-2.01, -10.35], %) | ||||
|     |> line([-27.65, -2.78], %) | ||||
|     |> close(%) | ||||
|     |> extrude(5, %) | ||||
|   sketch002 = startSketchOn('XZ') | ||||
|     ${extrudeAndEditAllowed} | ||||
|     |> line([10.32, 6.47], %) | ||||
|     |> line([9.71, -6.16], %) | ||||
|     |> line([-3.08, -9.86], %) | ||||
|     |> line([-12.02, -1.54], %) | ||||
|     |> close(%) | ||||
|   sketch003 = startSketchOn('XZ') | ||||
|     ${editOnly} | ||||
|     |> line([27.55, -1.65], %) | ||||
|     |> line([4.95, -8], %) | ||||
|     |> line([-20.38, -10.12], %) | ||||
|     |> line([-15.79, 17.08], %) | ||||
|  | ||||
|   fn yohey = (pos) => { | ||||
|     sketch004 = startSketchOn('XZ') | ||||
|     ${extrudeAndEditBlockedInFunction} | ||||
|     |> line([27.55, -1.65], %) | ||||
|     |> line([4.95, -10.53], %) | ||||
|     |> line([-20.38, -8], %) | ||||
|     |> line([-15.79, 17.08], %) | ||||
|     return '' | ||||
|   } | ||||
|  | ||||
|       yohey([15.79, -34.6]) | ||||
|   ` | ||||
|   ${extrudeAndEditBlocked} | ||||
|   |> line([25.96, 2.93], %) | ||||
|   |> line([5.25, -5.72], %) | ||||
|   |> line([-2.01, -10.35], %) | ||||
|   |> line([-27.65, -2.78], %) | ||||
|   |> close(%) | ||||
|   |> extrude(5, %) | ||||
|     sketch002 = startSketchOn('XZ') | ||||
|   ${extrudeAndEditAllowed} | ||||
|   |> line([10.32, 6.47], %) | ||||
|   |> line([9.71, -6.16], %) | ||||
|   |> line([-3.08, -9.86], %) | ||||
|   |> line([-12.02, -1.54], %) | ||||
|   |> close(%) | ||||
|     sketch003 = startSketchOn('XZ') | ||||
|   ${editOnly} | ||||
|   |> line([27.55, -1.65], %) | ||||
|   |> line([4.95, -8], %) | ||||
|   |> line([-20.38, -10.12], %) | ||||
|   |> line([-15.79, 17.08], %) | ||||
|    | ||||
|     fn yohey = (pos) => { | ||||
|   sketch004 = startSketchOn('XZ') | ||||
|   ${extrudeAndEditBlockedInFunction} | ||||
|   |> line([27.55, -1.65], %) | ||||
|   |> line([4.95, -10.53], %) | ||||
|   |> line([-20.38, -8], %) | ||||
|   |> line([-15.79, 17.08], %) | ||||
|   return '' | ||||
|     } | ||||
|    | ||||
|     yohey([15.79, -34.6]) | ||||
|     ` | ||||
|         ) | ||||
|       }, | ||||
|       selectionsSnippets | ||||
|     ) | ||||
|     await page.setViewportSize({ width: 1200, height: 1000 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 1000 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|  | ||||
|     // wait for execution done | ||||
|     await u.openDebugPanel() | ||||
| @ -1222,6 +1226,7 @@ extrude001 = extrude(50, sketch001) | ||||
|  | ||||
|   test('Deselecting line tool should mean nothing happens on click', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     /** | ||||
|      * If the line tool is clicked when the state is 'No Points' it will exit Sketch mode. | ||||
| @ -1230,9 +1235,9 @@ extrude001 = extrude(50, sketch001) | ||||
|      * To continue to test this workflow, we now enter sketch mode and place a single point before exiting the line tool. | ||||
|      */ | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.openDebugPanel() | ||||
|  | ||||
|     await expect( | ||||
|  | ||||
| @ -1,14 +1,7 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
| import * as fsp from 'fs/promises' | ||||
| import { join } from 'path' | ||||
| import { | ||||
|   getUtils, | ||||
|   setup, | ||||
|   setupElectron, | ||||
|   tearDown, | ||||
|   executorInputPath, | ||||
|   createProject, | ||||
| } from './test-utils' | ||||
| import { getUtils, executorInputPath, createProject } from './test-utils' | ||||
| import { SaveSettingsPayload, SettingsLevel } from 'lib/settings/settingsTypes' | ||||
| import { SETTINGS_FILE_NAME, PROJECT_SETTINGS_FILE_NAME } from 'lib/constants' | ||||
| import { | ||||
| @ -19,141 +12,136 @@ import { | ||||
| } from './storageStates' | ||||
| import * as TOML from '@iarna/toml' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.describe('Testing settings', () => { | ||||
|   test('Stored settings are validated and fall back to defaults', async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|   test( | ||||
|     'Stored settings are validated and fall back to defaults', | ||||
|     // Override beforeEach test setup | ||||
|     // with corrupted settings | ||||
|     await page.addInitScript( | ||||
|       async ({ settingsKey, settings }) => { | ||||
|         localStorage.setItem(settingsKey, settings) | ||||
|       }, | ||||
|       { | ||||
|         settingsKey: TEST_SETTINGS_KEY, | ||||
|         settings: TOML.stringify({ settings: TEST_SETTINGS_CORRUPTED }), | ||||
|       } | ||||
|     ) | ||||
|     { | ||||
|       appSettings: TEST_SETTINGS_CORRUPTED, | ||||
|     }, | ||||
|     async ({ page, homePage }) => { | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       // Check the settings were reset | ||||
|       const storedSettings = TOML.parse( | ||||
|         await page.evaluate( | ||||
|           ({ settingsKey }) => localStorage.getItem(settingsKey) || '', | ||||
|           { settingsKey: TEST_SETTINGS_KEY } | ||||
|         ) | ||||
|       ) as { settings: SaveSettingsPayload } | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|       expect(storedSettings.settings?.app?.theme).toBe('dark') | ||||
|  | ||||
|     // Check the settings were reset | ||||
|     const storedSettings = TOML.parse( | ||||
|       await page.evaluate( | ||||
|         ({ settingsKey }) => localStorage.getItem(settingsKey) || '', | ||||
|         { settingsKey: TEST_SETTINGS_KEY } | ||||
|       // Check that the invalid settings were changed to good defaults | ||||
|       expect(storedSettings.settings?.modeling?.defaultUnit).toBe('in') | ||||
|       expect(storedSettings.settings?.modeling?.mouseControls).toBe('KittyCAD') | ||||
|       expect(storedSettings.settings?.app?.projectDirectory).toBe('') | ||||
|       expect(storedSettings.settings?.projects?.defaultProjectName).toBe( | ||||
|         'project-$nnn' | ||||
|       ) | ||||
|     ) as { settings: SaveSettingsPayload } | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|     expect(storedSettings.settings?.app?.theme).toBe(undefined) | ||||
|   // The behavior is actually broken. Parent always takes precedence | ||||
|   test.fixme( | ||||
|     'Project settings can be set and override user settings', | ||||
|     async ({ page, homePage }) => { | ||||
|       const u = await getUtils(page) | ||||
|       await test.step(`Setup`, async () => { | ||||
|         await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|         await homePage.goToModelingScene() | ||||
|         await page | ||||
|           .getByRole('button', { name: 'Start Sketch' }) | ||||
|           .waitFor({ state: 'visible' }) | ||||
|       }) | ||||
|  | ||||
|     // Check that the invalid settings were removed | ||||
|     expect(storedSettings.settings?.modeling?.defaultUnit).toBe(undefined) | ||||
|     expect(storedSettings.settings?.modeling?.mouseControls).toBe(undefined) | ||||
|     expect(storedSettings.settings?.app?.projectDirectory).toBe(undefined) | ||||
|     expect(storedSettings.settings?.projects?.defaultProjectName).toBe( | ||||
|       undefined | ||||
|     ) | ||||
|   }) | ||||
|       // Selectors and constants | ||||
|       const paneButtonLocator = page.getByTestId('debug-pane-button') | ||||
|       const headingLocator = page.getByRole('heading', { | ||||
|         name: 'Settings', | ||||
|         exact: true, | ||||
|       }) | ||||
|       const inputLocator = page.locator('input[name="modeling-showDebugPanel"]') | ||||
|  | ||||
|   test('Project settings can be set and override user settings', async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await test.step(`Setup`, async () => { | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await test.step('Open settings dialog and set "Show debug panel" to on', async () => { | ||||
|         await page.keyboard.press('ControlOrMeta+,') | ||||
|         await expect(headingLocator).toBeVisible() | ||||
|  | ||||
|         /** Test to close https://github.com/KittyCAD/modeling-app/issues/2713 */ | ||||
|         await test.step(`Confirm that this dialog has a solid background`, async () => { | ||||
|           await expect | ||||
|             .poll( | ||||
|               () => u.getGreatestPixDiff({ x: 600, y: 250 }, [28, 28, 28]), | ||||
|               { | ||||
|                 timeout: 1000, | ||||
|                 message: | ||||
|                   'Checking for solid background, should not see default plane colors', | ||||
|               } | ||||
|             ) | ||||
|             .toBeLessThan(15) | ||||
|         }) | ||||
|  | ||||
|         await page.locator('#showDebugPanel').getByText('OffOn').click() | ||||
|       }) | ||||
|  | ||||
|       // Close it and open again with keyboard shortcut, while KCL editor is focused | ||||
|       // Put the cursor in the editor | ||||
|       await test.step('Open settings with keyboard shortcut', async () => { | ||||
|         await page.getByTestId('settings-close-button').click() | ||||
|         await page.locator('.cm-content').click() | ||||
|         await page.keyboard.press('ControlOrMeta+,') | ||||
|         await expect(headingLocator).toBeVisible() | ||||
|       }) | ||||
|  | ||||
|       // Verify the toast appeared | ||||
|       await expect( | ||||
|         page.getByText(`Set show debug panel to "false" for this project`) | ||||
|       ).toBeVisible() | ||||
|       await expect( | ||||
|         page.getByText(`Set show debug panel to "false" for this project`) | ||||
|       ).not.toBeVisible() | ||||
|  | ||||
|       // Check that the debug panel button is gone | ||||
|       await expect(paneButtonLocator).not.toBeVisible() | ||||
|  | ||||
|       // Check that the user setting was not changed | ||||
|       await page.getByRole('radio', { name: 'User' }).click() | ||||
|       await expect(inputLocator).toBeChecked() | ||||
|  | ||||
|       // Roll back to default of "off" | ||||
|       await await page | ||||
|         .getByText( | ||||
|           'show debug panelRoll back show debug panelRoll back to match' | ||||
|         ) | ||||
|         .hover() | ||||
|       await page | ||||
|         .getByRole('button', { name: 'Start Sketch' }) | ||||
|         .waitFor({ state: 'visible' }) | ||||
|     }) | ||||
|         .getByRole('button', { | ||||
|           name: 'Roll back show debug panel', | ||||
|         }) | ||||
|         .click() | ||||
|       await expect(inputLocator).not.toBeChecked() | ||||
|  | ||||
|     // Selectors and constants | ||||
|     const paneButtonLocator = page.getByTestId('debug-pane-button') | ||||
|     const headingLocator = page.getByRole('heading', { | ||||
|       name: 'Settings', | ||||
|       exact: true, | ||||
|     }) | ||||
|     const inputLocator = page.locator('input[name="modeling-showDebugPanel"]') | ||||
|  | ||||
|     await test.step('Open settings dialog and set "Show debug panel" to on', async () => { | ||||
|       await page.keyboard.press('ControlOrMeta+Shift+,') | ||||
|       await expect(headingLocator).toBeVisible() | ||||
|  | ||||
|       /** Test to close https://github.com/KittyCAD/modeling-app/issues/2713 */ | ||||
|       await test.step(`Confirm that this dialog has a solid background`, async () => { | ||||
|         await expect | ||||
|           .poll(() => u.getGreatestPixDiff({ x: 600, y: 250 }, [28, 28, 28]), { | ||||
|             timeout: 1000, | ||||
|             message: | ||||
|               'Checking for solid background, should not see default plane colors', | ||||
|           }) | ||||
|           .toBeLessThan(15) | ||||
|       }) | ||||
|  | ||||
|       await page.locator('#showDebugPanel').getByText('OffOn').click() | ||||
|     }) | ||||
|  | ||||
|     // Close it and open again with keyboard shortcut, while KCL editor is focused | ||||
|     // Put the cursor in the editor | ||||
|     await test.step('Open settings with keyboard shortcut', async () => { | ||||
|       await page.getByTestId('settings-close-button').click() | ||||
|       await page.locator('.cm-content').click() | ||||
|       await page.keyboard.press('ControlOrMeta+Shift+,') | ||||
|       await expect(headingLocator).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     // Verify the toast appeared | ||||
|     await expect( | ||||
|       page.getByText(`Set show debug panel to "false" for this project`) | ||||
|     ).toBeVisible() | ||||
|     // Check that the theme changed | ||||
|     await expect(paneButtonLocator).not.toBeVisible() | ||||
|  | ||||
|     // Check that the user setting was not changed | ||||
|     await page.getByRole('radio', { name: 'User' }).click() | ||||
|     await expect(inputLocator).toBeChecked() | ||||
|  | ||||
|     // Roll back to default of "off" | ||||
|     await await page | ||||
|       .getByText('show debug panelRoll back show debug panelRoll back to match') | ||||
|       .hover() | ||||
|     await page | ||||
|       .getByRole('button', { | ||||
|         name: 'Roll back show debug panel', | ||||
|       }) | ||||
|       .click() | ||||
|     await expect(inputLocator).not.toBeChecked() | ||||
|  | ||||
|     // Check that the project setting did not change | ||||
|     await page.getByRole('radio', { name: 'Project' }).click() | ||||
|     await expect( | ||||
|       page.locator('input[name="modeling-showDebugPanel"]') | ||||
|     ).not.toBeChecked() | ||||
|   }) | ||||
|       // Check that the project setting did not change | ||||
|       await page.getByRole('radio', { name: 'Project' }).click() | ||||
|       await expect( | ||||
|         page.locator('input[name="modeling-showDebugPanel"]') | ||||
|       ).not.toBeChecked() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test('Keybindings display the correct hotkey for Command Palette', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     await test.step('Open keybindings settings', async () => { | ||||
|       // Open the settings modal with the browser keyboard shortcut | ||||
|       await page.keyboard.press('ControlOrMeta+Shift+,') | ||||
|       // Open the settings modal with the keyboard shortcut | ||||
|       await page.keyboard.press('ControlOrMeta+,') | ||||
|  | ||||
|       // Go to Keybindings tab. | ||||
|       const keybindingsTab = page.getByRole('radio', { name: 'Keybindings' }) | ||||
| @ -174,116 +162,116 @@ test.describe('Testing settings', () => { | ||||
|     await expect(hotkey).toHaveText(text) | ||||
|   }) | ||||
|  | ||||
|   test('Project and user settings can be reset', async ({ page }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await test.step(`Setup`, async () => { | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|     }) | ||||
|  | ||||
|     // Selectors and constants | ||||
|     const projectSettingsTab = page.getByRole('radio', { name: 'Project' }) | ||||
|     const userSettingsTab = page.getByRole('radio', { name: 'User' }) | ||||
|     const resetButton = (level: SettingsLevel) => | ||||
|       page.getByRole('button', { | ||||
|         name: `Reset ${level}-level settings`, | ||||
|       }) | ||||
|     const themeColorSetting = page.locator('#themeColor').getByRole('slider') | ||||
|     const settingValues = { | ||||
|       default: '259', | ||||
|       user: '120', | ||||
|       project: '50', | ||||
|     } | ||||
|     const resetToast = (level: SettingsLevel) => | ||||
|       page.getByText(`${level}-level settings were reset`) | ||||
|  | ||||
|     await test.step(`Open the settings modal`, async () => { | ||||
|       await page.getByRole('link', { name: 'Settings' }).last().click() | ||||
|       await expect( | ||||
|         page.getByRole('heading', { name: 'Settings', exact: true }) | ||||
|       ).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await test.step('Set up theme color', async () => { | ||||
|       // Verify we're looking at the project-level settings, | ||||
|       // and it's set to default value | ||||
|       await expect(projectSettingsTab).toBeChecked() | ||||
|       await expect(themeColorSetting).toHaveValue(settingValues.default) | ||||
|  | ||||
|       // Set project-level value to 50 | ||||
|       await themeColorSetting.fill(settingValues.project) | ||||
|  | ||||
|       // Set user-level value to 120 | ||||
|       await userSettingsTab.click() | ||||
|       await themeColorSetting.fill(settingValues.user) | ||||
|       await projectSettingsTab.click() | ||||
|     }) | ||||
|  | ||||
|     await test.step('Reset project settings', async () => { | ||||
|       // Click the reset settings button. | ||||
|       await resetButton('project').click() | ||||
|  | ||||
|       await expect(resetToast('project')).toBeVisible() | ||||
|       await expect(resetToast('project')).not.toBeVisible() | ||||
|  | ||||
|       // Verify it is now set to the inherited user value | ||||
|       await expect(themeColorSetting).toHaveValue(settingValues.user) | ||||
|  | ||||
|       await test.step(`Check that the user settings did not change`, async () => { | ||||
|         await userSettingsTab.click() | ||||
|         await expect(themeColorSetting).toHaveValue(settingValues.user) | ||||
|   test.fixme( | ||||
|     'Project and user settings can be reset', | ||||
|     async ({ page, homePage }) => { | ||||
|       const u = await getUtils(page) | ||||
|       await test.step(`Setup`, async () => { | ||||
|         await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|         await homePage.goToModelingScene() | ||||
|         await u.waitForPageLoad() | ||||
|         await page.waitForTimeout(1000) | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Set project-level again to test the user-level reset`, async () => { | ||||
|         await projectSettingsTab.click() | ||||
|       // Selectors and constants | ||||
|       const projectSettingsTab = page.getByRole('radio', { name: 'Project' }) | ||||
|       const userSettingsTab = page.getByRole('radio', { name: 'User' }) | ||||
|       const resetButton = (level: SettingsLevel) => | ||||
|         page.getByRole('button', { | ||||
|           name: `Reset ${level}-level settings`, | ||||
|         }) | ||||
|       const themeColorSetting = page.locator('#themeColor').getByRole('slider') | ||||
|       const settingValues = { | ||||
|         default: '259', | ||||
|         user: '120', | ||||
|         project: '50', | ||||
|       } | ||||
|       const resetToast = (level: SettingsLevel) => | ||||
|         page.getByText(`${level}-level settings were reset`) | ||||
|  | ||||
|       await test.step(`Open the settings modal`, async () => { | ||||
|         await page.getByRole('link', { name: 'Settings' }).last().click() | ||||
|         await expect( | ||||
|           page.getByRole('heading', { name: 'Settings', exact: true }) | ||||
|         ).toBeVisible() | ||||
|       }) | ||||
|  | ||||
|       await test.step('Set up theme color', async () => { | ||||
|         // Verify we're looking at the project-level settings, | ||||
|         // and it's set to default value | ||||
|         await expect(projectSettingsTab).toBeChecked() | ||||
|         await expect(themeColorSetting).toHaveValue(settingValues.default) | ||||
|  | ||||
|         // Set project-level value to 50 | ||||
|         await themeColorSetting.fill(settingValues.project) | ||||
|  | ||||
|         // Set user-level value to 120 | ||||
|         await userSettingsTab.click() | ||||
|       }) | ||||
|     }) | ||||
|  | ||||
|     await test.step('Reset user settings', async () => { | ||||
|       // Click the reset settings button. | ||||
|       await resetButton('user').click() | ||||
|  | ||||
|       await expect(resetToast('user')).toBeVisible() | ||||
|       await expect(resetToast('user')).not.toBeVisible() | ||||
|  | ||||
|       // Verify it is now set to the default value | ||||
|       await expect(themeColorSetting).toHaveValue(settingValues.default) | ||||
|  | ||||
|       await test.step(`Check that the project settings did not change`, async () => { | ||||
|         await themeColorSetting.fill(settingValues.user) | ||||
|         await projectSettingsTab.click() | ||||
|         await expect(themeColorSetting).toHaveValue(settingValues.project) | ||||
|       }) | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|       await test.step('Reset project settings', async () => { | ||||
|         // Click the reset settings button. | ||||
|         await resetButton('project').click() | ||||
|  | ||||
|         await expect(resetToast('project')).toBeVisible() | ||||
|         await expect(resetToast('project')).not.toBeVisible() | ||||
|  | ||||
|         // Verify it is now set to the inherited user value | ||||
|         await expect(themeColorSetting).toHaveValue(settingValues.user) | ||||
|  | ||||
|         await test.step(`Check that the user settings did not change`, async () => { | ||||
|           await userSettingsTab.click() | ||||
|           await expect(themeColorSetting).toHaveValue(settingValues.user) | ||||
|         }) | ||||
|  | ||||
|         await test.step(`Set project-level again to test the user-level reset`, async () => { | ||||
|           await projectSettingsTab.click() | ||||
|           await themeColorSetting.fill(settingValues.project) | ||||
|           await userSettingsTab.click() | ||||
|         }) | ||||
|       }) | ||||
|  | ||||
|       await test.step('Reset user settings', async () => { | ||||
|         // Click the reset settings button. | ||||
|         await resetButton('user').click() | ||||
|  | ||||
|         await expect(resetToast('user')).toBeVisible() | ||||
|         await expect(resetToast('user')).not.toBeVisible() | ||||
|  | ||||
|         // Verify it is now set to the default value | ||||
|         await expect(themeColorSetting).toHaveValue(settingValues.default) | ||||
|  | ||||
|         await test.step(`Check that the project settings did not change`, async () => { | ||||
|           await projectSettingsTab.click() | ||||
|           await expect(themeColorSetting).toHaveValue(settingValues.project) | ||||
|         }) | ||||
|       }) | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test.fixme( | ||||
|     `Project settings override user settings on desktop`, | ||||
|     { tag: ['@electron', '@skipWin'] }, | ||||
|     async ({ browser: _ }, testInfo) => { | ||||
|     async ({ context, page }, testInfo) => { | ||||
|       test.skip( | ||||
|         process.platform === 'win32', | ||||
|         'TODO: remove this skip https://github.com/KittyCAD/modeling-app/issues/3557' | ||||
|       ) | ||||
|       const projectName = 'bracket' | ||||
|       const { | ||||
|         electronApp, | ||||
|         page, | ||||
|         dir: projectDirName, | ||||
|       } = await setupElectron({ | ||||
|         testInfo, | ||||
|         folderSetupFn: async (dir) => { | ||||
|       const { dir: projectDirName } = await context.folderSetupFn( | ||||
|         async (dir) => { | ||||
|           const bracketDir = join(dir, projectName) | ||||
|           await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|           await fsp.copyFile( | ||||
|             executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||
|             join(bracketDir, 'main.kcl') | ||||
|           ) | ||||
|         }, | ||||
|       }) | ||||
|         } | ||||
|       ) | ||||
|  | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|       // Selectors and constants | ||||
|       const tempProjectSettingsFilePath = join( | ||||
| @ -353,22 +341,18 @@ test.describe('Testing settings', () => { | ||||
|         await logoLink.click() | ||||
|         await expect(logoLink).toHaveCSS('--primary-hue', userThemeColor) | ||||
|       }) | ||||
|  | ||||
|       await electronApp.close() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test( | ||||
|     `Load desktop app with no settings file`, | ||||
|     { tag: '@electron' }, | ||||
|     async ({ browser: _ }, testInfo) => { | ||||
|       const { electronApp, page } = await setupElectron({ | ||||
|         // This is what makes no settings file get created | ||||
|         cleanProjectDir: false, | ||||
|         testInfo, | ||||
|       }) | ||||
|  | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     { | ||||
|       tag: '@electron', | ||||
|       // This is what makes no settings file get created | ||||
|       cleanProjectDir: false, | ||||
|     }, | ||||
|     async ({ page }, testInfo) => { | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|       // Selectors and constants | ||||
|       const errorHeading = page.getByRole('heading', { | ||||
| @ -379,25 +363,21 @@ test.describe('Testing settings', () => { | ||||
|       // If the app loads without exploding we're in the clear | ||||
|       await expect(errorHeading).not.toBeVisible() | ||||
|       await expect(projectDirLink).toBeVisible() | ||||
|  | ||||
|       await electronApp.close() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test( | ||||
|     `Load desktop app with a settings file, but no project directory setting`, | ||||
|     { tag: '@electron' }, | ||||
|     async ({ browser: _ }, testInfo) => { | ||||
|       const { electronApp, page } = await setupElectron({ | ||||
|         testInfo, | ||||
|         appSettings: { | ||||
|           app: { | ||||
|             themeColor: '259', | ||||
|           }, | ||||
|     { | ||||
|       tag: '@electron', | ||||
|       appSettings: { | ||||
|         app: { | ||||
|           themeColor: '259', | ||||
|         }, | ||||
|       }) | ||||
|  | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       }, | ||||
|     }, | ||||
|     async ({ context, page }, testInfo) => { | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|       // Selectors and constants | ||||
|       const errorHeading = page.getByRole('heading', { | ||||
| @ -408,32 +388,28 @@ test.describe('Testing settings', () => { | ||||
|       // If the app loads without exploding we're in the clear | ||||
|       await expect(errorHeading).not.toBeVisible() | ||||
|       await expect(projectDirLink).toBeVisible() | ||||
|  | ||||
|       await electronApp.close() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   // It was much easier to test the logo color than the background stream color. | ||||
|   test.fixme( | ||||
|     'user settings reload on external change, on project and modeling view', | ||||
|     { tag: '@electron' }, | ||||
|     async ({ browserName }, testInfo) => { | ||||
|       const { | ||||
|         electronApp, | ||||
|         page, | ||||
|         dir: projectDirName, | ||||
|       } = await setupElectron({ | ||||
|         testInfo, | ||||
|         appSettings: { | ||||
|           app: { | ||||
|             // Doesn't matter what you set it to. It will | ||||
|             // default to 264.5 | ||||
|             themeColor: '0', | ||||
|           }, | ||||
|     { | ||||
|       tag: '@electron', | ||||
|       appSettings: { | ||||
|         app: { | ||||
|           // Doesn't matter what you set it to. It will | ||||
|           // default to 264.5 | ||||
|           themeColor: '0', | ||||
|         }, | ||||
|       }) | ||||
|       }, | ||||
|     }, | ||||
|     async ({ context, page }, testInfo) => { | ||||
|       const { dir: projectDirName } = await context.folderSetupFn( | ||||
|         async () => {} | ||||
|       ) | ||||
|  | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|       const logoLink = page.getByTestId('app-logo') | ||||
|       const projectDirLink = page.getByText('Loaded from') | ||||
| @ -467,23 +443,18 @@ test.describe('Testing settings', () => { | ||||
|         await changeColor('21') | ||||
|         await expect(logoLink).toHaveCSS('--primary-hue', '21') | ||||
|       }) | ||||
|       await electronApp.close() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test( | ||||
|   test.fixme( | ||||
|     'project settings reload on external change', | ||||
|     { tag: '@electron' }, | ||||
|     async ({ browserName: _ }, testInfo) => { | ||||
|       const { | ||||
|         electronApp, | ||||
|         page, | ||||
|         dir: projectDirName, | ||||
|       } = await setupElectron({ | ||||
|         testInfo, | ||||
|       }) | ||||
|     async ({ context, page }, testInfo) => { | ||||
|       const { dir: projectDirName } = await context.folderSetupFn( | ||||
|         async () => {} | ||||
|       ) | ||||
|  | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|       const logoLink = page.getByTestId('app-logo') | ||||
|       const projectDirLink = page.getByText('Loaded from') | ||||
| @ -514,29 +485,24 @@ test.describe('Testing settings', () => { | ||||
|         await changeColorFs('99') | ||||
|         await expect(logoLink).toHaveCSS('--primary-hue', '99') | ||||
|       }) | ||||
|  | ||||
|       await electronApp.close() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test( | ||||
|     `Closing settings modal should go back to the original file being viewed`, | ||||
|     { tag: '@electron' }, | ||||
|     async ({ browser: _ }, testInfo) => { | ||||
|       const { electronApp, page } = await setupElectron({ | ||||
|         testInfo, | ||||
|         folderSetupFn: async (dir) => { | ||||
|           const bracketDir = join(dir, 'project-000') | ||||
|           await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|           await fsp.copyFile( | ||||
|             executorInputPath('cube.kcl'), | ||||
|             join(bracketDir, 'main.kcl') | ||||
|           ) | ||||
|           await fsp.copyFile( | ||||
|             executorInputPath('cylinder.kcl'), | ||||
|             join(bracketDir, '2.kcl') | ||||
|           ) | ||||
|         }, | ||||
|     async ({ context, page }, testInfo) => { | ||||
|       await context.folderSetupFn(async (dir) => { | ||||
|         const bracketDir = join(dir, 'project-000') | ||||
|         await fsp.mkdir(bracketDir, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           executorInputPath('cube.kcl'), | ||||
|           join(bracketDir, 'main.kcl') | ||||
|         ) | ||||
|         await fsp.copyFile( | ||||
|           executorInputPath('cylinder.kcl'), | ||||
|           join(bracketDir, '2.kcl') | ||||
|         ) | ||||
|       }) | ||||
|       const kclCube = await fsp.readFile(executorInputPath('cube.kcl'), 'utf-8') | ||||
|       const kclCylinder = await fsp.readFile( | ||||
| @ -552,7 +518,7 @@ test.describe('Testing settings', () => { | ||||
|         editorTextMatches, | ||||
|       } = await getUtils(page, test) | ||||
|  | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       page.on('console', console.log) | ||||
|  | ||||
|       await test.step('Precondition: Open to second project file', async () => { | ||||
| @ -583,16 +549,15 @@ test.describe('Testing settings', () => { | ||||
|       await test.step('Postcondition: Same file content is in editor as before settings opened', async () => { | ||||
|         await editorTextMatches(kclCylinder) | ||||
|       }) | ||||
|  | ||||
|       await electronApp.close() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test('Changing modeling default unit', async ({ page }) => { | ||||
|     const u = await getUtils(page) | ||||
|   test('Changing modeling default unit', async ({ page, homePage }) => { | ||||
|     await test.step(`Test setup`, async () => { | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       const toastMessage = page.getByText(`Successfully created "testDefault"`) | ||||
|       await expect(toastMessage).not.toBeVisible() | ||||
|       await page | ||||
|         .getByRole('button', { name: 'Start Sketch' }) | ||||
|         .waitFor({ state: 'visible' }) | ||||
| @ -619,7 +584,9 @@ test.describe('Testing settings', () => { | ||||
|       await userSettingsTab.click() | ||||
|       await defaultUnitSection.hover() | ||||
|       await defaultUnitRollbackButton.click() | ||||
|       await projectSettingsTab.hover() | ||||
|       await projectSettingsTab.click() | ||||
|       await page.waitForTimeout(1000) | ||||
|     }) | ||||
|  | ||||
|     await test.step('Change modeling default unit within project tab', async () => { | ||||
| @ -631,7 +598,10 @@ test.describe('Testing settings', () => { | ||||
|           const toastMessage = page.getByText( | ||||
|             `Set default unit to "${unitOfMeasure}" for this project` | ||||
|           ) | ||||
|  | ||||
|           // Assert visibility and disapperance | ||||
|           await expect(toastMessage).toBeVisible() | ||||
|           await expect(toastMessage).not.toBeVisible() | ||||
|         }) | ||||
|       } | ||||
|       await changeUnitOfMeasureInProjectTab('in') | ||||
| @ -643,7 +613,10 @@ test.describe('Testing settings', () => { | ||||
|     }) | ||||
|  | ||||
|     // Go to the user tab | ||||
|     await userSettingsTab.hover() | ||||
|     await userSettingsTab.click() | ||||
|     await page.waitForTimeout(1000) | ||||
|  | ||||
|     await test.step('Change modeling default unit within user tab', async () => { | ||||
|       const changeUnitOfMeasureInUserTab = async (unitOfMeasure: string) => { | ||||
|         await test.step(`Set modeling default unit to ${unitOfMeasure}`, async () => { | ||||
| @ -726,23 +699,26 @@ test.describe('Testing settings', () => { | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   test('Changing theme in sketch mode', async ({ page }) => { | ||||
|   test('Changing theme in sketch mode', async ({ context, page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(() => { | ||||
|     await context.addInitScript(() => { | ||||
|       localStorage.setItem( | ||||
|         'persistCode', | ||||
|         `sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> line([5, 0], %) | ||||
|   |> line([0, 5], %) | ||||
|   |> line([-5, 0], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| extrude001 = extrude(5, sketch001) | ||||
| ` | ||||
|     |> startProfileAt([0, 0], %) | ||||
|     |> line([5, 0], %) | ||||
|     |> line([0, 5], %) | ||||
|     |> line([-5, 0], %) | ||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|     |> close(%) | ||||
|   extrude001 = extrude(5, sketch001) | ||||
|   ` | ||||
|       ) | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|     await page.waitForTimeout(1000) | ||||
|  | ||||
|     // Selectors and constants | ||||
|     const editSketchButton = page.getByRole('button', { name: 'Edit Sketch' }) | ||||
| @ -753,7 +729,6 @@ extrude001 = extrude(5, sketch001) | ||||
|     const lightThemeSegmentColor: [number, number, number] = [90, 90, 90] | ||||
|  | ||||
|     await test.step(`Get into sketch mode`, async () => { | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await page.mouse.click(700, 200) | ||||
|       await expect(editSketchButton).toBeVisible() | ||||
|       await editSketchButton.click() | ||||
| @ -792,135 +767,125 @@ extrude001 = extrude(5, sketch001) | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   test(`Changing system theme preferences (via media query) should update UI and stream`, async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     // Override the settings so that the theme is set to `system` | ||||
|     await page.addInitScript( | ||||
|       ({ settingsKey, settings }) => { | ||||
|         localStorage.setItem(settingsKey, settings) | ||||
|       }, | ||||
|       { | ||||
|         settingsKey: TEST_SETTINGS_KEY, | ||||
|         settings: TOML.stringify({ | ||||
|           settings: TEST_SETTINGS_DEFAULT_THEME, | ||||
|         }), | ||||
|   test( | ||||
|     `Changing system theme preferences (via media query) should update UI and stream`, | ||||
|     { | ||||
|       // Override the settings so that the theme is set to `system` | ||||
|       appSettings: TEST_SETTINGS_DEFAULT_THEME, | ||||
|     }, | ||||
|     async ({ page, homePage }) => { | ||||
|       const u = await getUtils(page) | ||||
|  | ||||
|       // Selectors and constants | ||||
|       const darkBackgroundCss = 'oklch(0.3012 0 264.5)' | ||||
|       const lightBackgroundCss = 'oklch(0.9911 0 264.5)' | ||||
|       const darkBackgroundColor: [number, number, number] = [27, 27, 27] | ||||
|       const lightBackgroundColor: [number, number, number] = [245, 245, 245] | ||||
|       const streamBackgroundPixelIsColor = async ( | ||||
|         color: [number, number, number] | ||||
|       ) => { | ||||
|         return u.getGreatestPixDiff({ x: 1000, y: 200 }, color) | ||||
|       } | ||||
|     ) | ||||
|     const u = await getUtils(page) | ||||
|       const toolbar = page.locator('menu').filter({ hasText: 'Start Sketch' }) | ||||
|  | ||||
|     // Selectors and constants | ||||
|     const darkBackgroundCss = 'oklch(0.3012 0 264.5)' | ||||
|     const lightBackgroundCss = 'oklch(0.9911 0 264.5)' | ||||
|     const darkBackgroundColor: [number, number, number] = [27, 27, 27] | ||||
|     const lightBackgroundColor: [number, number, number] = [245, 245, 245] | ||||
|     const streamBackgroundPixelIsColor = async ( | ||||
|       color: [number, number, number] | ||||
|     ) => { | ||||
|       return u.getGreatestPixDiff({ x: 1000, y: 200 }, color) | ||||
|       await test.step(`Test setup`, async () => { | ||||
|         await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|         await homePage.goToModelingScene() | ||||
|         await u.waitForPageLoad() | ||||
|         await page.waitForTimeout(1000) | ||||
|         await expect(toolbar).toBeVisible() | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Check the background color is light before`, async () => { | ||||
|         await expect(toolbar).toHaveCSS('background-color', lightBackgroundCss) | ||||
|         await expect | ||||
|           .poll(() => streamBackgroundPixelIsColor(lightBackgroundColor)) | ||||
|           .toBeLessThan(15) | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Change media query preference to dark, emulating dusk with system theme`, async () => { | ||||
|         await page.emulateMedia({ colorScheme: 'dark' }) | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Check the background color is dark after`, async () => { | ||||
|         await expect(toolbar).toHaveCSS('background-color', darkBackgroundCss) | ||||
|         await expect | ||||
|           .poll(() => streamBackgroundPixelIsColor(darkBackgroundColor)) | ||||
|           .toBeLessThan(15) | ||||
|       }) | ||||
|     } | ||||
|     const toolbar = page.locator('menu').filter({ hasText: 'Start Sketch' }) | ||||
|   ) | ||||
|  | ||||
|     await test.step(`Test setup`, async () => { | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await expect(toolbar).toBeVisible() | ||||
|     }) | ||||
|   test( | ||||
|     `Turning off "Show debug panel" with debug panel open leaves no phantom panel`, | ||||
|     { | ||||
|       // Override beforeEach test setup | ||||
|       // with debug panel open | ||||
|       // but "show debug panel" set to false | ||||
|       appSettings: { | ||||
|         ...TEST_SETTINGS, | ||||
|         modeling: { ...TEST_SETTINGS.modeling, showDebugPanel: false }, | ||||
|       }, | ||||
|     }, | ||||
|     async ({ context, page, homePage }) => { | ||||
|       const u = await getUtils(page) | ||||
|  | ||||
|     await test.step(`Check the background color is light before`, async () => { | ||||
|       await expect(toolbar).toHaveCSS('background-color', lightBackgroundCss) | ||||
|       await expect | ||||
|         .poll(() => streamBackgroundPixelIsColor(lightBackgroundColor)) | ||||
|         .toBeLessThan(15) | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Change media query preference to dark, emulating dusk with system theme`, async () => { | ||||
|       await page.emulateMedia({ colorScheme: 'dark' }) | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Check the background color is dark after`, async () => { | ||||
|       await expect(toolbar).toHaveCSS('background-color', darkBackgroundCss) | ||||
|       await expect | ||||
|         .poll(() => streamBackgroundPixelIsColor(darkBackgroundColor)) | ||||
|         .toBeLessThan(15) | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   test(`Turning off "Show debug panel" with debug panel open leaves no phantom panel`, async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     // Override beforeEach test setup | ||||
|     // with debug panel open | ||||
|     // but "show debug panel" set to false | ||||
|     await page.addInitScript( | ||||
|       async ({ settingsKey, settings }) => { | ||||
|         localStorage.setItem(settingsKey, settings) | ||||
|       await context.addInitScript(async () => { | ||||
|         localStorage.setItem( | ||||
|           'persistModelingContext', | ||||
|           '{"openPanes":["debug"]}' | ||||
|         ) | ||||
|       }, | ||||
|       { | ||||
|         settingsKey: TEST_SETTINGS_KEY, | ||||
|         settings: TOML.stringify({ | ||||
|           settings: { | ||||
|             ...TEST_SETTINGS, | ||||
|             modeling: { ...TEST_SETTINGS.modeling, showDebugPanel: false }, | ||||
|           }, | ||||
|         }), | ||||
|       }) | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|  | ||||
|       // Constants and locators | ||||
|       const resizeHandle = page.locator('.sidebar-resize-handles > div.block') | ||||
|       const debugPaneButton = page.getByTestId('debug-pane-button') | ||||
|       const commandsButton = page.getByRole('button', { name: 'Commands' }) | ||||
|       const debugPaneOption = page.getByRole('option', { | ||||
|         name: 'Settings · modeling · show debug panel', | ||||
|       }) | ||||
|  | ||||
|       async function setShowDebugPanelTo(value: 'On' | 'Off') { | ||||
|         await commandsButton.click() | ||||
|         await debugPaneOption.click() | ||||
|         await page.getByRole('option', { name: value }).click() | ||||
|         await expect( | ||||
|           page.getByText( | ||||
|             `Set show debug panel to "${value === 'On'}" for this project` | ||||
|           ) | ||||
|         ).toBeVisible() | ||||
|       } | ||||
|     ) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|  | ||||
|     // Constants and locators | ||||
|     const resizeHandle = page.locator('.sidebar-resize-handles > div.block') | ||||
|     const debugPaneButton = page.getByTestId('debug-pane-button') | ||||
|     const commandsButton = page.getByRole('button', { name: 'Commands' }) | ||||
|     const debugPaneOption = page.getByRole('option', { | ||||
|       name: 'Settings · modeling · show debug panel', | ||||
|     }) | ||||
|       await test.step(`Initial load with corrupted settings`, async () => { | ||||
|         // Check that the debug panel is not visible | ||||
|         await expect(debugPaneButton).not.toBeVisible() | ||||
|         // Check the pane resize handle wrapper is not visible | ||||
|         await expect(resizeHandle).not.toBeVisible() | ||||
|       }) | ||||
|  | ||||
|     async function setShowDebugPanelTo(value: 'On' | 'Off') { | ||||
|       await commandsButton.click() | ||||
|       await debugPaneOption.click() | ||||
|       await page.getByRole('option', { name: value }).click() | ||||
|       await expect( | ||||
|         page.getByText( | ||||
|           `Set show debug panel to "${value === 'On'}" for this project` | ||||
|         ) | ||||
|       ).toBeVisible() | ||||
|       await test.step(`Open code pane to verify we see the resize handles`, async () => { | ||||
|         await u.openKclCodePanel() | ||||
|         await expect(resizeHandle).toBeVisible() | ||||
|         await u.closeKclCodePanel() | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Turn on debug panel, open it`, async () => { | ||||
|         await setShowDebugPanelTo('On') | ||||
|         await expect(debugPaneButton).toBeVisible() | ||||
|         // We want the logic to clear the phantom panel, so we shouldn't see | ||||
|         // the real panel (and therefore the resize handle) yet | ||||
|         await expect(resizeHandle).not.toBeVisible() | ||||
|         await u.openDebugPanel() | ||||
|         await expect(resizeHandle).toBeVisible() | ||||
|       }) | ||||
|  | ||||
|       await test.step(`Turn off debug panel setting with it open`, async () => { | ||||
|         await setShowDebugPanelTo('Off') | ||||
|         await expect(debugPaneButton).not.toBeVisible() | ||||
|         await expect(resizeHandle).not.toBeVisible() | ||||
|       }) | ||||
|     } | ||||
|  | ||||
|     await test.step(`Initial load with corrupted settings`, async () => { | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       // Check that the debug panel is not visible | ||||
|       await expect(debugPaneButton).not.toBeVisible() | ||||
|       // Check the pane resize handle wrapper is not visible | ||||
|       await expect(resizeHandle).not.toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Open code pane to verify we see the resize handles`, async () => { | ||||
|       await u.openKclCodePanel() | ||||
|       await expect(resizeHandle).toBeVisible() | ||||
|       await u.closeKclCodePanel() | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Turn on debug panel, open it`, async () => { | ||||
|       await setShowDebugPanelTo('On') | ||||
|       await expect(debugPaneButton).toBeVisible() | ||||
|       // We want the logic to clear the phantom panel, so we shouldn't see | ||||
|       // the real panel (and therefore the resize handle) yet | ||||
|       await expect(resizeHandle).not.toBeVisible() | ||||
|       await u.openDebugPanel() | ||||
|       await expect(resizeHandle).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await test.step(`Turn off debug panel setting with it open`, async () => { | ||||
|       await setShowDebugPanelTo('Off') | ||||
|       await expect(debugPaneButton).not.toBeVisible() | ||||
|       await expect(resizeHandle).not.toBeVisible() | ||||
|     }) | ||||
|   }) | ||||
|   ) | ||||
| }) | ||||
|  | ||||
| @ -1,29 +1,16 @@ | ||||
| import { test, expect, Page } from '@playwright/test' | ||||
| import { | ||||
|   getUtils, | ||||
|   setup, | ||||
|   tearDown, | ||||
|   setupElectron, | ||||
|   createProject, | ||||
| } from './test-utils' | ||||
| import { test, expect, Page } from './zoo-test' | ||||
| import { getUtils, createProject } from './test-utils' | ||||
| import { join } from 'path' | ||||
| import fs from 'fs' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }) => { | ||||
|   await setup(context, page) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.describe('Text-to-CAD tests', () => { | ||||
|   test('basic lego happy case', async ({ page }) => { | ||||
|   test('basic lego happy case', async ({ page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     await test.step('Set up', async () => { | ||||
|       await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await u.waitForPageLoad() | ||||
|     }) | ||||
|  | ||||
|     await sendPromptFromCommandBar(page, 'a 2x4 lego') | ||||
| @ -43,25 +30,17 @@ test.describe('Text-to-CAD tests', () => { | ||||
|     const successToastMessage = page.getByText(`Text-to-CAD successful`) | ||||
|     await expect(successToastMessage).toBeVisible({ timeout: 15000 }) | ||||
|  | ||||
|     await expect(page.getByText('Copied')).not.toBeVisible() | ||||
|  | ||||
|     // Hit copy to clipboard. | ||||
|     // Hit accept. | ||||
|     const copyToClipboardButton = page.getByRole('button', { | ||||
|       name: 'Copy to clipboard', | ||||
|       name: 'Accept', | ||||
|     }) | ||||
|     await expect(copyToClipboardButton).toBeVisible() | ||||
|  | ||||
|     await copyToClipboardButton.click() | ||||
|  | ||||
|     // Expect the code to be copied. | ||||
|     await expect(page.getByText('Copied')).toBeVisible() | ||||
|  | ||||
|     // Click in the code editor. | ||||
|     await page.locator('.cm-content').click() | ||||
|  | ||||
|     // Paste the code. | ||||
|     await page.keyboard.press('ControlOrMeta+KeyV') | ||||
|  | ||||
|     // Expect the code to be pasted. | ||||
|     await expect(page.locator('.cm-content')).toContainText(`const`) | ||||
|  | ||||
| @ -70,29 +49,18 @@ test.describe('Text-to-CAD tests', () => { | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
|     // Find the toast close button. | ||||
|     const closeButton = page | ||||
|       .getByRole('status') | ||||
|       .locator('div') | ||||
|       .filter({ hasText: 'Text-to-CAD successfulPrompt' }) | ||||
|       .first() | ||||
|       .getByRole('button', { name: 'Close' }) | ||||
|     await expect(closeButton).toBeVisible() | ||||
|     await closeButton.click() | ||||
|  | ||||
|     // The toast should disappear. | ||||
|     await expect(successToastMessage).not.toBeVisible() | ||||
|   }) | ||||
|  | ||||
|   test('success model, then ignore success toast, user can create new prompt from command bar', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     await sendPromptFromCommandBar(page, 'a 2x6 lego') | ||||
|  | ||||
| @ -111,10 +79,6 @@ test.describe('Text-to-CAD tests', () => { | ||||
|     const successToastMessage = page.getByText(`Text-to-CAD successful`) | ||||
|     await expect(successToastMessage).toBeVisible({ timeout: 15000 }) | ||||
|  | ||||
|     await expect(page.getByText('Copied')).not.toBeVisible() | ||||
|  | ||||
|     await expect(successToastMessage).toBeVisible() | ||||
|  | ||||
|     // Can send a new prompt from the command bar. | ||||
|     await sendPromptFromCommandBar(page, 'a 2x4 lego') | ||||
|  | ||||
| @ -133,12 +97,14 @@ test.describe('Text-to-CAD tests', () => { | ||||
|  | ||||
|   test('you can reject text-to-cad output and it does nothing', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     await sendPromptFromCommandBar(page, 'a 2x4 lego') | ||||
|  | ||||
| @ -170,12 +136,16 @@ test.describe('Text-to-CAD tests', () => { | ||||
|     await expect(page.locator('.cm-content')).toContainText(``) | ||||
|   }) | ||||
|  | ||||
|   test('sending a bad prompt fails, can dismiss', async ({ page }) => { | ||||
|   test('sending a bad prompt fails, can dismiss', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     const commandBarButton = page.getByRole('button', { name: 'Commands' }) | ||||
|     await expect(commandBarButton).toBeVisible() | ||||
| @ -236,12 +206,14 @@ test.describe('Text-to-CAD tests', () => { | ||||
|  | ||||
|   test('sending a bad prompt fails, can start over from toast', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     const commandBarButton = page.getByRole('button', { name: 'Commands' }) | ||||
|     await expect(commandBarButton).toBeVisible() | ||||
| @ -324,12 +296,14 @@ test.describe('Text-to-CAD tests', () => { | ||||
|  | ||||
|   test('sending a bad prompt fails, can ignore toast, can start over from command bar', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     const commandBarButton = page.getByRole('button', { name: 'Commands' }) | ||||
|     await expect(commandBarButton).toBeVisible() | ||||
| @ -391,19 +365,21 @@ test.describe('Text-to-CAD tests', () => { | ||||
|  | ||||
|     await expect(successToastMessage).toBeVisible({ timeout: 15000 }) | ||||
|  | ||||
|     await expect(page.getByText('Copied')).not.toBeVisible() | ||||
|  | ||||
|     // old failure toast should stick around. | ||||
|     await expect(failureToastMessage).toBeVisible() | ||||
|     await expect(page.getByText(`Text-to-CAD failed`)).toBeVisible() | ||||
|   }) | ||||
|  | ||||
|   test('ensure you can shift+enter in the prompt box', async ({ page }) => { | ||||
|   test('ensure you can shift+enter in the prompt box', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     const promptWithNewline = `a 2x4\nlego` | ||||
|  | ||||
| @ -456,7 +432,7 @@ test.describe('Text-to-CAD tests', () => { | ||||
|   test( | ||||
|     'can do many at once and get many prompts back, and interact with many', | ||||
|     { tag: ['@skipWin'] }, | ||||
|     async ({ page }) => { | ||||
|     async ({ page, homePage }) => { | ||||
|       // Let this test run longer since we've seen it timeout. | ||||
|       test.setTimeout(180_000) | ||||
|       // skip on windows | ||||
| @ -467,9 +443,10 @@ test.describe('Text-to-CAD tests', () => { | ||||
|  | ||||
|       const u = await getUtils(page) | ||||
|  | ||||
|       await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|       await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await homePage.goToModelingScene() | ||||
|       await u.waitForPageLoad() | ||||
|  | ||||
|       await sendPromptFromCommandBar(page, 'a 2x4 lego') | ||||
|  | ||||
| @ -495,8 +472,6 @@ test.describe('Text-to-CAD tests', () => { | ||||
|       // We should have three success toasts. | ||||
|       await expect(successToastMessage).toHaveCount(3, { timeout: 25_000 }) | ||||
|  | ||||
|       await expect(page.getByText('Copied')).not.toBeVisible() | ||||
|  | ||||
|       await expect(page.getByText(`a 2x4 lego`)).toBeVisible() | ||||
|       await expect(page.getByText(`a 2x8 lego`)).toBeVisible() | ||||
|       await expect(page.getByText(`a 2x10 lego`)).toBeVisible() | ||||
| @ -514,31 +489,15 @@ test.describe('Text-to-CAD tests', () => { | ||||
|  | ||||
|       // Ensure you can copy the code for one of the models remaining. | ||||
|       const copyToClipboardButton = page.getByRole('button', { | ||||
|         name: 'Copy to clipboard', | ||||
|         name: 'Accept', | ||||
|       }) | ||||
|       await expect(copyToClipboardButton.first()).toBeVisible() | ||||
|       // Click the button. | ||||
|       await copyToClipboardButton.first().click() | ||||
|  | ||||
|       // Expect the code to be copied. | ||||
|       await expect(page.getByText('Copied')).toBeVisible() | ||||
|  | ||||
|       // Click in the code editor. | ||||
|       await page.locator('.cm-content').click({ position: { x: 10, y: 10 } }) | ||||
|  | ||||
|       // Paste the code. | ||||
|       await page.keyboard.down('ControlOrMeta') | ||||
|       await page.keyboard.press('KeyV') | ||||
|       await page.keyboard.up('ControlOrMeta') | ||||
|  | ||||
|       // Expect the code to be pasted. | ||||
|       await expect(page.locator('.cm-content')).toContainText(`2x8`) | ||||
|  | ||||
|       // Find the toast close button. | ||||
|       const closeButton = page.locator('[data-negative-button="close"]').first() | ||||
|       await expect(closeButton).toBeVisible() | ||||
|       await closeButton.click() | ||||
|  | ||||
|       // Ensure the final toast remains. | ||||
|       await expect(page.getByText(`a 2x10 lego`)).not.toBeVisible() | ||||
|       await expect(page.getByText(`Prompt: "a 2x8 lego`)).not.toBeVisible() | ||||
| @ -549,40 +508,21 @@ test.describe('Text-to-CAD tests', () => { | ||||
|       // Click the button. | ||||
|       await copyToClipboardButton.click() | ||||
|  | ||||
|       // Expect the code to be copied. | ||||
|       await expect(page.getByText('Copied')).toBeVisible() | ||||
|  | ||||
|       // Click in the code editor. | ||||
|       await page.locator('.cm-content').click({ position: { x: 10, y: 10 } }) | ||||
|  | ||||
|       // Paste the code. | ||||
|       await page.keyboard.down('ControlOrMeta') | ||||
|       await page.keyboard.press('KeyA') | ||||
|       await page.keyboard.up('ControlOrMeta') | ||||
|       await page.keyboard.press('Backspace') | ||||
|       await page.keyboard.down('ControlOrMeta') | ||||
|       await page.keyboard.press('KeyV') | ||||
|       await page.keyboard.up('ControlOrMeta') | ||||
|  | ||||
|       // Expect the code to be pasted. | ||||
|       await expect(page.locator('.cm-content')).toContainText(`2x4`) | ||||
|  | ||||
|       // Expect the toast to disappear. | ||||
|       // Find the toast close button. | ||||
|       await expect(closeButton).toBeVisible() | ||||
|       await closeButton.click() | ||||
|       await expect(successToastMessage).not.toBeVisible() | ||||
|     } | ||||
|   ) | ||||
|  | ||||
|   test('can do many at once with errors, clicking dismiss error does not dismiss all', async ({ | ||||
|     page, | ||||
|     homePage, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await homePage.goToModelingScene() | ||||
|     await u.waitForPageLoad() | ||||
|  | ||||
|     await sendPromptFromCommandBar(page, 'a 2x4 lego') | ||||
|  | ||||
| @ -631,57 +571,37 @@ test.describe('Text-to-CAD tests', () => { | ||||
|  | ||||
|     // Ensure you can copy the code for one of the models remaining. | ||||
|     const copyToClipboardButton = page.getByRole('button', { | ||||
|       name: 'Copy to clipboard', | ||||
|       name: 'Accept', | ||||
|     }) | ||||
|     await expect(copyToClipboardButton.first()).toBeVisible() | ||||
|     // Click the button. | ||||
|     await copyToClipboardButton.first().click() | ||||
|  | ||||
|     // Expect the code to be copied. | ||||
|     await expect(page.getByText('Copied')).toBeVisible() | ||||
|  | ||||
|     // Click in the code editor. | ||||
|     await page.locator('.cm-content').click({ position: { x: 10, y: 10 } }) | ||||
|  | ||||
|     // Paste the code. | ||||
|     await page.keyboard.down('ControlOrMeta') | ||||
|     await page.keyboard.press('KeyV') | ||||
|     await page.keyboard.up('ControlOrMeta') | ||||
|  | ||||
|     // Expect the code to be pasted. | ||||
|     await expect(page.locator('.cm-content')).toContainText(`2x4`) | ||||
|  | ||||
|     // Find the toast close button. | ||||
|     const closeButton = page | ||||
|       .getByRole('status') | ||||
|       .locator('div') | ||||
|       .filter({ hasText: 'Text-to-CAD successfulPrompt' }) | ||||
|       .first() | ||||
|       .getByRole('button', { name: 'Close' }) | ||||
|     await expect(closeButton).toBeVisible() | ||||
|     await closeButton.click() | ||||
|  | ||||
|     // Expect the toast to disappear. | ||||
|     await expect(page.getByText('Copied')).not.toBeVisible() | ||||
|     await expect(successToastMessage).not.toBeVisible() | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| async function sendPromptFromCommandBar(page: Page, promptStr: string) { | ||||
|   await page.waitForTimeout(1000) | ||||
|   await test.step(`Send prompt from command bar: ${promptStr}`, async () => { | ||||
|     const commandBarButton = page.getByRole('button', { name: 'Commands' }) | ||||
|     await expect(commandBarButton).toBeVisible() | ||||
|     // Click the command bar button | ||||
|     await commandBarButton.hover() | ||||
|     await commandBarButton.click() | ||||
|     await page.waitForTimeout(1000) | ||||
|  | ||||
|     // Wait for the command bar to appear | ||||
|     const cmdSearchBar = page.getByPlaceholder('Search commands') | ||||
|     await expect(cmdSearchBar).toBeVisible() | ||||
|  | ||||
|     const textToCadCommand = page.getByText('Use the Zoo Text-to-CAD API ') | ||||
|     const textToCadCommand = page.getByText('Use the Zoo Text-to-CAD API') | ||||
|     await expect(textToCadCommand.first()).toBeVisible() | ||||
|     // Click the Text-to-CAD command | ||||
|     await textToCadCommand.first().scrollIntoViewIfNeeded() | ||||
|     await textToCadCommand.first().click() | ||||
|     await page.waitForTimeout(1000) | ||||
|  | ||||
|     // Enter the prompt. | ||||
|     const prompt = page.getByText('Prompt') | ||||
| @ -697,12 +617,13 @@ async function sendPromptFromCommandBar(page: Page, promptStr: string) { | ||||
| test( | ||||
|   'Text-to-CAD functionality', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browserName }, testInfo) => { | ||||
|   async ({ context, page }, testInfo) => { | ||||
|     const projectName = 'project-000' | ||||
|     const prompt = 'lego 2x4' | ||||
|     const textToCadFileName = 'lego-2x4.kcl' | ||||
|  | ||||
|     const { electronApp, page, dir } = await setupElectron({ testInfo }) | ||||
|     const { dir } = await context.folderSetupFn(async () => {}) | ||||
|  | ||||
|     const fileExists = () => | ||||
|       fs.existsSync(join(dir, projectName, textToCadFileName)) | ||||
|  | ||||
| @ -711,7 +632,7 @@ test( | ||||
|       test | ||||
|     ) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|     // Locators | ||||
|     const projectMenuButton = page | ||||
| @ -761,7 +682,5 @@ test( | ||||
|       // Confirm we've navigated back to the main.kcl file after deletion | ||||
|       await expect(projectMenuButton).toContainText('main.kcl') | ||||
|     }) | ||||
|  | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
|  | ||||
| @ -1,19 +1,10 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { test, expect } from './zoo-test' | ||||
|  | ||||
| import { doExport, getUtils, makeTemplate, setup, tearDown } from './test-utils' | ||||
| import { doExport, getUtils, makeTemplate } from './test-utils' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }, testInfo) => { | ||||
|   await setup(context, page, testInfo) | ||||
| }) | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test('Units menu', async ({ page }) => { | ||||
|   const u = await getUtils(page) | ||||
|   await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|   await u.waitForAuthSkipAppStart() | ||||
| test.fixme('Units menu', async ({ page, homePage }) => { | ||||
|   await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|   await homePage.goToModelingScene() | ||||
|  | ||||
|   const unitsMenuButton = page.getByRole('button', { | ||||
|     name: 'Current Units', | ||||
| @ -41,7 +32,7 @@ test('Units menu', async ({ page }) => { | ||||
|   await expect(unitsMenuButton).toContainText('mm') | ||||
| }) | ||||
|  | ||||
| test('Successful export shows a success toast', async ({ page }) => { | ||||
| test('Successful export shows a success toast', async ({ page, homePage }) => { | ||||
|   // FYI this test doesn't work with only engine running locally | ||||
|   // And you will need to have the KittyCAD CLI installed | ||||
|   const u = await getUtils(page) | ||||
| @ -57,41 +48,41 @@ totalHeightHalf = 2 | ||||
| armThick = 0.5 | ||||
| totalLen = 9.5 | ||||
| part001 = startSketchOn('-XZ') | ||||
|   |> startProfileAt([0, 0], %) | ||||
|   |> yLine(baseHeight, %) | ||||
|   |> xLine(baseLen, %) | ||||
|   |> angledLineToY({ | ||||
|         angle: topAng, | ||||
|         to: totalHeightHalf, | ||||
|       }, %, $seg04) | ||||
|   |> xLineTo(totalLen, %, $seg03) | ||||
|   |> yLine(-armThick, %, $seg01) | ||||
|   |> angledLineThatIntersects({ | ||||
|         angle: HALF_TURN, | ||||
|         offset: -armThick, | ||||
|         intersectTag: seg04 | ||||
|       }, %) | ||||
|   |> angledLineToY([segAng(seg04) + 180, ZERO], %) | ||||
|   |> angledLineToY({ | ||||
|         angle: -bottomAng, | ||||
|         to: -totalHeightHalf - armThick, | ||||
|       }, %, $seg02) | ||||
|   |> xLineTo(segEndX(seg03) + 0, %) | ||||
|   |> yLine(-segLen(seg01), %) | ||||
|   |> angledLineThatIntersects({ | ||||
|         angle: HALF_TURN, | ||||
|         offset: -armThick, | ||||
|         intersectTag: seg02 | ||||
|       }, %) | ||||
|   |> angledLineToY([segAng(seg02) + 180, -baseHeight], %) | ||||
|   |> xLineTo(ZERO, %) | ||||
|   |> close(%) | ||||
|   |> extrude(4, %)` | ||||
| |> startProfileAt([0, 0], %) | ||||
| |> yLine(baseHeight, %) | ||||
| |> xLine(baseLen, %) | ||||
| |> angledLineToY({ | ||||
|       angle = topAng, | ||||
|       to = totalHeightHalf, | ||||
|     }, %, $seg04) | ||||
| |> xLineTo(totalLen, %, $seg03) | ||||
| |> yLine(-armThick, %, $seg01) | ||||
| |> angledLineThatIntersects({ | ||||
|       angle = HALF_TURN, | ||||
|       offset = -armThick, | ||||
|       intersectTag = seg04 | ||||
|     }, %) | ||||
| |> angledLineToY([segAng(seg04) + 180, ZERO], %) | ||||
| |> angledLineToY({ | ||||
|       angle = -bottomAng, | ||||
|       to = -totalHeightHalf - armThick, | ||||
|     }, %, $seg02) | ||||
| |> xLineTo(segEndX(seg03) + 0, %) | ||||
| |> yLine(-segLen(seg01), %) | ||||
| |> angledLineThatIntersects({ | ||||
|       angle = HALF_TURN, | ||||
|       offset = -armThick, | ||||
|       intersectTag = seg02 | ||||
|     }, %) | ||||
| |> angledLineToY([segAng(seg02) + 180, -baseHeight], %) | ||||
| |> xLineTo(ZERO, %) | ||||
| |> close(%) | ||||
| |> extrude(4, %)` | ||||
|     ) | ||||
|   }) | ||||
|   await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|   await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|   await u.waitForAuthSkipAppStart() | ||||
|   await homePage.goToModelingScene() | ||||
|   await u.openDebugPanel() | ||||
|   await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|   await u.waitForCmdReceive('extrude') | ||||
| @ -106,25 +97,14 @@ part001 = startSketchOn('-XZ') | ||||
|     }, | ||||
|     page | ||||
|   ) | ||||
|  | ||||
|   // This is the main thing we're testing, | ||||
|   // We test the export functionality across all | ||||
|   // file types in snapshot-tests.spec.ts | ||||
|   await expect(page.getByText('Exported successfully')).toBeVisible() | ||||
| }) | ||||
|  | ||||
| test('Paste should not work unless an input is focused', async ({ | ||||
|   page, | ||||
|   browserName, | ||||
|   homePage, | ||||
| }) => { | ||||
|   // To run this test locally, uncomment Firefox in playwright.config.ts | ||||
|   test.skip( | ||||
|     browserName !== 'firefox', | ||||
|     "This bug is really Firefox-only, which we don't run in CI." | ||||
|   ) | ||||
|   const u = await getUtils(page) | ||||
|   await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|   await u.waitForAuthSkipAppStart() | ||||
|   await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|   await homePage.goToModelingScene() | ||||
|   await page | ||||
|     .getByRole('button', { name: 'Start Sketch' }) | ||||
|     .waitFor({ state: 'visible' }) | ||||
| @ -164,12 +144,12 @@ test('Paste should not work unless an input is focused', async ({ | ||||
|  | ||||
| test('Keyboard shortcuts can be viewed through the help menu', async ({ | ||||
|   page, | ||||
|   homePage, | ||||
| }) => { | ||||
|   const u = await getUtils(page) | ||||
|   await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|   await u.waitForAuthSkipAppStart() | ||||
|   await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|   await homePage.goToModelingScene() | ||||
|  | ||||
|   await page.waitForURL('**/file/**', { waitUntil: 'domcontentloaded' }) | ||||
|   await page.waitForURL('file:///**', { waitUntil: 'domcontentloaded' }) | ||||
|   await page | ||||
|     .getByRole('button', { name: 'Start Sketch' }) | ||||
|     .waitFor({ state: 'visible' }) | ||||
| @ -181,7 +161,7 @@ test('Keyboard shortcuts can be viewed through the help menu', async ({ | ||||
|   await page.getByRole('button', { name: 'Keyboard Shortcuts' }).click() | ||||
|  | ||||
|   // Verify the URL and that you can see a list of shortcuts | ||||
|   await expect(page.url()).toContain('?tab=keybindings') | ||||
|   await expect.poll(() => page.url()).toContain('?tab=keybindings') | ||||
|   await expect( | ||||
|     page.getByRole('heading', { name: 'Enter Sketch Mode' }) | ||||
|   ).toBeAttached() | ||||
| @ -189,12 +169,13 @@ test('Keyboard shortcuts can be viewed through the help menu', async ({ | ||||
|  | ||||
| test('First escape in tool pops you out of tool, second exits sketch mode', async ({ | ||||
|   page, | ||||
|   homePage, | ||||
| }) => { | ||||
|   // Wait for the app to be ready for use | ||||
|   const u = await getUtils(page) | ||||
|   await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|   await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|   await u.waitForAuthSkipAppStart() | ||||
|   await homePage.goToModelingScene() | ||||
|   await u.openDebugPanel() | ||||
|   await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|   await u.closeDebugPanel() | ||||
| @ -258,7 +239,7 @@ test('First escape in tool pops you out of tool, second exits sketch mode', asyn | ||||
|  | ||||
| test.fixme( | ||||
|   'Basic default modeling and sketch hotkeys work', | ||||
|   async ({ page }) => { | ||||
|   async ({ page, homePage }) => { | ||||
|     const u = await getUtils(page) | ||||
|  | ||||
|     // This test can run long if it takes a little too long to load | ||||
| @ -285,8 +266,8 @@ test.fixme( | ||||
|           }) | ||||
|         ) | ||||
|       }) | ||||
|       await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|       await u.waitForAuthSkipAppStart() | ||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|       await homePage.goToModelingScene() | ||||
|       await u.openDebugPanel() | ||||
|       await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|       await u.closeDebugPanel() | ||||
| @ -437,10 +418,11 @@ test.fixme( | ||||
|   } | ||||
| ) | ||||
|  | ||||
| test('Delete key does not navigate back', async ({ page }) => { | ||||
|   await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|   await page.goto('/') | ||||
|   await page.waitForURL('**/file/**', { waitUntil: 'domcontentloaded' }) | ||||
| test('Delete key does not navigate back', async ({ page, homePage }) => { | ||||
|   await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|   await homePage.goToModelingScene() | ||||
|  | ||||
|   await page.waitForURL('file:///**', { waitUntil: 'domcontentloaded' }) | ||||
|  | ||||
|   const settingsButton = page.getByRole('link', { | ||||
|     name: 'Settings', | ||||
| @ -449,45 +431,45 @@ test('Delete key does not navigate back', async ({ page }) => { | ||||
|   const settingsCloseButton = page.getByTestId('settings-close-button') | ||||
|  | ||||
|   await settingsButton.click() | ||||
|   await expect(page.url()).toContain('/settings') | ||||
|   await expect.poll(() => page.url()).toContain('/settings') | ||||
|  | ||||
|   // Make sure that delete doesn't go back from settings | ||||
|   await page.keyboard.press('Delete') | ||||
|   await expect(page.url()).toContain('/settings') | ||||
|   await expect.poll(() => page.url()).toContain('/settings') | ||||
|  | ||||
|   // Now close the settings and try delete again, | ||||
|   // make sure it doesn't go back to settings | ||||
|   await settingsCloseButton.click() | ||||
|   await page.keyboard.press('Delete') | ||||
|   await expect(page.url()).not.toContain('/settings') | ||||
|   await expect.poll(() => page.url()).not.toContain('/settings') | ||||
| }) | ||||
|  | ||||
| test('Sketch on face', async ({ page }) => { | ||||
| test('Sketch on face', async ({ page, homePage }) => { | ||||
|   test.setTimeout(90_000) | ||||
|   const u = await getUtils(page) | ||||
|   await page.addInitScript(async () => { | ||||
|     localStorage.setItem( | ||||
|       'persistCode', | ||||
|       `sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([3.29, 7.86], %) | ||||
|   |> line([2.48, 2.44], %) | ||||
|   |> line([2.66, 1.17], %) | ||||
|   |> line([3.75, 0.46], %) | ||||
|   |> line([4.99, -0.46], %) | ||||
|   |> line([3.3, -2.12], %) | ||||
|   |> line([2.16, -3.33], %) | ||||
|   |> line([0.85, -3.08], %) | ||||
|   |> line([-0.18, -3.36], %) | ||||
|   |> line([-3.86, -2.73], %) | ||||
|   |> line([-17.67, 0.85], %) | ||||
|   |> close(%) | ||||
|   extrude001 = extrude(5 + 7, sketch001)` | ||||
| |> startProfileAt([3.29, 7.86], %) | ||||
| |> line([2.48, 2.44], %) | ||||
| |> line([2.66, 1.17], %) | ||||
| |> line([3.75, 0.46], %) | ||||
| |> line([4.99, -0.46], %) | ||||
| |> line([3.3, -2.12], %) | ||||
| |> line([2.16, -3.33], %) | ||||
| |> line([0.85, -3.08], %) | ||||
| |> line([-0.18, -3.36], %) | ||||
| |> line([-3.86, -2.73], %) | ||||
| |> line([-17.67, 0.85], %) | ||||
| |> close(%) | ||||
| extrude001 = extrude(5 + 7, sketch001)` | ||||
|     ) | ||||
|   }) | ||||
|  | ||||
|   await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|   await page.setBodyDimensions({ width: 1200, height: 500 }) | ||||
|  | ||||
|   await u.waitForAuthSkipAppStart() | ||||
|   await homePage.goToModelingScene() | ||||
|  | ||||
|   // wait for execution done | ||||
|   await u.openDebugPanel() | ||||
| @ -541,7 +523,8 @@ test('Sketch on face', async ({ page }) => { | ||||
|   |> line([2.45, -0.2], %) | ||||
|   |> line([-2.6, -1.25], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%)`) | ||||
|   |> close(%) | ||||
| `) | ||||
|   ) | ||||
|  | ||||
|   await u.openAndClearDebugPanel() | ||||
| @ -556,7 +539,7 @@ test('Sketch on face', async ({ page }) => { | ||||
|   await page.getByRole('button', { name: 'Edit Sketch' }).click() | ||||
|   await page.waitForTimeout(400) | ||||
|   await page.waitForTimeout(150) | ||||
|   await page.setViewportSize({ width: 1200, height: 1200 }) | ||||
|   await page.setBodyDimensions({ width: 1200, height: 1200 }) | ||||
|   await u.openAndClearDebugPanel() | ||||
|   await u.updateCamPosition([452, -152, 1166]) | ||||
|   await u.closeDebugPanel() | ||||
| @ -574,11 +557,11 @@ test('Sketch on face', async ({ page }) => { | ||||
|   previousCodeContent = await page.locator('.cm-content').innerText() | ||||
|  | ||||
|   const result = makeTemplate`sketch002 = startSketchOn(extrude001, seg01) | ||||
|   |> startProfileAt([-12.83, 6.7], %) | ||||
|   |> line([${[2.28, 2.35]}, -${0.07}], %) | ||||
|   |> line([-3.05, -1.47], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%)` | ||||
| |> startProfileAt([-12.83, 6.7], %) | ||||
| |> line([${[2.28, 2.35]}, -${0.07}], %) | ||||
| |> line([-3.05, -1.47], %) | ||||
| |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
| |> close(%)` | ||||
|  | ||||
|   await expect(page.locator('.cm-content')).toHaveText(result.regExp) | ||||
|  | ||||
| @ -602,6 +585,6 @@ test('Sketch on face', async ({ page }) => { | ||||
|   await page.getByRole('button', { name: 'checkmark Submit command' }).click() | ||||
|  | ||||
|   const result2 = result.genNext` | ||||
|   const sketch002 = extrude(${[5, 5]} + 7, sketch002)` | ||||
| const sketch002 = extrude(${[5, 5]} + 7, sketch002)` | ||||
|   await expect(page.locator('.cm-content')).toHaveText(result2.regExp) | ||||
| }) | ||||
|  | ||||
							
								
								
									
										334
									
								
								e2e/playwright/zoo-test.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,334 @@ | ||||
| import { | ||||
|   test as playwrightTestFn, | ||||
|   TestInfo as TestInfoPlaywright, | ||||
|   BrowserContext as BrowserContextPlaywright, | ||||
|   Page as PagePlaywright, | ||||
|   TestDetails as TestDetailsPlaywright, | ||||
|   PlaywrightTestArgs, | ||||
|   PlaywrightTestOptions, | ||||
|   PlaywrightWorkerArgs, | ||||
|   PlaywrightWorkerOptions, | ||||
|   ElectronApplication, | ||||
| } from '@playwright/test' | ||||
|  | ||||
| import { | ||||
|   fixtures, | ||||
|   Fixtures, | ||||
|   AuthenticatedTronApp, | ||||
|   AuthenticatedApp, | ||||
| } from './fixtures/fixtureSetup' | ||||
|  | ||||
| import { SaveSettingsPayload } from 'lib/settings/settingsTypes' | ||||
| export { expect } from '@playwright/test' | ||||
|  | ||||
| declare module '@playwright/test' { | ||||
|   interface TestInfo { | ||||
|     tronApp?: AuthenticatedTronApp | ||||
|   } | ||||
|   interface BrowserContext { | ||||
|     folderSetupFn: ( | ||||
|       cb: (dir: string) => Promise<void> | ||||
|     ) => Promise<{ dir: string }> | ||||
|   } | ||||
|   interface Page { | ||||
|     dir: string | ||||
|     TEST_SETTINGS_FILE_KEY?: string | ||||
|     setBodyDimensions: (dims: { | ||||
|       width: number | ||||
|       height: number | ||||
|     }) => Promise<void> | ||||
|   } | ||||
| } | ||||
|  | ||||
| export type TestInfo = TestInfoPlaywright | ||||
| export type BrowserContext = BrowserContextPlaywright | ||||
| export type Page = PagePlaywright | ||||
| export type TestDetails = TestDetailsPlaywright & { | ||||
|   cleanProjectDir?: boolean | ||||
|   appSettings?: Partial<SaveSettingsPayload> | ||||
| } | ||||
|  | ||||
| // Our custom decorated Zoo test object. Makes it easier to add fixtures, and | ||||
| // switch between web and electron if needed. | ||||
| const pwTestFnWithFixtures = playwrightTestFn.extend<Fixtures>(fixtures) | ||||
|  | ||||
| // In JavaScript you cannot replace a function's body only (despite functions | ||||
| // are themselves objects, which you'd expect a body property or something...) | ||||
| // So we must redefine the function and then re-attach properties. | ||||
| type PWFunction = ( | ||||
|   args: PlaywrightTestArgs & | ||||
|     Fixtures & | ||||
|     PlaywrightWorkerArgs & | ||||
|     PlaywrightTestOptions & | ||||
|     PlaywrightWorkerOptions & { | ||||
|       electronApp?: ElectronApplication | ||||
|     }, | ||||
|   testInfo: TestInfo | ||||
| ) => void | Promise<void> | ||||
|  | ||||
| let firstUrl = '' | ||||
|  | ||||
| // The below error is due to the extreme type spaghetti going on. playwright/ | ||||
| // types/test.d.ts does not export 2 functions (below is one of them) but tsc | ||||
| // is trying to use a interface name it can't see. | ||||
| // e2e/playwright/zoo-test.ts:64:14 - error TS4023: Exported variable 'test' has | ||||
| // or is using name 'TestFunction' from external module | ||||
| // "/home/lee/Code/Zoo/modeling-app/dirty2/node_modules/playwright/types/test" | ||||
| // but cannot be named. | ||||
| export const test = ( | ||||
|   desc: string, | ||||
|   objOrFn: PWFunction | TestDetails, | ||||
|   fnMaybe?: PWFunction | ||||
| ) => { | ||||
|   const hasTestConf = typeof objOrFn === 'object' | ||||
|   const fn = hasTestConf ? fnMaybe : objOrFn | ||||
|  | ||||
|   return pwTestFnWithFixtures( | ||||
|     desc, | ||||
|     hasTestConf ? objOrFn : {}, | ||||
|     async ( | ||||
|       { | ||||
|         page, | ||||
|         context, | ||||
|         cmdBar, | ||||
|         editor, | ||||
|         toolbar, | ||||
|         scene, | ||||
|         homePage, | ||||
|         request, | ||||
|         playwright, | ||||
|         browser, | ||||
|         acceptDownloads, | ||||
|         bypassCSP, | ||||
|         colorScheme, | ||||
|         clientCertificates, | ||||
|         deviceScaleFactor, | ||||
|         extraHTTPHeaders, | ||||
|         geolocation, | ||||
|         hasTouch, | ||||
|         httpCredentials, | ||||
|         ignoreHTTPSErrors, | ||||
|         isMobile, | ||||
|         javaScriptEnabled, | ||||
|         locale, | ||||
|         offline, | ||||
|         permissions, | ||||
|         proxy, | ||||
|         storageState, | ||||
|         timezoneId, | ||||
|         userAgent, | ||||
|         viewport, | ||||
|         baseURL, | ||||
|         contextOptions, | ||||
|         actionTimeout, | ||||
|         navigationTimeout, | ||||
|         serviceWorkers, | ||||
|         testIdAttribute, | ||||
|         browserName, | ||||
|         defaultBrowserType, | ||||
|         headless, | ||||
|         channel, | ||||
|         launchOptions, | ||||
|         connectOptions, | ||||
|         screenshot, | ||||
|         trace, | ||||
|         video, | ||||
|       }, | ||||
|       testInfo | ||||
|     ) => { | ||||
|       // To switch to web, use PLATFORM=web environment variable. | ||||
|       // Only use this for debugging, since the playwright tracer is busted | ||||
|       // for electron. | ||||
|  | ||||
|       let tronApp | ||||
|  | ||||
|       if (process.env.PLATFORM === 'web') { | ||||
|         tronApp = new AuthenticatedApp(context, page, testInfo) | ||||
|       } else { | ||||
|         tronApp = new AuthenticatedTronApp(context, page, testInfo) | ||||
|       } | ||||
|  | ||||
|       const fixtures: Fixtures = { cmdBar, editor, toolbar, scene, homePage } | ||||
|       if (tronApp instanceof AuthenticatedTronApp) { | ||||
|         const options = { | ||||
|           fixtures, | ||||
|         } | ||||
|         if (hasTestConf) { | ||||
|           Object.assign(options, { | ||||
|             appSettings: objOrFn?.appSettings, | ||||
|             cleanProjectDir: objOrFn?.cleanProjectDir, | ||||
|           }) | ||||
|         } | ||||
|         await tronApp.initialise(options) | ||||
|       } else { | ||||
|         await tronApp.initialise('') | ||||
|       } | ||||
|  | ||||
|       // We need to patch this because addInitScript will bind too late in our | ||||
|       // electron tests, never running. We need to call reload() after each call | ||||
|       // to guarantee it runs. | ||||
|       const oldContextAddInitScript = tronApp.context.addInitScript | ||||
|       tronApp.context.addInitScript = async function (a, b) { | ||||
|         // @ts-ignore pretty sure way out of tsc's type checking capabilities. | ||||
|         // This code works perfectly fine. | ||||
|         await oldContextAddInitScript.apply(this, [a, b]) | ||||
|         await tronApp.page.reload() | ||||
|       } | ||||
|  | ||||
|       // No idea why we mix and match page and context's addInitScript but we do | ||||
|       const oldPageAddInitScript = tronApp.page.addInitScript | ||||
|       tronApp.page.addInitScript = async function (a: any, b: any) { | ||||
|         // @ts-ignore pretty sure way out of tsc's type checking capabilities. | ||||
|         // This code works perfectly fine. | ||||
|         await oldPageAddInitScript.apply(this, [a, b]) | ||||
|         await tronApp.page.reload() | ||||
|       } | ||||
|  | ||||
|       // Create a consistent way to resize the page across electron and web. | ||||
|       // (lee) I had to do everyhting in the book to make electron change its | ||||
|       // damn window size. I succeded in making it consistently and reliably | ||||
|       // do it after a whole afternoon. | ||||
|       tronApp.page.setBodyDimensions = async function (dims: { | ||||
|         width: number | ||||
|         height: number | ||||
|       }) { | ||||
|         await tronApp.page.setViewportSize(dims) | ||||
|  | ||||
|         if (!(tronApp instanceof AuthenticatedTronApp)) { | ||||
|           return | ||||
|         } | ||||
|  | ||||
|         await tronApp.electronApp?.evaluateHandle(async ({ app }, dims) => { | ||||
|           // @ts-ignore sorry jon but see comment in main.ts why this is ignored | ||||
|           await app.resizeWindow(dims.width, dims.height) | ||||
|         }, dims) | ||||
|  | ||||
|         return tronApp.page.evaluate( | ||||
|           async (dims: { width: number; height: number }) => { | ||||
|             await window.electron.resizeWindow(dims.width, dims.height) | ||||
|             window.document.body.style.width = dims.width + 'px' | ||||
|             window.document.body.style.height = dims.height + 'px' | ||||
|             window.document.documentElement.style.width = dims.width + 'px' | ||||
|             window.document.documentElement.style.height = dims.height + 'px' | ||||
|           }, | ||||
|           dims | ||||
|         ) | ||||
|       } | ||||
|  | ||||
|       await tronApp.page.setBodyDimensions(tronApp.viewPortSize) | ||||
|  | ||||
|       // We need to expose this in order for some tests that require folder | ||||
|       // creation. Before they used to do this by their own electronSetup({...}) | ||||
|       // calls. | ||||
|       if (tronApp instanceof AuthenticatedTronApp) { | ||||
|         tronApp.context.folderSetupFn = async function (fn) { | ||||
|           return fn(tronApp.dir) | ||||
|             .then(() => tronApp.page.reload()) | ||||
|             .then(() => ({ | ||||
|               dir: tronApp.dir, | ||||
|             })) | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       if (!firstUrl) { | ||||
|         await tronApp.page.getByText('Your Projects').count() | ||||
|         firstUrl = tronApp.page.url() | ||||
|       } | ||||
|  | ||||
|       // Due to the app controlling its own window context we need to inject new | ||||
|       // options and context here. | ||||
|       // NOTE TO LEE: Seems to destroy page context when calling an electron loadURL. | ||||
|       // await tronApp.electronApp.evaluate(({ app }) => { | ||||
|       //   return app.reuseWindowForTest(); | ||||
|       // }); | ||||
|  | ||||
|       await tronApp.electronApp.evaluate(({ app }, projectDirName) => { | ||||
|         console.log('ABCDEFGHI', app.testProperty['TEST_SETTINGS_FILE_KEY']) | ||||
|         app.testProperty['TEST_SETTINGS_FILE_KEY'] = projectDirName | ||||
|       }, tronApp.dir) | ||||
|  | ||||
|       // Always start at the root view | ||||
|       await tronApp.page.goto(firstUrl) | ||||
|  | ||||
|       // Force a hard reload, destroying the stream and other state | ||||
|       await tronApp.page.reload() | ||||
|  | ||||
|       // tsc aint smart enough to know this'll never be undefined | ||||
|       // but I dont blame it, the logic to know is complex | ||||
|       if (fn) { | ||||
|         await fn( | ||||
|           { | ||||
|             context: tronApp.context, | ||||
|             page: tronApp.page, | ||||
|             electronApp: | ||||
|               tronApp instanceof AuthenticatedTronApp | ||||
|                 ? tronApp.electronApp | ||||
|                 : undefined, | ||||
|             ...fixtures, | ||||
|             request, | ||||
|             playwright, | ||||
|             browser, | ||||
|             acceptDownloads, | ||||
|             bypassCSP, | ||||
|             colorScheme, | ||||
|             clientCertificates, | ||||
|             deviceScaleFactor, | ||||
|             extraHTTPHeaders, | ||||
|             geolocation, | ||||
|             hasTouch, | ||||
|             httpCredentials, | ||||
|             ignoreHTTPSErrors, | ||||
|             isMobile, | ||||
|             javaScriptEnabled, | ||||
|             locale, | ||||
|             offline, | ||||
|             permissions, | ||||
|             proxy, | ||||
|             storageState, | ||||
|             timezoneId, | ||||
|             userAgent, | ||||
|             viewport, | ||||
|             baseURL, | ||||
|             contextOptions, | ||||
|             actionTimeout, | ||||
|             navigationTimeout, | ||||
|             serviceWorkers, | ||||
|             testIdAttribute, | ||||
|             browserName, | ||||
|             defaultBrowserType, | ||||
|             headless, | ||||
|             channel, | ||||
|             launchOptions, | ||||
|             connectOptions, | ||||
|             screenshot, | ||||
|             trace, | ||||
|             video, | ||||
|           }, | ||||
|           testInfo | ||||
|         ) | ||||
|       } | ||||
|  | ||||
|       testInfo.tronApp = | ||||
|         tronApp instanceof AuthenticatedTronApp ? tronApp : undefined | ||||
|     } | ||||
|   ) | ||||
| } | ||||
|  | ||||
| type ZooTest = typeof test | ||||
|  | ||||
| test.describe = pwTestFnWithFixtures.describe | ||||
| test.beforeEach = pwTestFnWithFixtures.beforeEach | ||||
| test.afterEach = pwTestFnWithFixtures.afterEach | ||||
| test.step = pwTestFnWithFixtures.step | ||||
| test.skip = pwTestFnWithFixtures.skip | ||||
| test.setTimeout = pwTestFnWithFixtures.setTimeout | ||||
| test.fixme = pwTestFnWithFixtures.fixme as unknown as ZooTest | ||||
| test.only = pwTestFnWithFixtures.only | ||||
| test.fail = pwTestFnWithFixtures.fail | ||||
| test.slow = pwTestFnWithFixtures.slow | ||||
| test.beforeAll = pwTestFnWithFixtures.beforeAll | ||||
| test.afterAll = pwTestFnWithFixtures.afterAll | ||||
| test.use = pwTestFnWithFixtures.use | ||||
| test.expect = pwTestFnWithFixtures.expect | ||||
| test.extend = pwTestFnWithFixtures.extend | ||||
| test.info = pwTestFnWithFixtures.info | ||||
| @ -1,20 +1,9 @@ | ||||
| import type { ForgeConfig } from '@electron-forge/shared-types' | ||||
| import { MakerSquirrel } from '@electron-forge/maker-squirrel' | ||||
| import { MakerZIP } from '@electron-forge/maker-zip' | ||||
| import { MakerDeb } from '@electron-forge/maker-deb' | ||||
| import { MakerRpm } from '@electron-forge/maker-rpm' | ||||
| import { VitePlugin } from '@electron-forge/plugin-vite' | ||||
| import { MakerWix, MakerWixConfig } from '@electron-forge/maker-wix' | ||||
| import { FusesPlugin } from '@electron-forge/plugin-fuses' | ||||
| import { FuseV1Options, FuseVersion } from '@electron/fuses' | ||||
| import path from 'path' | ||||
|  | ||||
| interface ExtendedMakerWixConfig extends MakerWixConfig { | ||||
|   // see https://github.com/electron/forge/issues/3673 | ||||
|   // this is an undocumented property of electron-wix-msi | ||||
|   associateExtensions?: string | ||||
| } | ||||
|  | ||||
| const rootDir = process.cwd() | ||||
|  | ||||
| const config: ForgeConfig = { | ||||
| @ -39,26 +28,7 @@ const config: ForgeConfig = { | ||||
|     extendInfo: 'Info.plist', // Information for file associations. | ||||
|   }, | ||||
|   rebuildConfig: {}, | ||||
|   makers: [ | ||||
|     new MakerSquirrel({ | ||||
|       setupIcon: path.resolve(rootDir, 'assets', 'icon.ico'), | ||||
|     }), | ||||
|     new MakerWix({ | ||||
|       icon: path.resolve(rootDir, 'assets', 'icon.ico'), | ||||
|       associateExtensions: 'kcl', | ||||
|     } as ExtendedMakerWixConfig), | ||||
|     new MakerZIP({}, ['darwin']), | ||||
|     new MakerRpm({ | ||||
|       options: { | ||||
|         icon: path.resolve(rootDir, 'assets', 'icon.png'), | ||||
|       }, | ||||
|     }), | ||||
|     new MakerDeb({ | ||||
|       options: { | ||||
|         icon: path.resolve(rootDir, 'assets', 'icon.png'), | ||||
|       }, | ||||
|     }), | ||||
|   ], | ||||
|   makers: [], | ||||
|   plugins: [ | ||||
|     new VitePlugin({ | ||||
|       // `build` can specify multiple entry builds, which can be Main process, Preload scripts, Worker process, etc. | ||||
|  | ||||
							
								
								
									
										1
									
								
								interface.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -7,6 +7,7 @@ import { MachinesListing } from 'components/MachineManagerProvider' | ||||
| type EnvFn = (value?: string) => string | ||||
|  | ||||
| export interface IElectronAPI { | ||||
|   resizeWindow: (width: number, height: number) => Promise<void> | ||||
|   open: typeof dialog.showOpenDialog | ||||
|   save: typeof dialog.showSaveDialog | ||||
|   openExternal: typeof shell.openExternal | ||||
|  | ||||
							
								
								
									
										55
									
								
								package.json
									
									
									
									
									
								
							
							
						
						| @ -39,7 +39,6 @@ | ||||
|     "chokidar": "^4.0.1", | ||||
|     "codemirror": "^6.0.1", | ||||
|     "decamelize": "^6.0.0", | ||||
|     "electron-squirrel-startup": "^1.0.1", | ||||
|     "electron-updater": "6.3.0", | ||||
|     "fuse.js": "^7.0.0", | ||||
|     "html2canvas-pro": "^1.5.8", | ||||
| @ -69,7 +68,7 @@ | ||||
|     "yargs": "^17.7.2" | ||||
|   }, | ||||
|   "scripts": { | ||||
|     "start": "vite", | ||||
|     "start": "vite --port=3000 --host=0.0.0.0", | ||||
|     "start:prod": "vite preview --port=3000", | ||||
|     "serve": "vite serve --port=3000", | ||||
|     "build": "curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y && source \"$HOME/.cargo/env\" && curl https://rustwasm.github.io/wasm-pack/installer/init.sh -sSf | sh -s -- -y && yarn build:wasm && vite build", | ||||
| @ -81,6 +80,7 @@ | ||||
|     "simpleserver": "yarn pretest && http-server ./public --cors -p 3000", | ||||
|     "simpleserver:ci": "yarn pretest && http-server ./public --cors -p 3000 &", | ||||
|     "simpleserver:bg": "yarn pretest && http-server ./public --cors -p 3000 &", | ||||
|     "simpleserver:stop": "kill-port 3000", | ||||
|     "fmt": "prettier --write ./src *.ts *.json *.js ./e2e ./packages", | ||||
|     "fmt-check": "prettier --check ./src *.ts *.json *.js ./e2e ./packages", | ||||
|     "fetch:wasm": "./get-latest-wasm-bundle.sh", | ||||
| @ -95,6 +95,8 @@ | ||||
|     "files:set-version": "echo \"$(jq --arg v \"$VERSION\" '.version=$v' package.json --indent 2)\" > package.json", | ||||
|     "files:set-notes": "./scripts/set-files-notes.sh", | ||||
|     "files:flip-to-nightly": "./scripts/flip-files-to-nightly.sh", | ||||
|     "files:invalidate-bucket": "./scripts/invalidate-files-bucket.sh", | ||||
|     "files:invalidate-bucket:nightly": "./scripts/invalidate-files-bucket.sh --nightly", | ||||
|     "postinstall": "yarn fetch:samples && yarn xstate:typegen && ./node_modules/.bin/electron-rebuild", | ||||
|     "xstate:typegen": "yarn xstate typegen \"src/**/*.ts?(x)\"", | ||||
|     "make:dev": "make dev", | ||||
| @ -103,24 +105,22 @@ | ||||
|     "tron:package": "electron-forge package", | ||||
|     "tron:make": "electron-forge make", | ||||
|     "tron:publish": "electron-forge publish", | ||||
|     "tron:test": "NODE_ENV=development yarn playwright test --config=playwright.electron.config.ts --grep=@electron", | ||||
|     "chrome:test": "PLATFORM=web NODE_ENV=development yarn playwright test --config=playwright.config.ts --project='Google Chrome' --grep-invert='@snapshot'", | ||||
|     "tron:test": "NODE_ENV=development yarn playwright test --config=playwright.electron.config.ts --grep-invert='@snapshot'", | ||||
|     "tronb:vite": "vite build -c vite.main.config.ts && vite build -c vite.preload.config.ts && vite build -c vite.renderer.config.ts", | ||||
|     "tronb:package": "electron-builder --config electron-builder.yml", | ||||
|     "test-setup": "yarn install && yarn build:wasm", | ||||
|     "test": "vitest --mode development", | ||||
|     "test:unit": "vitest run --mode development --exclude **/kclSamples.test.ts", | ||||
|     "test:unit:kcl-samples": "vitest run --mode development ./src/lang/kclSamples.test.ts", | ||||
|     "test:playwright:browser:chrome": "playwright test --project='Google Chrome' --config=playwright.ci.config.ts --grep-invert='@snapshot|@electron'", | ||||
|     "test:playwright:browser:chrome:windows": "playwright test --project=\"Google Chrome\" --config=playwright.ci.config.ts --grep-invert=\"@snapshot|@electron|@skipWin\"", | ||||
|     "test:playwright:browser:chrome:ubuntu": "playwright test --project='Google Chrome' --config=playwright.ci.config.ts --grep-invert='@snapshot|@electron|@skipLinux'", | ||||
|     "test:playwright:electron": "playwright test --config=playwright.electron.config.ts --grep=@electron", | ||||
|     "test:playwright:electron:windows": "playwright test --config=playwright.electron.config.ts --grep=@electron --grep-invert=@skipWin", | ||||
|     "test:playwright:electron:macos": "playwright test --config=playwright.electron.config.ts --grep=@electron --grep-invert=@skipMacos", | ||||
|     "test:playwright:electron:ubuntu": "playwright test --config=playwright.electron.config.ts --grep=@electron --grep-invert=@skipLinux", | ||||
|     "test:playwright:electron:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep=@electron", | ||||
|     "test:playwright:electron:windows:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep=@electron --grep-invert=@skipWin", | ||||
|     "test:playwright:electron:macos:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep=@electron --grep-invert=@skipMacos", | ||||
|     "test:playwright:electron:ubuntu:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep=@electron --grep-invert=@skipLinux", | ||||
|     "test:playwright:electron": "playwright test --config=playwright.electron.config.ts --grep-invert='@snapshot'", | ||||
|     "test:playwright:electron:windows": "playwright test --config=playwright.electron.config.ts --grep-invert='@skipWin|@snapshot'", | ||||
|     "test:playwright:electron:macos": "playwright test --config=playwright.electron.config.ts --grep-invert='@skipMacos|@snapshot'", | ||||
|     "test:playwright:electron:ubuntu": "playwright test --config=playwright.electron.config.ts --grep-invert='@skipLinux|@snapshot'", | ||||
|     "test:playwright:electron:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert='@snapshot'", | ||||
|     "test:playwright:electron:windows:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert='@skipWin|@snapshot'", | ||||
|     "test:playwright:electron:macos:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert='@skipMacos|@snapshot'", | ||||
|     "test:playwright:electron:ubuntu:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert='@skipLinux|@snapshot'", | ||||
|     "test:unit:local": "yarn simpleserver:bg && yarn test:unit; kill-port 3000", | ||||
|     "test:unit:kcl-samples:local": "yarn simpleserver:bg && yarn test:unit:kcl-samples; kill-port 3000" | ||||
|   }, | ||||
| @ -145,20 +145,14 @@ | ||||
|   "devDependencies": { | ||||
|     "@babel/plugin-proposal-private-property-in-object": "^7.21.11", | ||||
|     "@babel/preset-env": "^7.25.4", | ||||
|     "@electron-forge/cli": "^7.4.0", | ||||
|     "@electron-forge/maker-deb": "^7.4.0", | ||||
|     "@electron-forge/maker-rpm": "^7.4.0", | ||||
|     "@electron-forge/maker-squirrel": "^7.4.0", | ||||
|     "@electron-forge/maker-wix": "^7.5.0", | ||||
|     "@electron-forge/maker-zip": "^7.5.0", | ||||
|     "@electron-forge/plugin-auto-unpack-natives": "^7.4.0", | ||||
|     "@electron-forge/plugin-fuses": "^7.4.0", | ||||
|     "@electron-forge/plugin-vite": "^7.4.0", | ||||
|     "@electron/fuses": "^1.8.0", | ||||
|     "@electron/rebuild": "^3.6.0", | ||||
|     "@electron-forge/cli": "7.4.0", | ||||
|     "@electron-forge/plugin-fuses": "7.4.0", | ||||
|     "@electron-forge/plugin-vite": "7.4.0", | ||||
|     "@electron/fuses": "1.8.0", | ||||
|     "@iarna/toml": "^2.2.5", | ||||
|     "@lezer/generator": "^1.7.1", | ||||
|     "@playwright/test": "^1.46.1", | ||||
|     "@nabla/vite-plugin-eslint": "^2.0.5", | ||||
|     "@playwright/test": "^1.49.0", | ||||
|     "@testing-library/jest-dom": "^5.14.1", | ||||
|     "@testing-library/react": "^15.0.2", | ||||
|     "@types/d3-force": "^3.0.10", | ||||
| @ -170,7 +164,7 @@ | ||||
|     "@types/pixelmatch": "^5.2.6", | ||||
|     "@types/pngjs": "^6.0.4", | ||||
|     "@types/react": "^18.3.4", | ||||
|     "@types/react-dom": "^18.2.25", | ||||
|     "@types/react-dom": "^18.3.1", | ||||
|     "@types/react-modal": "^3.16.3", | ||||
|     "@types/three": "^0.163.0", | ||||
|     "@types/ua-parser-js": "^0.7.39", | ||||
| @ -184,9 +178,9 @@ | ||||
|     "@xstate/cli": "^0.5.17", | ||||
|     "autoprefixer": "^10.4.19", | ||||
|     "d3-force": "^3.0.0", | ||||
|     "electron": "^32.1.2", | ||||
|     "electron-builder": "^24.13.3", | ||||
|     "electron-notarize": "^1.2.2", | ||||
|     "electron": "32.1.2", | ||||
|     "electron-builder": "24.13.3", | ||||
|     "electron-notarize": "1.2.2", | ||||
|     "eslint": "^8.0.1", | ||||
|     "eslint-config-react-app": "^7.0.1", | ||||
|     "eslint-plugin-css-modules": "^2.12.0", | ||||
| @ -207,7 +201,6 @@ | ||||
|     "ts-node": "^10.0.0", | ||||
|     "typescript": "^5.7.2", | ||||
|     "vite": "^5.4.6", | ||||
|     "vite-plugin-eslint": "^1.8.1", | ||||
|     "vite-plugin-package-version": "^1.1.0", | ||||
|     "vite-tsconfig-paths": "^4.3.2", | ||||
|     "vitest": "^1.6.0", | ||||
|  | ||||
| @ -1,58 +0,0 @@ | ||||
| import { defineConfig, devices } from '@playwright/test' | ||||
|  | ||||
| /** | ||||
|  * See https://playwright.dev/docs/test-configuration. | ||||
|  */ | ||||
| export default defineConfig({ | ||||
|   timeout: 120_000, // override the default 30s timeout | ||||
|   testDir: './e2e/playwright', | ||||
|   /* Run tests in files in parallel */ | ||||
|   fullyParallel: true, | ||||
|   /* Fail the build on CI if you accidentally left test.only in the source code. */ | ||||
|   forbidOnly: true, | ||||
|   /* Do not retry */ | ||||
|   retries: 0, | ||||
|   /* Different amount of parallelism on CI and local. */ | ||||
|   workers: 1, | ||||
|   /* Reporter to use. See https://playwright.dev/docs/test-reporters */ | ||||
|   reporter: [ | ||||
|     ['dot'], | ||||
|     ['list'], | ||||
|     ['json', { outputFile: './test-results/report.json' }], | ||||
|     ['html'], | ||||
|   ], | ||||
|   /* Shared settings for all the projects below. See https://playwright.dev/docs/api/class-testoptions. */ | ||||
|   use: { | ||||
|     /* Base URL to use in actions like `await page.goto('/')`. */ | ||||
|     baseURL: 'http://localhost:3000', | ||||
|  | ||||
|     /* Collect trace when retrying the failed test. See https://playwright.dev/docs/trace-viewer */ | ||||
|     trace: 'retain-on-failure', | ||||
|     actionTimeout: 15_000, | ||||
|     screenshot: 'only-on-failure', | ||||
|   }, | ||||
|  | ||||
|   /* Configure projects for major browsers */ | ||||
|   projects: [ | ||||
|     { | ||||
|       name: 'Google Chrome', | ||||
|       use: { | ||||
|         ...devices['Desktop Chrome'], | ||||
|         channel: 'chrome', | ||||
|         contextOptions: { | ||||
|           /* Chromium is the only one with these permission types */ | ||||
|           permissions: ['clipboard-write', 'clipboard-read'], | ||||
|         }, | ||||
|       }, // or 'chrome-beta' | ||||
|     }, | ||||
|     { | ||||
|       name: 'webkit', | ||||
|       use: { ...devices['Desktop Safari'] }, | ||||
|     }, | ||||
|   ], | ||||
|  | ||||
|   webServer: { | ||||
|     command: 'yarn start', | ||||
|     reuseExistingServer: false, | ||||
|   }, | ||||
| }) | ||||
| @ -13,7 +13,7 @@ export default defineConfig({ | ||||
|   /* Do not retry */ | ||||
|   retries: 0, | ||||
|   /* Different amount of parallelism on CI and local. */ | ||||
|   workers: 1, | ||||
|   workers: 30, | ||||
|   /* Reporter to use. See https://playwright.dev/docs/test-reporters */ | ||||
|   reporter: [ | ||||
|     ['dot'], | ||||
|  | ||||
| @ -119,6 +119,11 @@ | ||||
|     "title": "Pipe and Flange Assembly", | ||||
|     "description": "A crucial component in various piping systems, designed to facilitate the connection, disconnection, and access to piping for inspection, cleaning, and modifications. This assembly combines pipes (long cylindrical conduits) with flanges (plate-like fittings) to create a secure yet detachable joint." | ||||
|   }, | ||||
|   { | ||||
|     "file": "pipe-with-bend.kcl", | ||||
|     "title": "Pipe with bend", | ||||
|     "description": "A tubular section or hollow cylinder, usually but not necessarily of circular cross-section, used mainly to convey substances that can flow." | ||||
|   }, | ||||
|   { | ||||
|     "file": "poopy-shoe.kcl", | ||||
|     "title": "Poopy Shoe", | ||||
|  | ||||
							
								
								
									
										11
									
								
								scripts/invalidate-files-bucket.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @ -0,0 +1,11 @@ | ||||
| #!/bin/bash | ||||
| base_dir="/releases/modeling-app" | ||||
| if [[ $1 = "--nightly" ]]; then | ||||
|     base_dir="/releases/modeling-app/nightly" | ||||
| fi | ||||
|  | ||||
| echo "Invalidating json and yml files at $base_dir in the download bucket" | ||||
| gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="$base_dir/last_download.json" --async | ||||
| gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="$base_dir/latest-linux-arm64.yml" --async | ||||
| gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="$base_dir/latest-mac.yml" --async | ||||
| gcloud compute url-maps invalidate-cdn-cache dl-url-map --path="$base_dir/latest.yml" --async | ||||
| @ -505,7 +505,8 @@ const ConstraintSymbol = ({ | ||||
|   constrainInfo: ConstrainInfo | ||||
|   verticalPosition: 'top' | 'bottom' | ||||
| }) => { | ||||
|   const { context, send } = useModelingContext() | ||||
|   const { commandBarSend } = useCommandsContext() | ||||
|   const { context } = useModelingContext() | ||||
|   const varNameMap: { | ||||
|     [key in ConstrainInfo['type']]: { | ||||
|       varName: string | ||||
| @ -624,11 +625,18 @@ const ConstraintSymbol = ({ | ||||
|         // disabled={implicitDesc} TODO why does this change styles that are hard to override? | ||||
|         onClick={toSync(async () => { | ||||
|           if (!isConstrained) { | ||||
|             send({ | ||||
|               type: 'Convert to variable', | ||||
|             commandBarSend({ | ||||
|               type: 'Find and select command', | ||||
|               data: { | ||||
|                 pathToNode, | ||||
|                 variableName: varName, | ||||
|                 name: 'Constrain with named value', | ||||
|                 groupId: 'modeling', | ||||
|                 argDefaultValues: { | ||||
|                   currentValue: { | ||||
|                     pathToNode, | ||||
|                     variableName: varName, | ||||
|                     valueText: value, | ||||
|                   }, | ||||
|                 }, | ||||
|               }, | ||||
|             }) | ||||
|           } else if (isConstrained) { | ||||
|  | ||||
| @ -701,8 +701,7 @@ export class SceneEntities { | ||||
|       'VariableDeclaration' | ||||
|     ) | ||||
|     if (trap(_node1)) return Promise.reject(_node1) | ||||
|     const variableDeclarationName = | ||||
|       _node1.node?.declarations?.[0]?.id?.name || '' | ||||
|     const variableDeclarationName = _node1.node?.declaration.id?.name || '' | ||||
|  | ||||
|     const sg = sketchFromKclValue( | ||||
|       kclManager.programMemory.get(variableDeclarationName), | ||||
| @ -902,10 +901,9 @@ export class SceneEntities { | ||||
|       'VariableDeclaration' | ||||
|     ) | ||||
|     if (trap(_node1)) return Promise.reject(_node1) | ||||
|     const variableDeclarationName = | ||||
|       _node1.node?.declarations?.[0]?.id?.name || '' | ||||
|     const startSketchOn = _node1.node?.declarations | ||||
|     const startSketchOnInit = startSketchOn?.[0]?.init | ||||
|     const variableDeclarationName = _node1.node?.declaration.id?.name || '' | ||||
|     const startSketchOn = _node1.node?.declaration | ||||
|     const startSketchOnInit = startSketchOn?.init | ||||
|  | ||||
|     const tags: [string, string, string] = [ | ||||
|       findUniqueName(_ast, 'rectangleSegmentA'), | ||||
| @ -913,7 +911,7 @@ export class SceneEntities { | ||||
|       findUniqueName(_ast, 'rectangleSegmentC'), | ||||
|     ] | ||||
|  | ||||
|     startSketchOn[0].init = createPipeExpression([ | ||||
|     startSketchOn.init = createPipeExpression([ | ||||
|       startSketchOnInit, | ||||
|       ...getRectangleCallExpressions(rectangleOrigin, tags), | ||||
|     ]) | ||||
| @ -943,7 +941,7 @@ export class SceneEntities { | ||||
|           'VariableDeclaration' | ||||
|         ) | ||||
|         if (trap(_node)) return Promise.reject(_node) | ||||
|         const sketchInit = _node.node?.declarations?.[0]?.init | ||||
|         const sketchInit = _node.node?.declaration.init | ||||
|  | ||||
|         const x = (args.intersectionPoint.twoD.x || 0) - rectangleOrigin[0] | ||||
|         const y = (args.intersectionPoint.twoD.y || 0) - rectangleOrigin[1] | ||||
| @ -992,7 +990,7 @@ export class SceneEntities { | ||||
|           'VariableDeclaration' | ||||
|         ) | ||||
|         if (trap(_node)) return | ||||
|         const sketchInit = _node.node?.declarations?.[0]?.init | ||||
|         const sketchInit = _node.node?.declaration.init | ||||
|  | ||||
|         if (sketchInit.type !== 'PipeExpression') { | ||||
|           return | ||||
| @ -1058,10 +1056,9 @@ export class SceneEntities { | ||||
|     if (trap(_node1)) return Promise.reject(_node1) | ||||
|  | ||||
|     // startSketchOn already exists | ||||
|     const variableDeclarationName = | ||||
|       _node1.node?.declarations?.[0]?.id?.name || '' | ||||
|     const startSketchOn = _node1.node?.declarations | ||||
|     const startSketchOnInit = startSketchOn?.[0]?.init | ||||
|     const variableDeclarationName = _node1.node?.declaration.id?.name || '' | ||||
|     const startSketchOn = _node1.node?.declaration | ||||
|     const startSketchOnInit = startSketchOn?.init | ||||
|  | ||||
|     const tags: [string, string, string] = [ | ||||
|       findUniqueName(_ast, 'rectangleSegmentA'), | ||||
| @ -1069,7 +1066,7 @@ export class SceneEntities { | ||||
|       findUniqueName(_ast, 'rectangleSegmentC'), | ||||
|     ] | ||||
|  | ||||
|     startSketchOn[0].init = createPipeExpression([ | ||||
|     startSketchOn.init = createPipeExpression([ | ||||
|       startSketchOnInit, | ||||
|       ...getRectangleCallExpressions(rectangleOrigin, tags), | ||||
|     ]) | ||||
| @ -1099,7 +1096,7 @@ export class SceneEntities { | ||||
|           'VariableDeclaration' | ||||
|         ) | ||||
|         if (trap(_node)) return Promise.reject(_node) | ||||
|         const sketchInit = _node.node?.declarations?.[0]?.init | ||||
|         const sketchInit = _node.node?.declaration.init | ||||
|  | ||||
|         const x = (args.intersectionPoint.twoD.x || 0) - rectangleOrigin[0] | ||||
|         const y = (args.intersectionPoint.twoD.y || 0) - rectangleOrigin[1] | ||||
| @ -1155,7 +1152,7 @@ export class SceneEntities { | ||||
|           'VariableDeclaration' | ||||
|         ) | ||||
|         if (trap(_node)) return | ||||
|         const sketchInit = _node.node?.declarations?.[0]?.init | ||||
|         const sketchInit = _node.node?.declaration.init | ||||
|  | ||||
|         if (sketchInit.type === 'PipeExpression') { | ||||
|           updateCenterRectangleSketch( | ||||
| @ -1224,12 +1221,11 @@ export class SceneEntities { | ||||
|       'VariableDeclaration' | ||||
|     ) | ||||
|     if (trap(_node1)) return Promise.reject(_node1) | ||||
|     const variableDeclarationName = | ||||
|       _node1.node?.declarations?.[0]?.id?.name || '' | ||||
|     const startSketchOn = _node1.node?.declarations | ||||
|     const startSketchOnInit = startSketchOn?.[0]?.init | ||||
|     const variableDeclarationName = _node1.node?.declaration.id?.name || '' | ||||
|     const startSketchOn = _node1.node?.declaration | ||||
|     const startSketchOnInit = startSketchOn?.init | ||||
|  | ||||
|     startSketchOn[0].init = createPipeExpression([ | ||||
|     startSketchOn.init = createPipeExpression([ | ||||
|       startSketchOnInit, | ||||
|       createCallExpressionStdLib('circle', [ | ||||
|         createObjectExpression({ | ||||
| @ -1271,7 +1267,7 @@ export class SceneEntities { | ||||
|         ) | ||||
|         let modded = structuredClone(truncatedAst) | ||||
|         if (trap(_node)) return | ||||
|         const sketchInit = _node.node?.declarations?.[0]?.init | ||||
|         const sketchInit = _node.node.declaration.init | ||||
|  | ||||
|         const x = (args.intersectionPoint.twoD.x || 0) - circleCenter[0] | ||||
|         const y = (args.intersectionPoint.twoD.y || 0) - circleCenter[1] | ||||
| @ -1339,7 +1335,7 @@ export class SceneEntities { | ||||
|           'VariableDeclaration' | ||||
|         ) | ||||
|         if (trap(_node)) return | ||||
|         const sketchInit = _node.node?.declarations?.[0]?.init | ||||
|         const sketchInit = _node.node?.declaration.init | ||||
|  | ||||
|         let modded = structuredClone(_ast) | ||||
|         if (sketchInit.type === 'PipeExpression') { | ||||
| @ -2060,7 +2056,7 @@ function prepareTruncatedMemoryAndAst( | ||||
|     'VariableDeclaration' | ||||
|   ) | ||||
|   if (err(_node)) return _node | ||||
|   const variableDeclarationName = _node.node?.declarations?.[0]?.id?.name || '' | ||||
|   const variableDeclarationName = _node.node?.declaration.id?.name || '' | ||||
|   const sg = sketchFromKclValue( | ||||
|     programMemory.get(variableDeclarationName), | ||||
|     variableDeclarationName | ||||
| @ -2085,7 +2081,7 @@ function prepareTruncatedMemoryAndAst( | ||||
|       ]) | ||||
|     } | ||||
|     ;( | ||||
|       (_ast.body[bodyIndex] as VariableDeclaration).declarations[0] | ||||
|       (_ast.body[bodyIndex] as VariableDeclaration).declaration | ||||
|         .init as PipeExpression | ||||
|     ).body.push(newSegment) | ||||
|     // update source ranges to section we just added. | ||||
| @ -2096,19 +2092,19 @@ function prepareTruncatedMemoryAndAst( | ||||
|     const updatedSrcRangeAst = pResult.program | ||||
|  | ||||
|     const lastPipeItem = ( | ||||
|       (updatedSrcRangeAst.body[bodyIndex] as VariableDeclaration) | ||||
|         .declarations[0].init as PipeExpression | ||||
|       (updatedSrcRangeAst.body[bodyIndex] as VariableDeclaration).declaration | ||||
|         .init as PipeExpression | ||||
|     ).body.slice(-1)[0] | ||||
|  | ||||
|     ;( | ||||
|       (_ast.body[bodyIndex] as VariableDeclaration).declarations[0] | ||||
|       (_ast.body[bodyIndex] as VariableDeclaration).declaration | ||||
|         .init as PipeExpression | ||||
|     ).body.slice(-1)[0].start = lastPipeItem.start | ||||
|  | ||||
|     _ast.end = lastPipeItem.end | ||||
|     const varDec = _ast.body[bodyIndex] as Node<VariableDeclaration> | ||||
|     varDec.end = lastPipeItem.end | ||||
|     const declarator = varDec.declarations[0] | ||||
|     const declarator = varDec.declaration | ||||
|     declarator.end = lastPipeItem.end | ||||
|     const init = declarator.init as Node<PipeExpression> | ||||
|     init.end = lastPipeItem.end | ||||
| @ -2145,7 +2141,7 @@ function prepareTruncatedMemoryAndAst( | ||||
|     if (node.type !== 'VariableDeclaration') { | ||||
|       continue | ||||
|     } | ||||
|     const name = node.declarations[0].id.name | ||||
|     const name = node.declaration.id.name | ||||
|     const memoryItem = programMemory.get(name) | ||||
|     if (!memoryItem) { | ||||
|       continue | ||||
|  | ||||
| @ -169,11 +169,11 @@ export function useCalc({ | ||||
|         const resultDeclaration = ast.body.find( | ||||
|           (a) => | ||||
|             a.type === 'VariableDeclaration' && | ||||
|             a.declarations?.[0]?.id?.name === '__result__' | ||||
|             a.declaration.id?.name === '__result__' | ||||
|         ) | ||||
|         const init = | ||||
|           resultDeclaration?.type === 'VariableDeclaration' && | ||||
|           resultDeclaration?.declarations?.[0]?.init | ||||
|           resultDeclaration?.declaration.init | ||||
|         const result = execState.memory?.get('__result__')?.value | ||||
|         setCalcResult(typeof result === 'number' ? String(result) : 'NAN') | ||||
|         init && setValueNode(init) | ||||
|  | ||||
| @ -8,11 +8,16 @@ import { getSystemTheme } from 'lib/theme' | ||||
| import { useCalculateKclExpression } from 'lib/useCalculateKclExpression' | ||||
| import { roundOff } from 'lib/utils' | ||||
| import { varMentions } from 'lib/varCompletionExtension' | ||||
| import { useEffect, useRef, useState } from 'react' | ||||
| import { useEffect, useMemo, useRef, useState } from 'react' | ||||
| import { useHotkeys } from 'react-hotkeys-hook' | ||||
| import styles from './CommandBarKclInput.module.css' | ||||
| import { createIdentifier, createVariableDeclaration } from 'lang/modifyAst' | ||||
| import { useCodeMirror } from 'components/ModelingSidebar/ModelingPanes/CodeEditor' | ||||
| import { useSelector } from '@xstate/react' | ||||
|  | ||||
| const machineContextSelector = (snapshot?: { | ||||
|   context: Record<string, unknown> | ||||
| }) => snapshot?.context | ||||
|  | ||||
| function CommandBarKclInput({ | ||||
|   arg, | ||||
| @ -31,12 +36,44 @@ function CommandBarKclInput({ | ||||
|     arg.name | ||||
|   ] as KclCommandValue | undefined | ||||
|   const { settings } = useSettingsAuthContext() | ||||
|   const defaultValue = (arg.defaultValue as string) || '' | ||||
|   const argMachineContext = useSelector( | ||||
|     arg.machineActor, | ||||
|     machineContextSelector | ||||
|   ) | ||||
|   const defaultValue = useMemo( | ||||
|     () => | ||||
|       arg.defaultValue | ||||
|         ? arg.defaultValue instanceof Function | ||||
|           ? arg.defaultValue(commandBarState.context, argMachineContext) | ||||
|           : arg.defaultValue | ||||
|         : '', | ||||
|     [arg.defaultValue, commandBarState.context, argMachineContext] | ||||
|   ) | ||||
|   const initialVariableName = useMemo(() => { | ||||
|     // Use the configured variable name if it exists | ||||
|     if (arg.variableName !== undefined) { | ||||
|       return arg.variableName instanceof Function | ||||
|         ? arg.variableName(commandBarState.context, argMachineContext) | ||||
|         : arg.variableName | ||||
|     } | ||||
|     // or derive it from the previously set value or the argument name | ||||
|     return previouslySetValue && 'variableName' in previouslySetValue | ||||
|       ? previouslySetValue.variableName | ||||
|       : arg.name | ||||
|   }, [ | ||||
|     arg.variableName, | ||||
|     commandBarState.context, | ||||
|     argMachineContext, | ||||
|     arg.name, | ||||
|     previouslySetValue, | ||||
|   ]) | ||||
|   const [value, setValue] = useState( | ||||
|     previouslySetValue?.valueText || defaultValue || '' | ||||
|   ) | ||||
|   const [createNewVariable, setCreateNewVariable] = useState( | ||||
|     previouslySetValue && 'variableName' in previouslySetValue | ||||
|     (previouslySetValue && 'variableName' in previouslySetValue) || | ||||
|       arg.createVariableByDefault || | ||||
|       false | ||||
|   ) | ||||
|   const [canSubmit, setCanSubmit] = useState(true) | ||||
|   useHotkeys('mod + k, mod + /', () => commandBarSend({ type: 'Close' })) | ||||
| @ -52,10 +89,7 @@ function CommandBarKclInput({ | ||||
|     isNewVariableNameUnique, | ||||
|   } = useCalculateKclExpression({ | ||||
|     value, | ||||
|     initialVariableName: | ||||
|       previouslySetValue && 'variableName' in previouslySetValue | ||||
|         ? previouslySetValue.variableName | ||||
|         : arg.name, | ||||
|     initialVariableName, | ||||
|   }) | ||||
|   const varMentionData: Completion[] = prevVariables.map((v) => ({ | ||||
|     label: v.key, | ||||
|  | ||||
| @ -266,6 +266,7 @@ const FileTreeItem = ({ | ||||
|       // Let the lsp servers know we closed a file. | ||||
|       onFileClose(currentFile?.path || null, project?.path || null) | ||||
|       onFileOpen(fileOrDir.path, project?.path || null) | ||||
|       kclManager.switchedFiles = true | ||||
|  | ||||
|       // Open kcl files | ||||
|       navigate(`${PATHS.FILE}/${encodeURIComponent(fileOrDir.path)}`) | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { APP_VERSION } from 'routes/Settings' | ||||
| import { APP_VERSION, getReleaseUrl } from 'routes/Settings' | ||||
| import { CustomIcon } from 'components/CustomIcon' | ||||
| import Tooltip from 'components/Tooltip' | ||||
| import { PATHS } from 'lib/paths' | ||||
| @ -72,10 +72,8 @@ export function LowerRightControls({ | ||||
|       <menu className="flex items-center justify-end gap-3 pointer-events-auto"> | ||||
|         {!location.pathname.startsWith(PATHS.HOME) && <ModelStateIndicator />} | ||||
|         <a | ||||
|           onClick={openExternalBrowserIfDesktop( | ||||
|             `https://github.com/KittyCAD/modeling-app/releases/tag/v${APP_VERSION}` | ||||
|           )} | ||||
|           href={`https://github.com/KittyCAD/modeling-app/releases/tag/v${APP_VERSION}`} | ||||
|           onClick={openExternalBrowserIfDesktop(getReleaseUrl())} | ||||
|           href={getReleaseUrl()} | ||||
|           target="_blank" | ||||
|           rel="noopener noreferrer" | ||||
|           className={'!no-underline font-mono text-xs ' + linkOverrideClassName} | ||||
|  | ||||
| @ -69,14 +69,7 @@ export const LspProvider = ({ children }: { children: React.ReactNode }) => { | ||||
|   const [isKclLspReady, setIsKclLspReady] = useState(false) | ||||
|   const [isCopilotLspReady, setIsCopilotLspReady] = useState(false) | ||||
|  | ||||
|   const { | ||||
|     auth, | ||||
|     settings: { | ||||
|       context: { | ||||
|         modeling: { defaultUnit }, | ||||
|       }, | ||||
|     }, | ||||
|   } = useSettingsAuthContext() | ||||
|   const { auth } = useSettingsAuthContext() | ||||
|   const token = auth?.context.token | ||||
|   const navigate = useNavigate() | ||||
|  | ||||
| @ -92,7 +85,6 @@ export const LspProvider = ({ children }: { children: React.ReactNode }) => { | ||||
|     const initEvent: KclWorkerOptions = { | ||||
|       wasmUrl: wasmUrl(), | ||||
|       token: token, | ||||
|       baseUnit: defaultUnit.current, | ||||
|       apiBaseUrl: VITE_KC_API_BASE_URL, | ||||
|     } | ||||
|     lspWorker.postMessage({ | ||||
|  | ||||
| @ -41,7 +41,10 @@ import { | ||||
|   angleBetweenInfo, | ||||
|   applyConstraintAngleBetween, | ||||
| } from './Toolbar/SetAngleBetween' | ||||
| import { applyConstraintAngleLength } from './Toolbar/setAngleLength' | ||||
| import { | ||||
|   applyConstraintAngleLength, | ||||
|   applyConstraintLength, | ||||
| } from './Toolbar/setAngleLength' | ||||
| import { | ||||
|   canSweepSelection, | ||||
|   handleSelectionBatch, | ||||
| @ -51,6 +54,8 @@ import { | ||||
|   Selections, | ||||
|   updateSelections, | ||||
|   canLoftSelection, | ||||
|   canRevolveSelection, | ||||
|   canShellSelection, | ||||
| } from 'lib/selections' | ||||
| import { applyConstraintIntersect } from './Toolbar/Intersect' | ||||
| import { applyConstraintAbsDistance } from './Toolbar/SetAbsDistance' | ||||
| @ -62,13 +67,15 @@ import { | ||||
|   getSketchOrientationDetails, | ||||
| } from 'clientSideScene/sceneEntities' | ||||
| import { | ||||
|   moveValueIntoNewVariablePath, | ||||
|   insertNamedConstant, | ||||
|   replaceValueAtNodePath, | ||||
|   sketchOnExtrudedFace, | ||||
|   sketchOnOffsetPlane, | ||||
|   startSketchOnDefault, | ||||
| } from 'lang/modifyAst' | ||||
| import { Program, parse, recast, resultIsOk } from 'lang/wasm' | ||||
| import { PathToNode, Program, parse, recast, resultIsOk } from 'lang/wasm' | ||||
| import { | ||||
|   doesSceneHaveExtrudedSketch, | ||||
|   doesSceneHaveSweepableSketch, | ||||
|   getNodePathFromSourceRange, | ||||
|   isSingleCursorInPipe, | ||||
| @ -79,7 +86,6 @@ import toast from 'react-hot-toast' | ||||
| import { EditorSelection, Transaction } from '@codemirror/state' | ||||
| import { useLoaderData, useNavigate, useSearchParams } from 'react-router-dom' | ||||
| import { letEngineAnimateAndSyncCamAfter } from 'clientSideScene/CameraControls' | ||||
| import { getVarNameModal } from 'hooks/useToolbarGuards' | ||||
| import { err, reportRejection, trap } from 'lib/trap' | ||||
| import { useCommandsContext } from 'hooks/useCommandsContext' | ||||
| import { modelingMachineEvent } from 'editor/manager' | ||||
| @ -570,6 +576,26 @@ export const ModelingMachineProvider = ({ | ||||
|           if (err(canSweep)) return false | ||||
|           return canSweep | ||||
|         }, | ||||
|         'has valid revolve selection': ({ context: { selectionRanges } }) => { | ||||
|           // A user can begin extruding if they either have 1+ faces selected or nothing selected | ||||
|           // TODO: I believe this guard only allows for extruding a single face at a time | ||||
|           const hasNoSelection = | ||||
|             selectionRanges.graphSelections.length === 0 || | ||||
|             isRangeBetweenCharacters(selectionRanges) || | ||||
|             isSelectionLastLine(selectionRanges, codeManager.code) | ||||
|  | ||||
|           if (hasNoSelection) { | ||||
|             // they have no selection, we should enable the button | ||||
|             // so they can select the face through the cmdbar | ||||
|             // BUT only if there's extrudable geometry | ||||
|             return doesSceneHaveSweepableSketch(kclManager.ast) | ||||
|           } | ||||
|           if (!isSketchPipe(selectionRanges)) return false | ||||
|  | ||||
|           const canSweep = canRevolveSelection(selectionRanges) | ||||
|           if (err(canSweep)) return false | ||||
|           return canSweep | ||||
|         }, | ||||
|         'has valid loft selection': ({ context: { selectionRanges } }) => { | ||||
|           const hasNoSelection = | ||||
|             selectionRanges.graphSelections.length === 0 || | ||||
| @ -585,6 +611,24 @@ export const ModelingMachineProvider = ({ | ||||
|           if (err(canLoft)) return false | ||||
|           return canLoft | ||||
|         }, | ||||
|         'has valid shell selection': ({ | ||||
|           context: { selectionRanges }, | ||||
|           event, | ||||
|         }) => { | ||||
|           const hasNoSelection = | ||||
|             selectionRanges.graphSelections.length === 0 || | ||||
|             isRangeBetweenCharacters(selectionRanges) || | ||||
|             isSelectionLastLine(selectionRanges, codeManager.code) | ||||
|  | ||||
|           if (hasNoSelection) { | ||||
|             return doesSceneHaveExtrudedSketch(kclManager.ast) | ||||
|           } | ||||
|  | ||||
|           const canShell = canShellSelection(selectionRanges) | ||||
|           console.log('canShellSelection', canShellSelection(selectionRanges)) | ||||
|           if (err(canShell)) return false | ||||
|           return canShell | ||||
|         }, | ||||
|         'has valid selection for deletion': ({ | ||||
|           context: { selectionRanges }, | ||||
|         }) => { | ||||
| @ -869,12 +913,18 @@ export const ModelingMachineProvider = ({ | ||||
|             } | ||||
|           } | ||||
|         ), | ||||
|         'Get length info': fromPromise( | ||||
|           async ({ input: { selectionRanges, sketchDetails } }) => { | ||||
|             const { modifiedAst, pathToNodeMap } = | ||||
|               await applyConstraintAngleLength({ | ||||
|                 selectionRanges, | ||||
|               }) | ||||
|         astConstrainLength: fromPromise( | ||||
|           async ({ | ||||
|             input: { selectionRanges, sketchDetails, lengthValue }, | ||||
|           }) => { | ||||
|             if (!lengthValue) | ||||
|               return Promise.reject(new Error('No length value')) | ||||
|             const constraintResult = await applyConstraintLength({ | ||||
|               selectionRanges, | ||||
|               length: lengthValue, | ||||
|             }) | ||||
|             if (err(constraintResult)) return Promise.reject(constraintResult) | ||||
|             const { modifiedAst, pathToNodeMap } = constraintResult | ||||
|             const pResult = parse(recast(modifiedAst)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
| @ -1043,38 +1093,88 @@ export const ModelingMachineProvider = ({ | ||||
|             } | ||||
|           } | ||||
|         ), | ||||
|         'Get convert to variable info': fromPromise( | ||||
|         'Apply named value constraint': fromPromise( | ||||
|           async ({ input: { selectionRanges, sketchDetails, data } }) => { | ||||
|             if (!sketchDetails) | ||||
|             if (!sketchDetails) { | ||||
|               return Promise.reject(new Error('No sketch details')) | ||||
|             const { variableName } = await getVarNameModal({ | ||||
|               valueName: data?.variableName || 'var', | ||||
|             }) | ||||
|             } | ||||
|             if (!data) { | ||||
|               return Promise.reject(new Error('No data from command flow')) | ||||
|             } | ||||
|             let pResult = parse(recast(kclManager.ast)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
|             let parsed = pResult.program | ||||
|  | ||||
|             const { modifiedAst: _modifiedAst, pathToReplacedNode } = | ||||
|               moveValueIntoNewVariablePath( | ||||
|                 parsed, | ||||
|                 kclManager.programMemory, | ||||
|                 data?.pathToNode || [], | ||||
|                 variableName | ||||
|             let result: { | ||||
|               modifiedAst: Node<Program> | ||||
|               pathToReplaced: PathToNode | null | ||||
|             } = { | ||||
|               modifiedAst: parsed, | ||||
|               pathToReplaced: null, | ||||
|             } | ||||
|             // If the user provided a constant name, | ||||
|             // we need to insert the named constant | ||||
|             // and then replace the node with the constant's name. | ||||
|             if ('variableName' in data.namedValue) { | ||||
|               const astAfterReplacement = replaceValueAtNodePath({ | ||||
|                 ast: parsed, | ||||
|                 pathToNode: data.currentValue.pathToNode, | ||||
|                 newExpressionString: data.namedValue.variableName, | ||||
|               }) | ||||
|               if (trap(astAfterReplacement)) { | ||||
|                 return Promise.reject(astAfterReplacement) | ||||
|               } | ||||
|               const parseResultAfterInsertion = parse( | ||||
|                 recast( | ||||
|                   insertNamedConstant({ | ||||
|                     node: astAfterReplacement.modifiedAst, | ||||
|                     newExpression: data.namedValue, | ||||
|                   }) | ||||
|                 ) | ||||
|               ) | ||||
|             pResult = parse(recast(_modifiedAst)) | ||||
|               if ( | ||||
|                 trap(parseResultAfterInsertion) || | ||||
|                 !resultIsOk(parseResultAfterInsertion) | ||||
|               ) | ||||
|                 return Promise.reject(parseResultAfterInsertion) | ||||
|               result = { | ||||
|                 modifiedAst: parseResultAfterInsertion.program, | ||||
|                 pathToReplaced: astAfterReplacement.pathToReplaced, | ||||
|               } | ||||
|             } else if ('valueText' in data.namedValue) { | ||||
|               // If they didn't provide a constant name, | ||||
|               // just replace the node with the value. | ||||
|               const astAfterReplacement = replaceValueAtNodePath({ | ||||
|                 ast: parsed, | ||||
|                 pathToNode: data.currentValue.pathToNode, | ||||
|                 newExpressionString: data.namedValue.valueText, | ||||
|               }) | ||||
|               if (trap(astAfterReplacement)) { | ||||
|                 return Promise.reject(astAfterReplacement) | ||||
|               } | ||||
|               // The `replacer` function returns a pathToNode that assumes | ||||
|               // an identifier is also being inserted into the AST, creating an off-by-one error. | ||||
|               // This corrects that error, but TODO we should fix this upstream | ||||
|               // to avoid this kind of error in the future. | ||||
|               astAfterReplacement.pathToReplaced[1][0] = | ||||
|                 (astAfterReplacement.pathToReplaced[1][0] as number) - 1 | ||||
|               result = astAfterReplacement | ||||
|             } | ||||
|  | ||||
|             pResult = parse(recast(result.modifiedAst)) | ||||
|             if (trap(pResult) || !resultIsOk(pResult)) | ||||
|               return Promise.reject(new Error('Unexpected compilation error')) | ||||
|             parsed = pResult.program | ||||
|  | ||||
|             if (trap(parsed)) return Promise.reject(parsed) | ||||
|             parsed = parsed as Node<Program> | ||||
|             if (!pathToReplacedNode) | ||||
|             if (!result.pathToReplaced) | ||||
|               return Promise.reject(new Error('No path to replaced node')) | ||||
|  | ||||
|             const updatedAst = | ||||
|               await sceneEntitiesManager.updateAstAndRejigSketch( | ||||
|                 pathToReplacedNode || [], | ||||
|                 result.pathToReplaced || [], | ||||
|                 parsed, | ||||
|                 sketchDetails.zAxis, | ||||
|                 sketchDetails.yAxis, | ||||
| @ -1087,7 +1187,7 @@ export const ModelingMachineProvider = ({ | ||||
|             ) | ||||
|  | ||||
|             const selection = updateSelections( | ||||
|               { 0: pathToReplacedNode }, | ||||
|               { 0: result.pathToReplaced }, | ||||
|               selectionRanges, | ||||
|               updatedAst.newAst | ||||
|             ) | ||||
| @ -1095,7 +1195,7 @@ export const ModelingMachineProvider = ({ | ||||
|             return { | ||||
|               selectionType: 'completeSelection', | ||||
|               selection, | ||||
|               updatedPathToNode: pathToReplacedNode, | ||||
|               updatedPathToNode: result.pathToReplaced, | ||||
|             } | ||||
|           } | ||||
|         ), | ||||
|  | ||||
| @ -76,7 +76,7 @@ export const ModelingPane = ({ | ||||
|   return ( | ||||
|     <section | ||||
|       {...props} | ||||
|       title={title && typeof title === 'string' ? title : ''} | ||||
|       aria-label={title && typeof title === 'string' ? title : ''} | ||||
|       data-testid={detailsTestId} | ||||
|       id={id} | ||||
|       className={ | ||||
|  | ||||
| @ -40,7 +40,9 @@ export const KclEditorMenu = ({ children }: PropsWithChildren) => { | ||||
|         <Menu.Items className="absolute right-0 left-auto w-72 flex flex-col gap-1 divide-y divide-chalkboard-20 dark:divide-chalkboard-70 align-stretch px-0 py-1 bg-chalkboard-10 dark:bg-chalkboard-100 rounded-sm shadow-lg border border-solid border-chalkboard-20/50 dark:border-chalkboard-80/50"> | ||||
|           <Menu.Item> | ||||
|             <button | ||||
|               onClick={() => kclManager.format()} | ||||
|               onClick={() => { | ||||
|                 kclManager.format().catch(reportRejection) | ||||
|               }} | ||||
|               className={styles.button} | ||||
|             > | ||||
|               <span>Format code</span> | ||||
|  | ||||
| @ -10,7 +10,7 @@ import { APP_NAME } from 'lib/constants' | ||||
| import { useCommandsContext } from 'hooks/useCommandsContext' | ||||
| import { CustomIcon } from './CustomIcon' | ||||
| import { useLspContext } from './LspProvider' | ||||
| import { engineCommandManager } from 'lib/singletons' | ||||
| import { engineCommandManager, kclManager } from 'lib/singletons' | ||||
| import { MachineManagerContext } from 'components/MachineManagerProvider' | ||||
| import usePlatform from 'hooks/usePlatform' | ||||
| import { useAbsoluteFilePath } from 'hooks/useAbsoluteFilePath' | ||||
| @ -68,8 +68,7 @@ function AppLogoLink({ | ||||
|       data-testid="app-logo" | ||||
|       onClick={() => { | ||||
|         onProjectClose(file || null, project?.path || null, false) | ||||
|         // Clear the scene. | ||||
|         engineCommandManager.clearScene() | ||||
|         kclManager.switchedFiles = true | ||||
|       }} | ||||
|       to={PATHS.HOME} | ||||
|       className={wrapperClassName + ' hover:before:brightness-110'} | ||||
| @ -190,8 +189,7 @@ function ProjectMenuPopover({ | ||||
|           className: !isDesktop() ? 'hidden' : '', | ||||
|           onClick: () => { | ||||
|             onProjectClose(file || null, project?.path || null, true) | ||||
|             // Clear the scene. | ||||
|             engineCommandManager.clearScene() | ||||
|             kclManager.switchedFiles = true | ||||
|           }, | ||||
|         }, | ||||
|       ].filter( | ||||
|  | ||||
| @ -10,7 +10,7 @@ interface AllKeybindingsFieldsProps {} | ||||
|  | ||||
| export const AllKeybindingsFields = forwardRef( | ||||
|   ( | ||||
|     props: AllKeybindingsFieldsProps, | ||||
|     _props: AllKeybindingsFieldsProps, | ||||
|     scrollRef: ForwardedRef<HTMLDivElement> | ||||
|   ) => { | ||||
|     // This is how we will get the interaction map from the context | ||||
| @ -25,7 +25,7 @@ export const AllKeybindingsFields = forwardRef( | ||||
|             .map(([category, categoryItems]) => ( | ||||
|               <div className="flex flex-col gap-4 px-2 pr-4"> | ||||
|                 <h2 | ||||
|                   id={`category-${category}`} | ||||
|                   id={`category-${category.replaceAll(/\s/g, '-')}`} | ||||
|                   className="text-xl mt-6 first-of-type:mt-0 capitalize font-bold" | ||||
|                 > | ||||
|                   {category} | ||||
|  | ||||
| @ -13,7 +13,7 @@ import { isDesktop } from 'lib/isDesktop' | ||||
| import { ActionButton } from 'components/ActionButton' | ||||
| import { SettingsFieldInput } from './SettingsFieldInput' | ||||
| import toast from 'react-hot-toast' | ||||
| import { APP_VERSION, PACKAGE_NAME } from 'routes/Settings' | ||||
| import { APP_VERSION, IS_NIGHTLY, getReleaseUrl } from 'routes/Settings' | ||||
| import { PATHS } from 'lib/paths' | ||||
| import { | ||||
|   createAndOpenNewTutorialProject, | ||||
| @ -246,10 +246,8 @@ export const AllSettingsFields = forwardRef( | ||||
|                   to inject the version from package.json */} | ||||
|               App version {APP_VERSION}.{' '} | ||||
|               <a | ||||
|                 onClick={openExternalBrowserIfDesktop( | ||||
|                   `https://github.com/KittyCAD/modeling-app/releases/tag/v${APP_VERSION}` | ||||
|                 )} | ||||
|                 href={`https://github.com/KittyCAD/modeling-app/releases/tag/v${APP_VERSION}`} | ||||
|                 onClick={openExternalBrowserIfDesktop(getReleaseUrl())} | ||||
|                 href={getReleaseUrl()} | ||||
|                 target="_blank" | ||||
|                 rel="noopener noreferrer" | ||||
|               > | ||||
| @ -271,7 +269,7 @@ export const AllSettingsFields = forwardRef( | ||||
|               , and start a discussion if you don't see it! Your feedback will | ||||
|               help us prioritize what to build next. | ||||
|             </p> | ||||
|             {PACKAGE_NAME.indexOf('-nightly') === -1 && ( | ||||
|             {!IS_NIGHTLY && ( | ||||
|               <p className="max-w-2xl mt-6"> | ||||
|                 Want to experience the latest and (hopefully) greatest from our | ||||
|                 main development branch?{' '} | ||||
|  | ||||
| @ -19,7 +19,7 @@ export function KeybindingsSectionsList({ | ||||
|             key={category} | ||||
|             onClick={() => | ||||
|               scrollRef.current | ||||
|                 ?.querySelector(`#category-${category}`) | ||||
|                 ?.querySelector(`#category-${category.replaceAll(/\s/g, '-')}`) | ||||
|                 ?.scrollIntoView({ | ||||
|                   block: 'center', | ||||
|                   behavior: 'smooth', | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| import { trap } from 'lib/trap' | ||||
| import { useMachine } from '@xstate/react' | ||||
| import { useMachine, useSelector } from '@xstate/react' | ||||
| import { useNavigate, useRouteLoaderData, useLocation } from 'react-router-dom' | ||||
| import { PATHS, BROWSER_PATH } from 'lib/paths' | ||||
| import { authMachine, TOKEN_PERSIST_KEY } from '../machines/authMachine' | ||||
| @ -23,7 +23,6 @@ import { | ||||
|   engineCommandManager, | ||||
|   sceneEntitiesManager, | ||||
| } from 'lib/singletons' | ||||
| import { uuidv4 } from 'lib/utils' | ||||
| import { IndexLoaderData } from 'lib/types' | ||||
| import { settings } from 'lib/settings/initialSettings' | ||||
| import { | ||||
| @ -55,11 +54,15 @@ type SettingsAuthContextType = { | ||||
|   settings: MachineContext<typeof settingsMachine> | ||||
| } | ||||
|  | ||||
| // a little hacky for sure, open to changing it | ||||
| // this implies that we should only even have one instance of this provider mounted at any one time | ||||
| // but I think that's a safe assumption | ||||
| let settingsStateRef: ContextFrom<typeof settingsMachine> | undefined | ||||
| export const getSettingsState = () => settingsStateRef | ||||
| /** | ||||
|  * This variable is used to store the last snapshot of the settings context | ||||
|  * for use outside of React, such as in `wasm.ts`. It is updated every time | ||||
|  * the settings machine changes with `useSelector`. | ||||
|  * TODO: when we decouple XState from React, we can just subscribe to the actor directly from `wasm.ts` | ||||
|  */ | ||||
| export let lastSettingsContextSnapshot: | ||||
|   | ContextFrom<typeof settingsMachine> | ||||
|   | undefined | ||||
|  | ||||
| export const SettingsAuthContext = createContext({} as SettingsAuthContextType) | ||||
|  | ||||
| @ -129,27 +132,11 @@ export const SettingsAuthProviderBase = ({ | ||||
|             .setTheme(context.app.theme.current) | ||||
|             .catch(reportRejection) | ||||
|         }, | ||||
|         setEngineScaleGridVisibility: ({ context }) => { | ||||
|           engineCommandManager.setScaleGridVisibility( | ||||
|             context.modeling.showScaleGrid.current | ||||
|           ) | ||||
|         }, | ||||
|         setClientTheme: ({ context }) => { | ||||
|           const opposingTheme = getOppositeTheme(context.app.theme.current) | ||||
|           sceneInfra.theme = opposingTheme | ||||
|           sceneEntitiesManager.updateSegmentBaseColor(opposingTheme) | ||||
|         }, | ||||
|         setEngineEdges: ({ context }) => { | ||||
|           // eslint-disable-next-line @typescript-eslint/no-floating-promises | ||||
|           engineCommandManager.sendSceneCommand({ | ||||
|             cmd_id: uuidv4(), | ||||
|             type: 'modeling_cmd_req', | ||||
|             cmd: { | ||||
|               type: 'edge_lines_visible' as any, // TODO update kittycad.ts to get this new command type | ||||
|               hidden: !context.modeling.highlightEdges.current, | ||||
|             }, | ||||
|           }) | ||||
|         }, | ||||
|         toastSuccess: ({ event }) => { | ||||
|           if (!('data' in event)) return | ||||
|           const eventParts = event.type.replace(/^set./, '').split('.') as [ | ||||
| @ -175,17 +162,27 @@ export const SettingsAuthProviderBase = ({ | ||||
|         }, | ||||
|         'Execute AST': ({ context, event }) => { | ||||
|           try { | ||||
|             const relevantSetting = (s: typeof settings) => { | ||||
|               return ( | ||||
|                 s.modeling?.defaultUnit?.current !== | ||||
|                   context.modeling.defaultUnit.current || | ||||
|                 s.modeling.showScaleGrid.current !== | ||||
|                   context.modeling.showScaleGrid.current || | ||||
|                 s.modeling?.highlightEdges.current !== | ||||
|                   context.modeling.highlightEdges.current | ||||
|               ) | ||||
|             } | ||||
|  | ||||
|             const allSettingsIncludesUnitChange = | ||||
|               event.type === 'Set all settings' && | ||||
|               event.settings?.modeling?.defaultUnit?.current !== | ||||
|                 context.modeling.defaultUnit.current | ||||
|               relevantSetting(event.settings) | ||||
|             const resetSettingsIncludesUnitChange = | ||||
|               event.type === 'Reset settings' && | ||||
|               context.modeling.defaultUnit.current !== | ||||
|                 settings?.modeling?.defaultUnit?.default | ||||
|               event.type === 'Reset settings' && relevantSetting(settings) | ||||
|  | ||||
|             if ( | ||||
|               event.type === 'set.modeling.defaultUnit' || | ||||
|               event.type === 'set.modeling.showScaleGrid' || | ||||
|               event.type === 'set.modeling.highlightEdges' || | ||||
|               allSettingsIncludesUnitChange || | ||||
|               resetSettingsIncludesUnitChange | ||||
|             ) { | ||||
| @ -214,7 +211,10 @@ export const SettingsAuthProviderBase = ({ | ||||
|     }), | ||||
|     { input: loadedSettings } | ||||
|   ) | ||||
|   settingsStateRef = settingsState.context | ||||
|   // Any time the actor changes, update the settings state for external use | ||||
|   useSelector(settingsActor, (s) => { | ||||
|     lastSettingsContextSnapshot = s.context | ||||
|   }) | ||||
|  | ||||
|   useEffect(() => { | ||||
|     if (!isDesktop()) return | ||||
|  | ||||
| @ -2,6 +2,7 @@ import toast from 'react-hot-toast' | ||||
| import { ActionButton } from './ActionButton' | ||||
| import { openExternalBrowserIfDesktop } from 'lib/openWindow' | ||||
| import { Marked } from '@ts-stack/markdown' | ||||
| import { getReleaseUrl } from 'routes/Settings' | ||||
|  | ||||
| export function ToastUpdate({ | ||||
|   version, | ||||
| @ -32,10 +33,8 @@ export function ToastUpdate({ | ||||
|             A new update has downloaded and will be available next time you | ||||
|             start the app. You can view the release notes{' '} | ||||
|             <a | ||||
|               onClick={openExternalBrowserIfDesktop( | ||||
|                 `https://github.com/KittyCAD/modeling-app/releases/tag/v${version}` | ||||
|               )} | ||||
|               href={`https://github.com/KittyCAD/modeling-app/releases/tag/v${version}`} | ||||
|               onClick={openExternalBrowserIfDesktop(getReleaseUrl(version))} | ||||
|               href={getReleaseUrl(version)} | ||||
|               target="_blank" | ||||
|               rel="noreferrer" | ||||
|             > | ||||
|  | ||||
| @ -22,6 +22,7 @@ import { removeDoubleNegatives } from '../AvailableVarsHelpers' | ||||
| import { normaliseAngle } from '../../lib/utils' | ||||
| import { kclManager } from 'lib/singletons' | ||||
| import { err } from 'lib/trap' | ||||
| import { KclCommandValue } from 'lib/commandTypes' | ||||
|  | ||||
| const getModalInfo = createSetAngleLengthModal(SetAngleLengthModal) | ||||
|  | ||||
| @ -63,6 +64,57 @@ export function angleLengthInfo({ | ||||
|   return { enabled, transforms } | ||||
| } | ||||
|  | ||||
| export async function applyConstraintLength({ | ||||
|   length, | ||||
|   selectionRanges, | ||||
| }: { | ||||
|   length: KclCommandValue | ||||
|   selectionRanges: Selections | ||||
| }) { | ||||
|   const ast = kclManager.ast | ||||
|   const angleLength = angleLengthInfo({ selectionRanges }) | ||||
|   if (err(angleLength)) return angleLength | ||||
|   const { transforms } = angleLength | ||||
|  | ||||
|   let distanceExpression: Expr = length.valueAst | ||||
|  | ||||
|   /** | ||||
|    * To be "constrained", the value must be a binary expression, a named value, or a function call. | ||||
|    * If it has a variable name, we need to insert a variable declaration at the correct index. | ||||
|    */ | ||||
|   if ( | ||||
|     'variableName' in length && | ||||
|     length.variableName && | ||||
|     length.insertIndex !== undefined | ||||
|   ) { | ||||
|     const newBody = [...ast.body] | ||||
|     newBody.splice(length.insertIndex, 0, length.variableDeclarationAst) | ||||
|     ast.body = newBody | ||||
|     distanceExpression = createIdentifier(length.variableName) | ||||
|   } | ||||
|  | ||||
|   if (!isExprBinaryPart(distanceExpression)) { | ||||
|     return new Error('Invalid valueNode, is not a BinaryPart') | ||||
|   } | ||||
|  | ||||
|   const retval = transformAstSketchLines({ | ||||
|     ast, | ||||
|     selectionRanges, | ||||
|     transformInfos: transforms, | ||||
|     programMemory: kclManager.programMemory, | ||||
|     referenceSegName: '', | ||||
|     forceValueUsedInTransform: distanceExpression, | ||||
|   }) | ||||
|   if (err(retval)) return Promise.reject(retval) | ||||
|  | ||||
|   const { modifiedAst: _modifiedAst, pathToNodeMap } = retval | ||||
|  | ||||
|   return { | ||||
|     modifiedAst: _modifiedAst, | ||||
|     pathToNodeMap, | ||||
|   } | ||||
| } | ||||
|  | ||||
| export async function applyConstraintAngleLength({ | ||||
|   selectionRanges, | ||||
|   angleOrLength = 'setLength', | ||||
|  | ||||
| @ -41,7 +41,10 @@ export function UnitsMenu() { | ||||
|                       close() | ||||
|                     }} | ||||
|                   > | ||||
|                     {baseUnitLabels[unit]} | ||||
|                     <span className="flex-1">{baseUnitLabels[unit]}</span> | ||||
|                     {unit === settings.context.modeling.defaultUnit.current && ( | ||||
|                       <span className="text-chalkboard-60">current</span> | ||||
|                     )} | ||||
|                   </button> | ||||
|                 </li> | ||||
|               ))} | ||||
|  | ||||
| @ -1,7 +1,5 @@ | ||||
| import { LspWorkerEventType } from '@kittycad/codemirror-lsp-client' | ||||
|  | ||||
| import { UnitLength } from 'wasm-lib/kcl/bindings/UnitLength' | ||||
|  | ||||
| export enum LspWorker { | ||||
|   Kcl = 'kcl', | ||||
|   Copilot = 'copilot', | ||||
| @ -9,7 +7,6 @@ export enum LspWorker { | ||||
| export interface KclWorkerOptions { | ||||
|   wasmUrl: string | ||||
|   token: string | ||||
|   baseUnit: UnitLength | ||||
|   apiBaseUrl: string | ||||
| } | ||||
|  | ||||
|  | ||||
| @ -17,7 +17,6 @@ import { | ||||
|   KclWorkerOptions, | ||||
|   CopilotWorkerOptions, | ||||
| } from 'editor/plugins/lsp/types' | ||||
| import { EngineCommandManager } from 'lang/std/engineConnection' | ||||
| import { err, reportRejection } from 'lib/trap' | ||||
|  | ||||
| const intoServer: IntoServer = new IntoServer() | ||||
| @ -46,14 +45,12 @@ export async function copilotLspRun( | ||||
|  | ||||
| export async function kclLspRun( | ||||
|   config: ServerConfig, | ||||
|   engineCommandManager: EngineCommandManager | null, | ||||
|   token: string, | ||||
|   baseUnit: string, | ||||
|   baseUrl: string | ||||
| ) { | ||||
|   try { | ||||
|     console.log('start kcl lsp') | ||||
|     await kcl_lsp_run(config, engineCommandManager, baseUnit, token, baseUrl) | ||||
|     await kcl_lsp_run(config, null, undefined, token, baseUrl) | ||||
|   } catch (e: any) { | ||||
|     console.log('kcl lsp failed', e) | ||||
|     // We can't restart here because a moved value, we should do this another way. | ||||
| @ -82,13 +79,7 @@ onmessage = function (event: MessageEvent) { | ||||
|           switch (worker) { | ||||
|             case LspWorker.Kcl: | ||||
|               const kclData = eventData as KclWorkerOptions | ||||
|               await kclLspRun( | ||||
|                 config, | ||||
|                 null, | ||||
|                 kclData.token, | ||||
|                 kclData.baseUnit, | ||||
|                 kclData.apiBaseUrl | ||||
|               ) | ||||
|               await kclLspRun(config, kclData.token, kclData.apiBaseUrl) | ||||
|               break | ||||
|             case LspWorker.Copilot: | ||||
|               let copilotData = eventData as CopilotWorkerOptions | ||||
|  | ||||
| @ -2,7 +2,7 @@ import { useLayoutEffect, useEffect, useRef } from 'react' | ||||
| import { engineCommandManager, kclManager } from 'lib/singletons' | ||||
| import { deferExecution } from 'lib/utils' | ||||
| import { Themes } from 'lib/theme' | ||||
| import { makeDefaultPlanes, modifyGrid } from 'lang/wasm' | ||||
| import { makeDefaultPlanes } from 'lang/wasm' | ||||
| import { useModelingContext } from './useModelingContext' | ||||
| import { useNetworkContext } from 'hooks/useNetworkContext' | ||||
| import { useAppState, useAppStream } from 'AppState' | ||||
| @ -56,9 +56,6 @@ export function useSetupEngineManager( | ||||
|       makeDefaultPlanes: () => { | ||||
|         return makeDefaultPlanes(kclManager.engineCommandManager) | ||||
|       }, | ||||
|       modifyGrid: (hidden: boolean) => { | ||||
|         return modifyGrid(kclManager.engineCommandManager, hidden) | ||||
|       }, | ||||
|     }) | ||||
|     hasSetNonZeroDimensions.current = true | ||||
|   } | ||||
|  | ||||
| @ -24,6 +24,8 @@ export function useConvertToVariable(range?: SourceRange) { | ||||
|   }, [enable]) | ||||
|  | ||||
|   useEffect(() => { | ||||
|     // Return early if there are no selection ranges for whatever reason | ||||
|     if (!context.selectionRanges) return | ||||
|     const parsed = ast | ||||
|  | ||||
|     const meta = isNodeSafeToReplace( | ||||
|  | ||||
| @ -317,3 +317,8 @@ code { | ||||
| #code-mirror-override .cm-editor { | ||||
|   height: 100% !important; | ||||
| } | ||||
|  | ||||
| /* Can't use #code-mirror-override here as we're outside of this div */ | ||||
| .body-bg .cm-diagnosticAction { | ||||
|   @apply bg-primary; | ||||
| } | ||||
|  | ||||
| @ -12,6 +12,7 @@ import { EXECUTE_AST_INTERRUPT_ERROR_MESSAGE } from 'lib/constants' | ||||
|  | ||||
| import { | ||||
|   CallExpression, | ||||
|   clearSceneAndBustCache, | ||||
|   emptyExecState, | ||||
|   ExecState, | ||||
|   initPromise, | ||||
| @ -60,6 +61,7 @@ export class KclManager { | ||||
|   private _executeIsStale: ExecuteArgs | null = null | ||||
|   private _wasmInitFailed = true | ||||
|   private _hasErrors = false | ||||
|   private _switchedFiles = false | ||||
|  | ||||
|   engineCommandManager: EngineCommandManager | ||||
|  | ||||
| @ -79,6 +81,10 @@ export class KclManager { | ||||
|     this._astCallBack(ast) | ||||
|   } | ||||
|  | ||||
|   set switchedFiles(switchedFiles: boolean) { | ||||
|     this._switchedFiles = switchedFiles | ||||
|   } | ||||
|  | ||||
|   get programMemory() { | ||||
|     return this._programMemory | ||||
|   } | ||||
| @ -166,8 +172,12 @@ export class KclManager { | ||||
|     this.engineCommandManager = engineCommandManager | ||||
|  | ||||
|     // eslint-disable-next-line @typescript-eslint/no-floating-promises | ||||
|     this.ensureWasmInit().then(() => { | ||||
|       this.ast = this.safeParse(codeManager.code) || this.ast | ||||
|     this.ensureWasmInit().then(async () => { | ||||
|       await this.safeParse(codeManager.code).then((ast) => { | ||||
|         if (ast) { | ||||
|           this.ast = ast | ||||
|         } | ||||
|       }) | ||||
|     }) | ||||
|   } | ||||
|  | ||||
| @ -211,7 +221,25 @@ export class KclManager { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   safeParse(code: string): Node<Program> | null { | ||||
|   // (jess) I'm not in love with this, but it ensures we clear the scene and | ||||
|   // bust the cache on | ||||
|   // errors from parsing when opening new files. | ||||
|   // Why not just clear the cache on all parse errors, you ask? well its actually | ||||
|   // really nice to keep the cache on parse errors within the same file, and | ||||
|   // only bust on engine errors esp if they take a long time to execute and | ||||
|   // you hit the wrong key! | ||||
|   private async checkIfSwitchedFilesShouldClear() { | ||||
|     // If we were switching files and we hit an error on parse we need to bust | ||||
|     // the cache and clear the scene. | ||||
|     if (this._hasErrors && this._switchedFiles) { | ||||
|       await clearSceneAndBustCache(this.engineCommandManager) | ||||
|     } else if (this._switchedFiles) { | ||||
|       // Reset the switched files boolean. | ||||
|       this._switchedFiles = false | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   async safeParse(code: string): Promise<Node<Program> | null> { | ||||
|     const result = parse(code) | ||||
|     this.diagnostics = [] | ||||
|     this._hasErrors = false | ||||
| @ -220,6 +248,8 @@ export class KclManager { | ||||
|       const kclerror: KCLError = result as KCLError | ||||
|       this.diagnostics = kclErrorsToDiagnostics([kclerror]) | ||||
|       this._hasErrors = true | ||||
|  | ||||
|       await this.checkIfSwitchedFilesShouldClear() | ||||
|       return null | ||||
|     } | ||||
|  | ||||
| @ -228,6 +258,7 @@ export class KclManager { | ||||
|     if (result.errors.length > 0) { | ||||
|       this._hasErrors = true | ||||
|  | ||||
|       await this.checkIfSwitchedFilesShouldClear() | ||||
|       return null | ||||
|     } | ||||
|  | ||||
| @ -353,7 +384,7 @@ export class KclManager { | ||||
|       console.error(newCode) | ||||
|       return | ||||
|     } | ||||
|     const newAst = this.safeParse(newCode) | ||||
|     const newAst = await this.safeParse(newCode) | ||||
|     if (!newAst) { | ||||
|       this.clearAst() | ||||
|       return | ||||
| @ -408,7 +439,7 @@ export class KclManager { | ||||
|     }) | ||||
|   } | ||||
|   async executeCode(zoomToFit?: boolean): Promise<void> { | ||||
|     const ast = this.safeParse(codeManager.code) | ||||
|     const ast = await this.safeParse(codeManager.code) | ||||
|     if (!ast) { | ||||
|       this.clearAst() | ||||
|       return | ||||
| @ -416,9 +447,9 @@ export class KclManager { | ||||
|     this.ast = { ...ast } | ||||
|     return this.executeAst({ zoomToFit }) | ||||
|   } | ||||
|   format() { | ||||
|   async format() { | ||||
|     const originalCode = codeManager.code | ||||
|     const ast = this.safeParse(originalCode) | ||||
|     const ast = await this.safeParse(originalCode) | ||||
|     if (!ast) { | ||||
|       this.clearAst() | ||||
|       return | ||||
| @ -458,7 +489,7 @@ export class KclManager { | ||||
|     const newCode = recast(ast) | ||||
|     if (err(newCode)) return Promise.reject(newCode) | ||||
|  | ||||
|     const astWithUpdatedSource = this.safeParse(newCode) | ||||
|     const astWithUpdatedSource = await this.safeParse(newCode) | ||||
|     if (!astWithUpdatedSource) return Promise.reject(new Error('bad ast')) | ||||
|     let returnVal: Selections | undefined = undefined | ||||
|  | ||||
|  | ||||
| @ -45,7 +45,7 @@ export default class CodeManager { | ||||
|     } else if (storedCode === null) { | ||||
|       this.code = bracket | ||||
|     } else { | ||||
|       this.code = storedCode | ||||
|       this.code = storedCode || '' | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @ -57,6 +57,10 @@ export default class CodeManager { | ||||
|     return this._code | ||||
|   } | ||||
|  | ||||
|   localStoragePersistCode(): string { | ||||
|     return safeLSGetItem(PERSIST_CODE_KEY) || '' | ||||
|   } | ||||
|  | ||||
|   registerCallBacks({ setCode }: { setCode: (arg: string) => void }) { | ||||
|     this.#updateState = setCode | ||||
|   } | ||||
| @ -165,7 +169,7 @@ export default class CodeManager { | ||||
| } | ||||
|  | ||||
| function safeLSGetItem(key: string) { | ||||
|   if (typeof window === 'undefined') return null | ||||
|   if (typeof window === 'undefined') return | ||||
|   return localStorage?.getItem(key) | ||||
| } | ||||
|  | ||||
|  | ||||
| @ -60,8 +60,7 @@ const b1 = cube([0,0], 10)` | ||||
|     expect(nodePath).toEqual([ | ||||
|       ['body', ''], | ||||
|       [0, 'index'], | ||||
|       ['declarations', 'VariableDeclaration'], | ||||
|       [0, 'index'], | ||||
|       ['declaration', 'VariableDeclaration'], | ||||
|       ['init', ''], | ||||
|       ['params', 'FunctionExpression'], | ||||
|       [0, 'index'], | ||||
| @ -96,14 +95,12 @@ const b1 = cube([0,0], 10)` | ||||
|     expect(nodePath).toEqual([ | ||||
|       ['body', ''], | ||||
|       [0, 'index'], | ||||
|       ['declarations', 'VariableDeclaration'], | ||||
|       [0, 'index'], | ||||
|       ['declaration', 'VariableDeclaration'], | ||||
|       ['init', ''], | ||||
|       ['body', 'FunctionExpression'], | ||||
|       ['body', 'FunctionExpression'], | ||||
|       [0, 'index'], | ||||
|       ['declarations', 'VariableDeclaration'], | ||||
|       [0, 'index'], | ||||
|       ['declaration', 'VariableDeclaration'], | ||||
|       ['init', ''], | ||||
|       ['body', 'PipeExpression'], | ||||
|       [2, 'index'], | ||||
|  | ||||
| @ -82,11 +82,11 @@ describe('Testing createVariableDeclaration', () => { | ||||
|   it('should create a variable declaration', () => { | ||||
|     const result = createVariableDeclaration('myVar', createLiteral(5)) | ||||
|     expect(result.type).toBe('VariableDeclaration') | ||||
|     expect(result.declarations[0].type).toBe('VariableDeclarator') | ||||
|     expect(result.declarations[0].id.type).toBe('Identifier') | ||||
|     expect(result.declarations[0].id.name).toBe('myVar') | ||||
|     expect(result.declarations[0].init.type).toBe('Literal') | ||||
|     expect((result.declarations[0].init as any).value).toBe(5) | ||||
|     expect(result.declaration.type).toBe('VariableDeclarator') | ||||
|     expect(result.declaration.id.type).toBe('Identifier') | ||||
|     expect(result.declaration.id.name).toBe('myVar') | ||||
|     expect(result.declaration.init.type).toBe('Literal') | ||||
|     expect((result.declaration.init as any).value).toBe(5) | ||||
|   }) | ||||
| }) | ||||
| describe('Testing createPipeExpression', () => { | ||||
|  | ||||
| @ -45,6 +45,7 @@ import { TagDeclarator } from 'wasm-lib/kcl/bindings/TagDeclarator' | ||||
| import { Models } from '@kittycad/lib' | ||||
| import { ExtrudeFacePlane } from 'machines/modelingMachine' | ||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||
| import { KclExpressionWithVariable } from 'lib/commandTypes' | ||||
|  | ||||
| export function startSketchOnDefault( | ||||
|   node: Node<Program>, | ||||
| @ -66,8 +67,7 @@ export function startSketchOnDefault( | ||||
|   let pathToNode: PathToNode = [ | ||||
|     ['body', ''], | ||||
|     [sketchIndex, 'index'], | ||||
|     ['declarations', 'VariableDeclaration'], | ||||
|     ['0', 'index'], | ||||
|     ['declaration', 'VariableDeclaration'], | ||||
|     ['init', 'VariableDeclarator'], | ||||
|   ] | ||||
|  | ||||
| @ -94,7 +94,7 @@ export function addStartProfileAt( | ||||
|     return new Error('variableDeclaration.init.type !== PipeExpression') | ||||
|   } | ||||
|   const _node = { ...node } | ||||
|   const init = variableDeclaration.declarations[0].init | ||||
|   const init = variableDeclaration.declaration.init | ||||
|   const startProfileAt = createCallExpressionStdLib('startProfileAt', [ | ||||
|     createArrayExpression([ | ||||
|       createLiteral(roundOff(at[0])), | ||||
| @ -105,7 +105,7 @@ export function addStartProfileAt( | ||||
|   if (init.type === 'PipeExpression') { | ||||
|     init.body.splice(1, 0, startProfileAt) | ||||
|   } else { | ||||
|     variableDeclaration.declarations[0].init = createPipeExpression([ | ||||
|     variableDeclaration.declaration.init = createPipeExpression([ | ||||
|       init, | ||||
|       startProfileAt, | ||||
|     ]) | ||||
| @ -149,8 +149,7 @@ export function addSketchTo( | ||||
|   let pathToNode: PathToNode = [ | ||||
|     ['body', ''], | ||||
|     [sketchIndex, 'index'], | ||||
|     ['declarations', 'VariableDeclaration'], | ||||
|     ['0', 'index'], | ||||
|     ['declaration', 'VariableDeclaration'], | ||||
|     ['init', 'VariableDeclarator'], | ||||
|   ] | ||||
|   if (axis !== 'xy') { | ||||
| @ -333,8 +332,7 @@ export function extrudeSketch( | ||||
|   const pathToExtrudeArg: PathToNode = [ | ||||
|     ['body', ''], | ||||
|     [sketchIndexInBody + 1, 'index'], | ||||
|     ['declarations', 'VariableDeclaration'], | ||||
|     [0, 'index'], | ||||
|     ['declaration', 'VariableDeclaration'], | ||||
|     ['init', 'VariableDeclarator'], | ||||
|     ['arguments', 'CallExpression'], | ||||
|     [0, 'index'], | ||||
| @ -364,8 +362,7 @@ export function loftSketches( | ||||
|   const pathToNode: PathToNode = [ | ||||
|     ['body', ''], | ||||
|     [modifiedAst.body.length - 1, 'index'], | ||||
|     ['declarations', 'VariableDeclaration'], | ||||
|     ['0', 'index'], | ||||
|     ['declaration', 'VariableDeclaration'], | ||||
|     ['init', 'VariableDeclarator'], | ||||
|     ['arguments', 'CallExpression'], | ||||
|     [0, 'index'], | ||||
| @ -460,8 +457,7 @@ export function revolveSketch( | ||||
|   const pathToRevolveArg: PathToNode = [ | ||||
|     ['body', ''], | ||||
|     [sketchIndexInBody + 1, 'index'], | ||||
|     ['declarations', 'VariableDeclaration'], | ||||
|     [0, 'index'], | ||||
|     ['declaration', 'VariableDeclaration'], | ||||
|     ['init', 'VariableDeclarator'], | ||||
|     ['arguments', 'CallExpression'], | ||||
|     [0, 'index'], | ||||
| @ -547,8 +543,7 @@ export function sketchOnExtrudedFace( | ||||
|   const newpathToNode: PathToNode = [ | ||||
|     ['body', ''], | ||||
|     [expressionIndex + 1, 'index'], | ||||
|     ['declarations', 'VariableDeclaration'], | ||||
|     [0, 'index'], | ||||
|     ['declaration', 'VariableDeclaration'], | ||||
|     ['init', 'VariableDeclarator'], | ||||
|   ] | ||||
|  | ||||
| @ -585,8 +580,7 @@ export function addOffsetPlane({ | ||||
|   const pathToNode: PathToNode = [ | ||||
|     ['body', ''], | ||||
|     [modifiedAst.body.length - 1, 'index'], | ||||
|     ['declarations', 'VariableDeclaration'], | ||||
|     ['0', 'index'], | ||||
|     ['declaration', 'VariableDeclaration'], | ||||
|     ['init', 'VariableDeclarator'], | ||||
|     ['arguments', 'CallExpression'], | ||||
|     [0, 'index'], | ||||
| @ -597,6 +591,25 @@ export function addOffsetPlane({ | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Return a modified clone of an AST with a named constant inserted into the body | ||||
|  */ | ||||
| export function insertNamedConstant({ | ||||
|   node, | ||||
|   newExpression, | ||||
| }: { | ||||
|   node: Node<Program> | ||||
|   newExpression: KclExpressionWithVariable | ||||
| }): Node<Program> { | ||||
|   const ast = structuredClone(node) | ||||
|   ast.body.splice( | ||||
|     newExpression.insertIndex, | ||||
|     0, | ||||
|     newExpression.variableDeclarationAst | ||||
|   ) | ||||
|   return ast | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Modify the AST to create a new sketch using the variable declaration | ||||
|  * of an offset plane. The new sketch just has to come after the offset | ||||
| @ -823,17 +836,15 @@ export function createVariableDeclaration( | ||||
|     end: 0, | ||||
|     moduleId: 0, | ||||
|  | ||||
|     declarations: [ | ||||
|       { | ||||
|         type: 'VariableDeclarator', | ||||
|         start: 0, | ||||
|         end: 0, | ||||
|         moduleId: 0, | ||||
|     declaration: { | ||||
|       type: 'VariableDeclarator', | ||||
|       start: 0, | ||||
|       end: 0, | ||||
|       moduleId: 0, | ||||
|  | ||||
|         id: createIdentifier(varName), | ||||
|         init, | ||||
|       }, | ||||
|     ], | ||||
|       id: createIdentifier(varName), | ||||
|       init, | ||||
|     }, | ||||
|     visibility, | ||||
|     kind, | ||||
|   } | ||||
| @ -942,6 +953,31 @@ export function giveSketchFnCallTag( | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Replace a | ||||
|  */ | ||||
| export function replaceValueAtNodePath({ | ||||
|   ast, | ||||
|   pathToNode, | ||||
|   newExpressionString, | ||||
| }: { | ||||
|   ast: Node<Program> | ||||
|   pathToNode: PathToNode | ||||
|   newExpressionString: string | ||||
| }) { | ||||
|   const replaceCheckResult = isNodeSafeToReplacePath(ast, pathToNode) | ||||
|   if (err(replaceCheckResult)) { | ||||
|     return replaceCheckResult | ||||
|   } | ||||
|   const { isSafe, value, replacer } = replaceCheckResult | ||||
|  | ||||
|   if (!isSafe || value.type === 'Identifier') { | ||||
|     return new Error('Not safe to replace') | ||||
|   } | ||||
|  | ||||
|   return replacer(ast, newExpressionString) | ||||
| } | ||||
|  | ||||
| export function moveValueIntoNewVariablePath( | ||||
|   ast: Node<Program>, | ||||
|   programMemory: ProgramMemory, | ||||
| @ -1120,7 +1156,7 @@ export async function deleteFromSelection( | ||||
|     traverse(astClone, { | ||||
|       enter: (node, path) => { | ||||
|         if (node.type === 'VariableDeclaration') { | ||||
|           const dec = node.declarations[0] | ||||
|           const dec = node.declaration | ||||
|           if ( | ||||
|             dec.init.type === 'CallExpression' && | ||||
|             (dec.init.callee.name === 'extrude' || | ||||
| @ -1155,7 +1191,7 @@ export async function deleteFromSelection( | ||||
|             enter: (node, path) => { | ||||
|               ;(async () => { | ||||
|                 if (node.type === 'VariableDeclaration') { | ||||
|                   currentVariableName = node.declarations[0].id.name | ||||
|                   currentVariableName = node.declaration.id.name | ||||
|                 } | ||||
|                 if ( | ||||
|                   // match startSketchOn(${extrudeNameToDelete}) | ||||
|  | ||||
| @ -22,7 +22,7 @@ import { | ||||
| import { getNodeFromPath, getNodePathFromSourceRange } from '../queryAst' | ||||
| import { createLiteral } from 'lang/modifyAst' | ||||
| import { err } from 'lib/trap' | ||||
| import { Selections } from 'lib/selections' | ||||
| import { Selection, Selections } from 'lib/selections' | ||||
| import { engineCommandManager, kclManager } from 'lib/singletons' | ||||
| import { VITE_KC_DEV_TOKEN } from 'env' | ||||
| import { isOverlap } from 'lib/utils' | ||||
| @ -40,7 +40,6 @@ beforeAll(async () => { | ||||
|       makeDefaultPlanes: () => makeDefaultPlanes(engineCommandManager), | ||||
|       setMediaStream: () => {}, | ||||
|       setIsStreamReady: () => {}, | ||||
|       modifyGrid: async () => {}, | ||||
|       callbackOnEngineLiteConnect: () => { | ||||
|         resolve(true) | ||||
|       }, | ||||
| @ -118,13 +117,8 @@ const runGetPathToExtrudeForSegmentSelectionTest = async ( | ||||
|     code.indexOf(selectedSegmentSnippet) + selectedSegmentSnippet.length, | ||||
|     true, | ||||
|   ] | ||||
|   const selection: Selections = { | ||||
|     graphSelections: [ | ||||
|       { | ||||
|         codeRef: codeRefFromRange(segmentRange, ast), | ||||
|       }, | ||||
|     ], | ||||
|     otherSelections: [], | ||||
|   const selection: Selection = { | ||||
|     codeRef: codeRefFromRange(segmentRange, ast), | ||||
|   } | ||||
|  | ||||
|   // executeAst and artifactGraph | ||||
|  | ||||
| @ -29,7 +29,7 @@ import { | ||||
|   sketchLineHelperMap, | ||||
| } from '../std/sketch' | ||||
| import { err, trap } from 'lib/trap' | ||||
| import { Selections } from 'lib/selections' | ||||
| import { Selection, Selections } from 'lib/selections' | ||||
| import { KclCommandValue } from 'lib/commandTypes' | ||||
| import { | ||||
|   Artifact, | ||||
| @ -99,14 +99,9 @@ export function modifyAstWithEdgeTreatmentAndTag( | ||||
|   const lookupMap: Map<string, PathToNode> = new Map() // work around for Map key comparison | ||||
|  | ||||
|   for (const selection of selections.graphSelections) { | ||||
|     const singleSelection = { | ||||
|       graphSelections: [selection], | ||||
|       otherSelections: [], | ||||
|     } | ||||
|  | ||||
|     const result = getPathToExtrudeForSegmentSelection( | ||||
|       clonedAstForGetExtrude, | ||||
|       singleSelection, | ||||
|       selection, | ||||
|       artifactGraph | ||||
|     ) | ||||
|     if (err(result)) return result | ||||
| @ -259,12 +254,12 @@ function insertParametersIntoAst( | ||||
|  | ||||
| export function getPathToExtrudeForSegmentSelection( | ||||
|   ast: Program, | ||||
|   selection: Selections, | ||||
|   selection: Selection, | ||||
|   artifactGraph: ArtifactGraph | ||||
| ): { pathToSegmentNode: PathToNode; pathToExtrudeNode: PathToNode } | Error { | ||||
|   const pathToSegmentNode = getNodePathFromSourceRange( | ||||
|     ast, | ||||
|     selection.graphSelections[0]?.codeRef?.range | ||||
|     selection.codeRef?.range | ||||
|   ) | ||||
|  | ||||
|   const varDecNode = getNodeFromPath<VariableDeclaration>( | ||||
| @ -273,7 +268,7 @@ export function getPathToExtrudeForSegmentSelection( | ||||
|     'VariableDeclaration' | ||||
|   ) | ||||
|   if (err(varDecNode)) return varDecNode | ||||
|   const sketchVar = varDecNode.node.declarations[0].id.name | ||||
|   const sketchVar = varDecNode.node.declaration.id.name | ||||
|  | ||||
|   const sketch = sketchFromKclValue( | ||||
|     kclManager.programMemory.get(sketchVar), | ||||
| @ -308,7 +303,7 @@ async function updateAstAndFocus( | ||||
|   } | ||||
| } | ||||
|  | ||||
| function mutateAstWithTagForSketchSegment( | ||||
| export function mutateAstWithTagForSketchSegment( | ||||
|   astClone: Node<Program>, | ||||
|   pathToSegmentNode: PathToNode | ||||
| ): { modifiedAst: Program; tag: string } | Error { | ||||
| @ -340,7 +335,7 @@ function mutateAstWithTagForSketchSegment( | ||||
|   return { modifiedAst: astClone, tag } | ||||
| } | ||||
|  | ||||
| function getEdgeTagCall( | ||||
| export function getEdgeTagCall( | ||||
|   tag: string, | ||||
|   artifact: Artifact | ||||
| ): Node<Identifier | CallExpression> { | ||||
| @ -367,7 +362,7 @@ function locateExtrudeDeclarator( | ||||
|   if (err(nodeOfExtrudeCall)) return nodeOfExtrudeCall | ||||
|  | ||||
|   const { node: extrudeVarDecl } = nodeOfExtrudeCall | ||||
|   const extrudeDeclarator = extrudeVarDecl.declarations[0] | ||||
|   const extrudeDeclarator = extrudeVarDecl.declaration | ||||
|   if (!extrudeDeclarator) { | ||||
|     return new Error('Extrude Declarator not found.') | ||||
|   } | ||||
|  | ||||
							
								
								
									
										154
									
								
								src/lang/modifyAst/addRevolve.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,154 @@ | ||||
| import { err } from 'lib/trap' | ||||
| import { KCL_DEFAULT_CONSTANT_PREFIXES } from 'lib/constants' | ||||
| import { | ||||
|   Program, | ||||
|   PathToNode, | ||||
|   Expr, | ||||
|   CallExpression, | ||||
|   PipeExpression, | ||||
|   VariableDeclarator, | ||||
| } from 'lang/wasm' | ||||
| import { Selections } from 'lib/selections' | ||||
| import { Node } from 'wasm-lib/kcl/bindings/Node' | ||||
| import { | ||||
|   createLiteral, | ||||
|   createCallExpressionStdLib, | ||||
|   createObjectExpression, | ||||
|   createIdentifier, | ||||
|   createPipeExpression, | ||||
|   findUniqueName, | ||||
|   createVariableDeclaration, | ||||
| } from 'lang/modifyAst' | ||||
| import { getNodeFromPath, getNodePathFromSourceRange } from 'lang/queryAst' | ||||
| import { | ||||
|   mutateAstWithTagForSketchSegment, | ||||
|   getEdgeTagCall, | ||||
| } from 'lang/modifyAst/addEdgeTreatment' | ||||
| export function revolveSketch( | ||||
|   ast: Node<Program>, | ||||
|   pathToSketchNode: PathToNode, | ||||
|   shouldPipe = false, | ||||
|   angle: Expr = createLiteral(4), | ||||
|   axis: Selections | ||||
| ): | ||||
|   | { | ||||
|       modifiedAst: Node<Program> | ||||
|       pathToSketchNode: PathToNode | ||||
|       pathToRevolveArg: PathToNode | ||||
|     } | ||||
|   | Error { | ||||
|   const clonedAst = structuredClone(ast) | ||||
|   const sketchNode = getNodeFromPath(clonedAst, pathToSketchNode) | ||||
|   if (err(sketchNode)) return sketchNode | ||||
|  | ||||
|   // testing code | ||||
|   const pathToAxisSelection = getNodePathFromSourceRange( | ||||
|     clonedAst, | ||||
|     axis.graphSelections[0]?.codeRef.range | ||||
|   ) | ||||
|  | ||||
|   const lineNode = getNodeFromPath<CallExpression>( | ||||
|     clonedAst, | ||||
|     pathToAxisSelection, | ||||
|     'CallExpression' | ||||
|   ) | ||||
|   if (err(lineNode)) return lineNode | ||||
|  | ||||
|   // TODO Kevin: What if |> close(%)? | ||||
|   // TODO Kevin: What if opposite edge | ||||
|   // TODO Kevin: What if the edge isn't planar to the sketch? | ||||
|   // TODO Kevin: add a tag. | ||||
|   const tagResult = mutateAstWithTagForSketchSegment( | ||||
|     clonedAst, | ||||
|     pathToAxisSelection | ||||
|   ) | ||||
|  | ||||
|   // Have the tag whether it is already created or a new one is generated | ||||
|   if (err(tagResult)) return tagResult | ||||
|   const { tag } = tagResult | ||||
|  | ||||
|   /* Original Code */ | ||||
|   const { node: sketchExpression } = sketchNode | ||||
|  | ||||
|   // determine if sketchExpression is in a pipeExpression or not | ||||
|   const sketchPipeExpressionNode = getNodeFromPath<PipeExpression>( | ||||
|     clonedAst, | ||||
|     pathToSketchNode, | ||||
|     'PipeExpression' | ||||
|   ) | ||||
|   if (err(sketchPipeExpressionNode)) return sketchPipeExpressionNode | ||||
|   const { node: sketchPipeExpression } = sketchPipeExpressionNode | ||||
|   const isInPipeExpression = sketchPipeExpression.type === 'PipeExpression' | ||||
|  | ||||
|   const sketchVariableDeclaratorNode = getNodeFromPath<VariableDeclarator>( | ||||
|     clonedAst, | ||||
|     pathToSketchNode, | ||||
|     'VariableDeclarator' | ||||
|   ) | ||||
|   if (err(sketchVariableDeclaratorNode)) return sketchVariableDeclaratorNode | ||||
|   const { | ||||
|     node: sketchVariableDeclarator, | ||||
|     shallowPath: sketchPathToDecleration, | ||||
|   } = sketchVariableDeclaratorNode | ||||
|  | ||||
|   const axisSelection = axis?.graphSelections[0]?.artifact | ||||
|  | ||||
|   if (!axisSelection) return new Error('Axis selection is missing.') | ||||
|  | ||||
|   const revolveCall = createCallExpressionStdLib('revolve', [ | ||||
|     createObjectExpression({ | ||||
|       angle: angle, | ||||
|       axis: getEdgeTagCall(tag, axisSelection), | ||||
|     }), | ||||
|     createIdentifier(sketchVariableDeclarator.id.name), | ||||
|   ]) | ||||
|  | ||||
|   if (shouldPipe) { | ||||
|     const pipeChain = createPipeExpression( | ||||
|       isInPipeExpression | ||||
|         ? [...sketchPipeExpression.body, revolveCall] | ||||
|         : [sketchExpression as any, revolveCall] | ||||
|     ) | ||||
|  | ||||
|     sketchVariableDeclarator.init = pipeChain | ||||
|     const pathToRevolveArg: PathToNode = [ | ||||
|       ...sketchPathToDecleration, | ||||
|       ['init', 'VariableDeclarator'], | ||||
|       ['body', ''], | ||||
|       [pipeChain.body.length - 1, 'index'], | ||||
|       ['arguments', 'CallExpression'], | ||||
|       [0, 'index'], | ||||
|     ] | ||||
|  | ||||
|     return { | ||||
|       modifiedAst: clonedAst, | ||||
|       pathToSketchNode, | ||||
|       pathToRevolveArg, | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // We're not creating a pipe expression, | ||||
|   // but rather a separate constant for the extrusion | ||||
|   const name = findUniqueName(clonedAst, KCL_DEFAULT_CONSTANT_PREFIXES.REVOLVE) | ||||
|   const VariableDeclaration = createVariableDeclaration(name, revolveCall) | ||||
|   const sketchIndexInPathToNode = | ||||
|     sketchPathToDecleration.findIndex((a) => a[0] === 'body') + 1 | ||||
|   const sketchIndexInBody = sketchPathToDecleration[sketchIndexInPathToNode][0] | ||||
|   if (typeof sketchIndexInBody !== 'number') | ||||
|     return new Error('expected sketchIndexInBody to be a number') | ||||
|   clonedAst.body.splice(sketchIndexInBody + 1, 0, VariableDeclaration) | ||||
|  | ||||
|   const pathToRevolveArg: PathToNode = [ | ||||
|     ['body', ''], | ||||
|     [sketchIndexInBody + 1, 'index'], | ||||
|     ['declaration', 'VariableDeclaration'], | ||||
|     ['init', 'VariableDeclarator'], | ||||
|     ['arguments', 'CallExpression'], | ||||
|     [0, 'index'], | ||||
|   ] | ||||
|   return { | ||||
|     modifiedAst: clonedAst, | ||||
|     pathToSketchNode: [...pathToSketchNode.slice(0, -1), [-1, 'index']], | ||||
|     pathToRevolveArg, | ||||
|   } | ||||
| } | ||||
