Compare commits
	
		
			1 Commits
		
	
	
		
			nightly-v2
			...
			pierremtb/
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 34272b872d | 
| @ -1,3 +1,3 @@ | |||||||
| [codespell] | [codespell] | ||||||
| ignore-words-list: crate,everytime,inout,co-ordinate,ot,nwo,atleast,ue,afterall | ignore-words-list: crate,everytime,inout,co-ordinate,ot,nwo,absolutey,atleast,ue,afterall | ||||||
| skip: **/target,node_modules,build,**/Cargo.lock,./docs/kcl/*.md,.yarn.lock,**/yarn.lock,./openapi/*.json,./src/lib/machine-api.d.ts,./packages/codemirror-lang-kcl/test/all.test.ts | skip: **/target,node_modules,build,**/Cargo.lock,./docs/kcl/*.md,.yarn.lock,**/yarn.lock,./openapi/*.json,./src/lib/machine-api.d.ts | ||||||
|  | |||||||
							
								
								
									
										59
									
								
								.github/ci-cd-scripts/playwright-browser-chrome.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						| @ -0,0 +1,59 @@ | |||||||
|  | # 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,17 +1,15 @@ | |||||||
| #!/bin/bash |  | ||||||
|  |  | ||||||
| # bash strict mode | # bash strict mode | ||||||
| set -euo pipefail | set -euo pipefail | ||||||
|  |  | ||||||
| if [[ ! -f "test-results/.last-run.json" ]]; then | if [[ ! -f "test-results/.last-run.json" ]]; then | ||||||
|     # if no last run artifact, than run plawright normally |     # if no last run artifact, than run plawright normally | ||||||
|     echo "run playwright normally" |     echo "run playwright normally" | ||||||
|         if [[ "$3" == *ubuntu* ]]; then |         if [[ "$1" == ubuntu-latest* ]]; then | ||||||
|             xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn test:playwright:electron:ubuntu -- --shard=$1/$2 || true |             xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn test:playwright:electron:ubuntu || true | ||||||
|         elif [[ "$3" == *windows* ]]; then |         elif [[ "$1" == windows-latest* ]]; then | ||||||
|             yarn test:playwright:electron:windows -- --shard=$1/$2 || true |             yarn test:playwright:electron:windows || true | ||||||
|         elif [[ "$3" == *macos* ]]; then |         elif [[ "$1" == macos-14* ]]; then | ||||||
|             yarn test:playwright:electron:macos  -- --shard=$1/$2 || true |             yarn test:playwright:electron:macos || true | ||||||
|         else |         else | ||||||
|             echo "Do not run playwright. Unable to detect os runtime." |             echo "Do not run playwright. Unable to detect os runtime." | ||||||
|             exit 1 |             exit 1 | ||||||
| @ -30,11 +28,11 @@ while [[ $retry -le $max_retrys ]]; do | |||||||
|         if [[ $failed_tests -gt 0 ]]; then |         if [[ $failed_tests -gt 0 ]]; then | ||||||
|             echo "retried=true" >>$GITHUB_OUTPUT |             echo "retried=true" >>$GITHUB_OUTPUT | ||||||
|             echo "run playwright with last failed tests and retry $retry" |             echo "run playwright with last failed tests and retry $retry" | ||||||
|             if [[ "$3" == *ubuntu* ]]; then |             if [[ "$1" == ubuntu-latest* ]]; then | ||||||
|                 xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn test:playwright:electron:ubuntu -- --last-failed || true |                 xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn test:playwright:electron:ubuntu -- --last-failed || true | ||||||
|             elif [[ "$3" == *windows* ]]; then |             elif [[ "$1" == windows-latest* ]]; then | ||||||
|                 yarn test:playwright:electron:windows -- --last-failed || true |                 yarn test:playwright:electron:windows -- --last-failed || true | ||||||
|             elif [[ "$3" == *macos* ]]; then |             elif [[ "$1" == macos-14* ]]; then | ||||||
|                 yarn test:playwright:electron:macos -- --last-failed || true |                 yarn test:playwright:electron:macos -- --last-failed || true | ||||||
|             else |             else | ||||||
|                 echo "Do not run playwright. Unable to detect os runtime." |                 echo "Do not run playwright. Unable to detect os runtime." | ||||||
|  | |||||||
							
								
								
									
										46
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -5,28 +5,24 @@ | |||||||
|  |  | ||||||
| version: 2 | version: 2 | ||||||
| updates: | updates: | ||||||
|   - package-ecosystem: 'npm' # See documentation for possible values |     - package-ecosystem: 'npm' # See documentation for possible values | ||||||
|     directory: '/' # Location of package manifests |       directory: '/' # Location of package manifests | ||||||
|     schedule: |       schedule: | ||||||
|       interval: 'weekly' |           interval: 'weekly' | ||||||
|     reviewers: |       reviewers: | ||||||
|       - franknoirot |           - franknoirot | ||||||
|       - irev-dev |           - irev-dev | ||||||
|   - package-ecosystem: 'github-actions' # See documentation for possible values |     - package-ecosystem: 'github-actions' # See documentation for possible values | ||||||
|     directory: '/' # Location of package manifests |       directory: '/' # Location of package manifests | ||||||
|     schedule: |       schedule: | ||||||
|       interval: 'weekly' |           interval: 'weekly' | ||||||
|     reviewers: |       reviewers: | ||||||
|       - adamchalmers |           - adamchalmers | ||||||
|       - jessfraz |           - jessfraz | ||||||
|   - package-ecosystem: 'cargo' # See documentation for possible values |     - package-ecosystem: 'cargo' # See documentation for possible values | ||||||
|     directory: '/src/wasm-lib/' # Location of package manifests |       directory: '/src/wasm-lib/' # Location of package manifests | ||||||
|     schedule: |       schedule: | ||||||
|       interval: 'weekly' |           interval: 'weekly' | ||||||
|     reviewers: |       reviewers: | ||||||
|       - adamchalmers |           - adamchalmers | ||||||
|       - jessfraz |           - jessfraz | ||||||
|     groups: |  | ||||||
|       serde-dependencies: |  | ||||||
|         patterns: |  | ||||||
|           - "serde*" |  | ||||||
|  | |||||||
							
								
								
									
										19
									
								
								.github/workflows/build-apps.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -165,6 +165,7 @@ jobs: | |||||||
|       - name: Build the app (release) |       - name: Build the app (release) | ||||||
|         if: ${{ env.IS_RELEASE == 'true' || env.IS_NIGHTLY == 'true' }} |         if: ${{ env.IS_RELEASE == 'true' || env.IS_NIGHTLY == 'true' }} | ||||||
|         env: |         env: | ||||||
|  |           PUBLISH_FOR_PULL_REQUEST: true | ||||||
|           APPLE_ID: ${{ secrets.APPLE_ID }} |           APPLE_ID: ${{ secrets.APPLE_ID }} | ||||||
|           APPLE_PASSWORD: ${{ secrets.APPLE_PASSWORD }} |           APPLE_PASSWORD: ${{ secrets.APPLE_PASSWORD }} | ||||||
|           APPLE_APP_SPECIFIC_PASSWORD: ${{ secrets.APPLE_PASSWORD }} |           APPLE_APP_SPECIFIC_PASSWORD: ${{ secrets.APPLE_PASSWORD }} | ||||||
| @ -172,14 +173,9 @@ jobs: | |||||||
|           CSC_LINK: ${{ secrets.APPLE_CERTIFICATE }} |           CSC_LINK: ${{ secrets.APPLE_CERTIFICATE }} | ||||||
|           CSC_KEY_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }} |           CSC_KEY_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }} | ||||||
|           CSC_KEYCHAIN: ${{ secrets.APPLE_SIGNING_IDENTITY }} |           CSC_KEYCHAIN: ${{ secrets.APPLE_SIGNING_IDENTITY }} | ||||||
|  |           CSC_FOR_PULL_REQUEST: true | ||||||
|           WINDOWS_CERTIFICATE_THUMBPRINT: ${{ secrets.WINDOWS_CERTIFICATE_THUMBPRINT }} |           WINDOWS_CERTIFICATE_THUMBPRINT: ${{ secrets.WINDOWS_CERTIFICATE_THUMBPRINT }} | ||||||
|           DEBUG: "electron-notarize*" |         run: yarn electron-builder --config --publish always | ||||||
|         # TODO: Fix electron-notarize flakes. The logs above should help gather more data on failures |  | ||||||
|         uses: nick-fields/retry@v3.0.0 |  | ||||||
|         with: |  | ||||||
|           timeout_minutes: 10 |  | ||||||
|           max_attempts: 3 |  | ||||||
|           command: yarn electron-builder --config --publish always |  | ||||||
|  |  | ||||||
|       - name: List artifacts in out/ |       - name: List artifacts in out/ | ||||||
|         run: ls -R out |         run: ls -R out | ||||||
| @ -233,14 +229,9 @@ jobs: | |||||||
|           CSC_LINK: ${{ secrets.APPLE_CERTIFICATE }} |           CSC_LINK: ${{ secrets.APPLE_CERTIFICATE }} | ||||||
|           CSC_KEY_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }} |           CSC_KEY_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }} | ||||||
|           CSC_KEYCHAIN: ${{ secrets.APPLE_SIGNING_IDENTITY }} |           CSC_KEYCHAIN: ${{ secrets.APPLE_SIGNING_IDENTITY }} | ||||||
|  |           CSC_FOR_PULL_REQUEST: true | ||||||
|           WINDOWS_CERTIFICATE_THUMBPRINT: ${{ secrets.WINDOWS_CERTIFICATE_THUMBPRINT }} |           WINDOWS_CERTIFICATE_THUMBPRINT: ${{ secrets.WINDOWS_CERTIFICATE_THUMBPRINT }} | ||||||
|           DEBUG: "electron-notarize*" |         run: yarn electron-builder --config --publish always | ||||||
|         # TODO: Fix electron-notarize flakes. The logs above should help gather more data on failures |  | ||||||
|         uses: nick-fields/retry@v3.0.0 |  | ||||||
|         with: |  | ||||||
|           timeout_minutes: 10 |  | ||||||
|           max_attempts: 3 |  | ||||||
|           command: yarn electron-builder --config --publish always |  | ||||||
|  |  | ||||||
|       - uses: actions/upload-artifact@v4 |       - uses: actions/upload-artifact@v4 | ||||||
|         if: ${{ env.IS_RELEASE == 'true' }} |         if: ${{ env.IS_RELEASE == 'true' }} | ||||||
|  | |||||||
							
								
								
									
										22
									
								
								.github/workflows/cargo-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -2,8 +2,28 @@ on: | |||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|       - main |       - main | ||||||
|  |     paths: | ||||||
|  |       - 'src/wasm-lib/**.rs' | ||||||
|  |       - 'src/wasm-lib/**.hbs' | ||||||
|  |       - 'src/wasm-lib/**.gen' | ||||||
|  |       - 'src/wasm-lib/**.snap' | ||||||
|  |       - '**/Cargo.toml' | ||||||
|  |       - '**/Cargo.lock' | ||||||
|  |       - '**/rust-toolchain.toml' | ||||||
|  |       - 'src/wasm-lib/**.kcl' | ||||||
|  |       - .github/workflows/cargo-test.yml | ||||||
|  |  | ||||||
|   pull_request: |   pull_request: | ||||||
|  |     paths: | ||||||
|  |       - 'src/wasm-lib/**.rs' | ||||||
|  |       - 'src/wasm-lib/**.hbs' | ||||||
|  |       - 'src/wasm-lib/**.gen' | ||||||
|  |       - 'src/wasm-lib/**.snap' | ||||||
|  |       - '**/Cargo.toml' | ||||||
|  |       - '**/Cargo.lock' | ||||||
|  |       - '**/rust-toolchain.toml' | ||||||
|  |       - 'src/wasm-lib/**.kcl' | ||||||
|  |       - .github/workflows/cargo-test.yml | ||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
| permissions: read-all | permissions: read-all | ||||||
| concurrency: | concurrency: | ||||||
| @ -51,7 +71,7 @@ jobs: | |||||||
|           KITTYCAD_API_TOKEN: ${{secrets.KITTYCAD_API_TOKEN}} |           KITTYCAD_API_TOKEN: ${{secrets.KITTYCAD_API_TOKEN}} | ||||||
|           RUST_MIN_STACK: 10485760000 |           RUST_MIN_STACK: 10485760000 | ||||||
|       - name: Upload to codecov.io |       - name: Upload to codecov.io | ||||||
|         uses: codecov/codecov-action@v5 |         uses: codecov/codecov-action@v4 | ||||||
|         with: |         with: | ||||||
|           token: ${{secrets.CODECOV_TOKEN}} |           token: ${{secrets.CODECOV_TOKEN}} | ||||||
|           fail_ci_if_error: true |           fail_ci_if_error: true | ||||||
|  | |||||||
							
								
								
									
										166
									
								
								.github/workflows/e2e-tests.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -33,19 +33,20 @@ jobs: | |||||||
|             rust: |             rust: | ||||||
|               - 'src/wasm-lib/**' |               - 'src/wasm-lib/**' | ||||||
|  |  | ||||||
|   electron: |   browser: | ||||||
|     timeout-minutes: 60 |     timeout-minutes: ${{ matrix.os == 'macos-14' && 60 || 50 }} | ||||||
|     name: playwright:electron:${{ matrix.os }} ${{ matrix.shardIndex }} ${{ matrix.shardTotal }} |     name: playwright:browser:${{ matrix.os }} ${{ matrix.shardIndex }} ${{ matrix.shardTotal }} | ||||||
|     strategy: |     strategy: | ||||||
|       fail-fast: false |       fail-fast: false | ||||||
|       matrix: |       matrix: | ||||||
|         # TODO: enable self-hosted-windows-8-cores once available |         os: [ubuntu-latest-8-cores, windows-latest-8-cores] | ||||||
|         os: [namespace-profile-ubuntu-8-cores, namespace-profile-macos-8-cores, windows-16-cores] |  | ||||||
|         shardIndex: [1, 2, 3, 4] |         shardIndex: [1, 2, 3, 4] | ||||||
|         shardTotal: [4] |         shardTotal: [4] | ||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|     needs: check-rust-changes |     needs: check-rust-changes | ||||||
|     steps: |     steps: | ||||||
|  |     - name: Tune GitHub-hosted runner network | ||||||
|  |       uses: smorimoto/tune-github-hosted-runner-network@v1 | ||||||
|     - uses: actions/checkout@v4 |     - uses: actions/checkout@v4 | ||||||
|     - uses: actions/setup-node@v4 |     - uses: actions/setup-node@v4 | ||||||
|       with: |       with: | ||||||
| @ -100,8 +101,7 @@ jobs: | |||||||
|         echo "/opt/homebrew/opt/gnu-sed/libexec/gnubin" >> $GITHUB_PATH |         echo "/opt/homebrew/opt/gnu-sed/libexec/gnubin" >> $GITHUB_PATH | ||||||
|     - name: Install vector |     - name: Install vector | ||||||
|       shell: bash |       shell: bash | ||||||
|       # TODO: figure out what to do with this, it's failing |       if:  ${{ !startsWith(matrix.os, 'windows') }} | ||||||
|       if: false |  | ||||||
|       run: | |       run: | | ||||||
|         curl --proto '=https' --tlsv1.2 -sSfL https://sh.vector.dev > /tmp/vector.sh |         curl --proto '=https' --tlsv1.2 -sSfL https://sh.vector.dev > /tmp/vector.sh | ||||||
|         chmod +x /tmp/vector.sh |         chmod +x /tmp/vector.sh | ||||||
| @ -123,16 +123,13 @@ jobs: | |||||||
|       if: steps.download-wasm.outcome == 'failure' |       if: steps.download-wasm.outcome == 'failure' | ||||||
|       shell: bash |       shell: bash | ||||||
|       run: yarn build:wasm |       run: yarn build:wasm | ||||||
|     - name: build electron |     - name: build web | ||||||
|  |       run: yarn build:local | ||||||
|       shell: bash |       shell: bash | ||||||
|       run: yarn tron:package |  | ||||||
|     - name: Run ubuntu/chrome snapshots |     - name: Run ubuntu/chrome snapshots | ||||||
|       if: ${{ matrix.os == 'namespace-profile-ubuntu-8-cores' && matrix.shardIndex == 1 }} |  | ||||||
|       shell: bash |       shell: bash | ||||||
|       # TODO: break this in its own job, for now it's not slowing down the overall execution as ubuntu is the quickest, |  | ||||||
|       # but we could do better. This forces a large 1/1 shard of all 20 snapshot tests that runs in about 3 minutes. |  | ||||||
|       run: | |       run: | | ||||||
|         PLATFORM=web yarn playwright test --config=playwright.config.ts --retries="3" --update-snapshots --grep=@snapshot  --shard=1/1 |         yarn playwright test --project="Google Chrome" --config=playwright.ci.config.ts --retries="3" --update-snapshots --grep=@snapshot  --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} | ||||||
|       env: |       env: | ||||||
|         CI: true |         CI: true | ||||||
|         NODE_ENV: development |         NODE_ENV: development | ||||||
| @ -153,7 +150,6 @@ jobs: | |||||||
|       continue-on-error: true |       continue-on-error: true | ||||||
|       run: rm -r test-results |       run: rm -r test-results | ||||||
|     - name: check for changes |     - name: check for changes | ||||||
|       if: ${{ matrix.os == 'namespace-profile-ubuntu-8-cores' && matrix.shardIndex == 1 }} |  | ||||||
|       shell: bash |       shell: bash | ||||||
|       id: git-check |       id: git-check | ||||||
|       run: | |       run: | | ||||||
| @ -190,12 +186,12 @@ jobs: | |||||||
|       with: |       with: | ||||||
|         name: test-results-${{ matrix.os }}-${{ matrix.shardIndex }}-${{ github.sha }} |         name: test-results-${{ matrix.os }}-${{ matrix.shardIndex }}-${{ github.sha }} | ||||||
|         path: test-results/ |         path: test-results/ | ||||||
|     - name: Run playwright/electron flow (with retries) |     - name: Run playwright/chrome flow (with retries) | ||||||
|       id: retry |       id: retry | ||||||
|       if: ${{ !cancelled() && (success() || failure()) }} |       if: ${{ !cancelled() && (success() || failure()) }} | ||||||
|       shell: bash |       shell: bash | ||||||
|       run: | |       run: | | ||||||
|         .github/ci-cd-scripts/playwright-electron.sh ${{matrix.shardIndex}} ${{matrix.shardTotal}} ${{matrix.os}} |         .github/ci-cd-scripts/playwright-browser-chrome.sh ${{matrix.shardIndex}} ${{matrix.shardTotal}} ${{matrix.os}} | ||||||
|       env: |       env: | ||||||
|         CI: true |         CI: true | ||||||
|         FAIL_ON_CONSOLE_ERRORS: true |         FAIL_ON_CONSOLE_ERRORS: true | ||||||
| @ -203,6 +199,11 @@ jobs: | |||||||
|         VITE_KC_DEV_TOKEN: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} |         VITE_KC_DEV_TOKEN: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||||
|         VITE_KC_SKIP_AUTH: true |         VITE_KC_SKIP_AUTH: true | ||||||
|         token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} |         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 |     - uses: actions/upload-artifact@v4 | ||||||
|       if: always() |       if: always() | ||||||
|       with: |       with: | ||||||
| @ -220,3 +221,136 @@ jobs: | |||||||
|         retention-days: 30 |         retention-days: 30 | ||||||
|         overwrite: true |         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 | ||||||
|  | |||||||
							
								
								
									
										55
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @ -337,47 +337,13 @@ For individual testing: | |||||||
| yarn test abstractSyntaxTree -t "unexpected closed curly brace" --silent=false | yarn test abstractSyntaxTree -t "unexpected closed curly brace" --silent=false | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Which will run our suite of [Vitest unit](https://vitest.dev/) and [React Testing Library E2E](https://testing-library.com/docs/react-testing-library/intro) tests, in interactive mode by default. | Which will run our suite of [Vitest unit](https://vitest.dev/) and [React Testing Library E2E](https://testing-library.com/docs/react-testing-library/intro/) tests, in interactive mode by default. | ||||||
|  |  | ||||||
| ### Rust tests | ### Rust tests | ||||||
|  |  | ||||||
| **Dependencies** |  | ||||||
|  |  | ||||||
| - `KITTYCAD_API_TOKEN` |  | ||||||
| - `cargo-nextest` |  | ||||||
| - `just` |  | ||||||
|  |  | ||||||
| #### Setting KITTYCAD_API_TOKEN |  | ||||||
| Use the production zoo.dev token, set this environment variable before running the tests |  | ||||||
|  |  | ||||||
| #### Installing cargonextest |  | ||||||
|  |  | ||||||
| ``` |  | ||||||
| cd src/wasm-lib |  | ||||||
| cargo search cargo-nextest |  | ||||||
| cargo install cargo-nextest |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| #### just |  | ||||||
| install [`just`](https://github.com/casey/just?tab=readme-ov-file#pre-built-binaries) |  | ||||||
|  |  | ||||||
| #### Running the tests |  | ||||||
|  |  | ||||||
| ```bash | ```bash | ||||||
| # With just |  | ||||||
| # Make sure KITTYCAD_API_TOKEN=<prod zoo.dev token> is set |  | ||||||
| # Make sure you installed cargo-nextest |  | ||||||
| # Make sure you installed just |  | ||||||
| cd src/wasm-lib | cd src/wasm-lib | ||||||
| just test | KITTYCAD_API_TOKEN=XXX cargo test -- --test-threads=1 | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ```bash |  | ||||||
| # Without just |  | ||||||
| # Make sure KITTYCAD_API_TOKEN=<prod zoo.dev token> is set |  | ||||||
| # Make sure you installed cargo-nextest |  | ||||||
| cd src/wasm-lib |  | ||||||
| export RUST_BRACKTRACE="full" && cargo nextest run --workspace --test-threads=1 |  | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Where `XXX` is an API token from the production engine (NOT the dev environment). | Where `XXX` is an API token from the production engine (NOT the dev environment). | ||||||
| @ -422,6 +388,23 @@ yarn test:unit:local | |||||||
|  |  | ||||||
| #### E2E Tests | #### 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** | **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. | 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,7 +22,3 @@ once fixed in engine will just start working here with no language changes. | |||||||
|  |  | ||||||
| - **Chamfers**: Chamfers cannot intersect, you will get an error. Only simple | - **Chamfers**: Chamfers cannot intersect, you will get an error. Only simple | ||||||
|     chamfer cases work currently. |     chamfer cases work currently. | ||||||
|  |  | ||||||
| - **Appearance**: Changing the appearance on a loft does not work. |  | ||||||
|  |  | ||||||
| - **Helix**: Currently sweeping a helix does not work. |  | ||||||
|  | |||||||
| @ -19,7 +19,6 @@ layout: manual | |||||||
| * [`angledLineThatIntersects`](kcl/angledLineThatIntersects) | * [`angledLineThatIntersects`](kcl/angledLineThatIntersects) | ||||||
| * [`angledLineToX`](kcl/angledLineToX) | * [`angledLineToX`](kcl/angledLineToX) | ||||||
| * [`angledLineToY`](kcl/angledLineToY) | * [`angledLineToY`](kcl/angledLineToY) | ||||||
| * [`appearance`](kcl/appearance) |  | ||||||
| * [`arc`](kcl/arc) | * [`arc`](kcl/arc) | ||||||
| * [`arcTo`](kcl/arcTo) | * [`arcTo`](kcl/arcTo) | ||||||
| * [`asin`](kcl/asin) | * [`asin`](kcl/asin) | ||||||
| @ -30,12 +29,10 @@ layout: manual | |||||||
| * [`assertLessThan`](kcl/assertLessThan) | * [`assertLessThan`](kcl/assertLessThan) | ||||||
| * [`assertLessThanOrEq`](kcl/assertLessThanOrEq) | * [`assertLessThanOrEq`](kcl/assertLessThanOrEq) | ||||||
| * [`atan`](kcl/atan) | * [`atan`](kcl/atan) | ||||||
| * [`atan2`](kcl/atan2) |  | ||||||
| * [`bezierCurve`](kcl/bezierCurve) | * [`bezierCurve`](kcl/bezierCurve) | ||||||
| * [`ceil`](kcl/ceil) | * [`ceil`](kcl/ceil) | ||||||
| * [`chamfer`](kcl/chamfer) | * [`chamfer`](kcl/chamfer) | ||||||
| * [`circle`](kcl/circle) | * [`circle`](kcl/circle) | ||||||
| * [`circleThreePoint`](kcl/circleThreePoint) |  | ||||||
| * [`close`](kcl/close) | * [`close`](kcl/close) | ||||||
| * [`cm`](kcl/cm) | * [`cm`](kcl/cm) | ||||||
| * [`cos`](kcl/cos) | * [`cos`](kcl/cos) | ||||||
| @ -48,7 +45,6 @@ layout: manual | |||||||
| * [`getOppositeEdge`](kcl/getOppositeEdge) | * [`getOppositeEdge`](kcl/getOppositeEdge) | ||||||
| * [`getPreviousAdjacentEdge`](kcl/getPreviousAdjacentEdge) | * [`getPreviousAdjacentEdge`](kcl/getPreviousAdjacentEdge) | ||||||
| * [`helix`](kcl/helix) | * [`helix`](kcl/helix) | ||||||
| * [`helixRevolutions`](kcl/helixRevolutions) |  | ||||||
| * [`hole`](kcl/hole) | * [`hole`](kcl/hole) | ||||||
| * [`hollow`](kcl/hollow) | * [`hollow`](kcl/hollow) | ||||||
| * [`import`](kcl/import) | * [`import`](kcl/import) | ||||||
| @ -82,7 +78,6 @@ layout: manual | |||||||
| * [`pi`](kcl/pi) | * [`pi`](kcl/pi) | ||||||
| * [`polar`](kcl/polar) | * [`polar`](kcl/polar) | ||||||
| * [`polygon`](kcl/polygon) | * [`polygon`](kcl/polygon) | ||||||
| * [`pop`](kcl/pop) |  | ||||||
| * [`pow`](kcl/pow) | * [`pow`](kcl/pow) | ||||||
| * [`profileStart`](kcl/profileStart) | * [`profileStart`](kcl/profileStart) | ||||||
| * [`profileStartX`](kcl/profileStartX) | * [`profileStartX`](kcl/profileStartX) | ||||||
| @ -104,8 +99,8 @@ layout: manual | |||||||
| * [`sin`](kcl/sin) | * [`sin`](kcl/sin) | ||||||
| * [`sqrt`](kcl/sqrt) | * [`sqrt`](kcl/sqrt) | ||||||
| * [`startProfileAt`](kcl/startProfileAt) | * [`startProfileAt`](kcl/startProfileAt) | ||||||
|  | * [`startSketchAt`](kcl/startSketchAt) | ||||||
| * [`startSketchOn`](kcl/startSketchOn) | * [`startSketchOn`](kcl/startSketchOn) | ||||||
| * [`sweep`](kcl/sweep) |  | ||||||
| * [`tan`](kcl/tan) | * [`tan`](kcl/tan) | ||||||
| * [`tangentToEnd`](kcl/tangentToEnd) | * [`tangentToEnd`](kcl/tangentToEnd) | ||||||
| * [`tangentialArc`](kcl/tangentialArc) | * [`tangentialArc`](kcl/tangentialArc) | ||||||
|  | |||||||
| @ -35,7 +35,7 @@ The transform function returns a transform object. All properties of the object | |||||||
|    - `rotation.origin` (either "local" i.e. rotate around its own center, "global" i.e. rotate around the scene's center, or a 3D point, defaults to "local") |    - `rotation.origin` (either "local" i.e. rotate around its own center, "global" i.e. rotate around the scene's center, or a 3D point, defaults to "local") | ||||||
|  |  | ||||||
| ```js | ```js | ||||||
| patternTransform(total_instances: integer, transform_function: FunctionParam, solid_set: SolidSet) -> [Solid] | patternTransform(total_instances: u32, transform_function: FunctionParam, solid_set: SolidSet) -> [Solid] | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -43,7 +43,7 @@ patternTransform(total_instances: integer, transform_function: FunctionParam, so | |||||||
|  |  | ||||||
| | Name | Type | Description | Required | | | Name | Type | Description | Required | | ||||||
| |----------|------|-------------|----------| | |----------|------|-------------|----------| | ||||||
| | `total_instances` | `integer` |  | Yes | | | `total_instances` | `u32` |  | Yes | | ||||||
| | `transform_function` | `FunctionParam` |  | Yes | | | `transform_function` | `FunctionParam` |  | Yes | | ||||||
| | `solid_set` | [`SolidSet`](/docs/kcl/types/SolidSet) | A solid or a group of solids. | Yes | | | `solid_set` | [`SolidSet`](/docs/kcl/types/SolidSet) | A solid or a group of solids. | Yes | | ||||||
|  |  | ||||||
| @ -95,8 +95,7 @@ fn cube(length, center) { | |||||||
|   p2 = [l + x, l + y] |   p2 = [l + x, l + y] | ||||||
|   p3 = [l + x, -l + y] |   p3 = [l + x, -l + y] | ||||||
|  |  | ||||||
|   return startSketchOn('XY') |   return startSketchAt(p0) | ||||||
|     |> startProfileAt(p0, %) |  | ||||||
|     |> lineTo(p1, %) |     |> lineTo(p1, %) | ||||||
|     |> lineTo(p2, %) |     |> lineTo(p2, %) | ||||||
|     |> lineTo(p3, %) |     |> lineTo(p3, %) | ||||||
| @ -133,8 +132,7 @@ fn cube(length, center) { | |||||||
|   p2 = [l + x, l + y] |   p2 = [l + x, l + y] | ||||||
|   p3 = [l + x, -l + y] |   p3 = [l + x, -l + y] | ||||||
|  |  | ||||||
|   return startSketchOn('XY') |   return startSketchAt(p0) | ||||||
|     |> startProfileAt(p0, %) |  | ||||||
|     |> lineTo(p1, %) |     |> lineTo(p1, %) | ||||||
|     |> lineTo(p2, %) |     |> lineTo(p2, %) | ||||||
|     |> lineTo(p3, %) |     |> lineTo(p3, %) | ||||||
| @ -197,8 +195,7 @@ fn transform(i) { | |||||||
|     { rotation = { angle = 45 * i } } |     { rotation = { angle = 45 * i } } | ||||||
|   ] |   ] | ||||||
| } | } | ||||||
| startSketchOn('XY') | startSketchAt([0, 0]) | ||||||
|   |> startProfileAt([0, 0], %) |  | ||||||
|   |> polygon({ |   |> polygon({ | ||||||
|        radius = 10, |        radius = 10, | ||||||
|        numSides = 4, |        numSides = 4, | ||||||
|  | |||||||
| @ -9,7 +9,7 @@ Just like patternTransform, but works on 2D sketches not 3D solids. | |||||||
|  |  | ||||||
|  |  | ||||||
| ```js | ```js | ||||||
| patternTransform2d(total_instances: integer, transform_function: FunctionParam, solid_set: SketchSet) -> [Sketch] | patternTransform2d(total_instances: u32, transform_function: FunctionParam, solid_set: SketchSet) -> [Sketch] | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -17,7 +17,7 @@ patternTransform2d(total_instances: integer, transform_function: FunctionParam, | |||||||
|  |  | ||||||
| | Name | Type | Description | Required | | | Name | Type | Description | Required | | ||||||
| |----------|------|-------------|----------| | |----------|------|-------------|----------| | ||||||
| | `total_instances` | `integer` |  | Yes | | | `total_instances` | `u32` |  | Yes | | ||||||
| | `transform_function` | `FunctionParam` |  | Yes | | | `transform_function` | `FunctionParam` |  | Yes | | ||||||
| | `solid_set` | [`SketchSet`](/docs/kcl/types/SketchSet) | A sketch or a group of sketches. | Yes | | | `solid_set` | [`SketchSet`](/docs/kcl/types/SketchSet) | A sketch or a group of sketches. | Yes | | ||||||
|  |  | ||||||
|  | |||||||
| @ -43,7 +43,7 @@ fn sum(arr) { | |||||||
|  |  | ||||||
| /* The above is basically like this pseudo-code: | /* The above is basically like this pseudo-code: | ||||||
| fn sum(arr): | fn sum(arr): | ||||||
|     sumSoFar = 0 |     let sumSoFar = 0 | ||||||
|     for i in arr: |     for i in arr: | ||||||
|         sumSoFar = add(sumSoFar, i) |         sumSoFar = add(sumSoFar, i) | ||||||
|     return sumSoFar */ |     return sumSoFar */ | ||||||
| @ -79,11 +79,10 @@ fn decagon(radius) { | |||||||
|   stepAngle = 1 / 10 * tau() |   stepAngle = 1 / 10 * tau() | ||||||
|  |  | ||||||
|   // Start the decagon sketch at this point. |   // Start the decagon sketch at this point. | ||||||
|   startOfDecagonSketch = startSketchOn('XY') |   startOfDecagonSketch = startSketchAt([cos(0) * radius, sin(0) * radius]) | ||||||
|     |> startProfileAt([cos(0) * radius, sin(0) * radius], %) |  | ||||||
|  |  | ||||||
|     // Use a `reduce` to draw the remaining decagon sides. |   // Use a `reduce` to draw the remaining decagon sides. | ||||||
|     // For each number in the array 1..10, run the given function, |   // 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. |   // which takes a partially-sketched decagon and adds one more edge to it. | ||||||
|   fullDecagon = reduce([1..10], startOfDecagonSketch, fn(i, partialDecagon) { |   fullDecagon = reduce([1..10], startOfDecagonSketch, fn(i, partialDecagon) { | ||||||
|     // Draw one edge of the decagon. |     // Draw one edge of the decagon. | ||||||
| @ -97,15 +96,14 @@ fn decagon(radius) { | |||||||
|  |  | ||||||
| /* The `decagon` above is basically like this pseudo-code: | /* The `decagon` above is basically like this pseudo-code: | ||||||
| fn decagon(radius): | fn decagon(radius): | ||||||
|     stepAngle = (1/10) * tau() |     let stepAngle = (1/10) * tau() | ||||||
|     plane = startSketchOn('XY') |     let startOfDecagonSketch = startSketchAt([(cos(0)*radius), (sin(0) * radius)]) | ||||||
|     startOfDecagonSketch = startProfileAt([(cos(0)*radius), (sin(0) * radius)], plane) |  | ||||||
|  |  | ||||||
|     // Here's the reduce part. |     // Here's the reduce part. | ||||||
|     partialDecagon = startOfDecagonSketch |     let partialDecagon = startOfDecagonSketch | ||||||
|     for i in [1..10]: |     for i in [1..10]: | ||||||
|         x = cos(stepAngle * i) * radius |         let x = cos(stepAngle * i) * radius | ||||||
|         y = sin(stepAngle * i) * radius |         let y = sin(stepAngle * i) * radius | ||||||
|         partialDecagon = lineTo([x, y], partialDecagon) |         partialDecagon = lineTo([x, y], partialDecagon) | ||||||
|     fullDecagon = partialDecagon // it's now full |     fullDecagon = partialDecagon // it's now full | ||||||
|     return fullDecagon */ |     return fullDecagon */ | ||||||
|  | |||||||
| @ -28,8 +28,7 @@ segEnd(tag: TagIdentifier) -> [number] | |||||||
|  |  | ||||||
| ```js | ```js | ||||||
| w = 15 | w = 15 | ||||||
| cube = startSketchOn('XY') | cube = startSketchAt([0, 0]) | ||||||
|   |> startProfileAt([0, 0], %) |  | ||||||
|   |> line([w, 0], %, $line1) |   |> line([w, 0], %, $line1) | ||||||
|   |> line([0, w], %, $line2) |   |> line([0, w], %, $line2) | ||||||
|   |> line([-w, 0], %, $line3) |   |> line([-w, 0], %, $line3) | ||||||
| @ -38,8 +37,7 @@ cube = startSketchOn('XY') | |||||||
|   |> extrude(5, %) |   |> extrude(5, %) | ||||||
|  |  | ||||||
| fn cylinder(radius, tag) { | fn cylinder(radius, tag) { | ||||||
|   return startSketchOn('XY') |   return startSketchAt([0, 0]) | ||||||
|     |> startProfileAt([0, 0], %) |  | ||||||
|     |> circle({ |     |> circle({ | ||||||
|          radius = radius, |          radius = radius, | ||||||
|          center = segEnd(tag) |          center = segEnd(tag) | ||||||
|  | |||||||
| @ -28,8 +28,7 @@ segStart(tag: TagIdentifier) -> [number] | |||||||
|  |  | ||||||
| ```js | ```js | ||||||
| w = 15 | w = 15 | ||||||
| cube = startSketchOn('XY') | cube = startSketchAt([0, 0]) | ||||||
|   |> startProfileAt([0, 0], %) |  | ||||||
|   |> line([w, 0], %, $line1) |   |> line([w, 0], %, $line1) | ||||||
|   |> line([0, w], %, $line2) |   |> line([0, w], %, $line2) | ||||||
|   |> line([-w, 0], %, $line3) |   |> line([-w, 0], %, $line3) | ||||||
| @ -38,8 +37,7 @@ cube = startSketchOn('XY') | |||||||
|   |> extrude(5, %) |   |> extrude(5, %) | ||||||
|  |  | ||||||
| fn cylinder(radius, tag) { | fn cylinder(radius, tag) { | ||||||
|   return startSketchOn('XY') |   return startSketchAt([0, 0]) | ||||||
|     |> startProfileAt([0, 0], %) |  | ||||||
|     |> circle({ |     |> circle({ | ||||||
|          radius = radius, |          radius = radius, | ||||||
|          center = segStart(tag) |          center = segStart(tag) | ||||||
|  | |||||||
| @ -4,8 +4,6 @@ excerpt: "Start a new 2-dimensional sketch at a given point on the 'XY' plane." | |||||||
| layout: manual | layout: manual | ||||||
| --- | --- | ||||||
|  |  | ||||||
| **WARNING:** This function is deprecated. |  | ||||||
|  |  | ||||||
| Start a new 2-dimensional sketch at a given point on the 'XY' plane. | Start a new 2-dimensional sketch at a given point on the 'XY' plane. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										22517
									
								
								docs/kcl/std.json
									
									
									
									
									
								
							
							
						
						| @ -13,18 +13,13 @@ Data to draw an angled line. | |||||||
|  |  | ||||||
| An angle and length with explicitly named parameters | An angle and length with explicitly named parameters | ||||||
|  |  | ||||||
| **Type:** `object` | [`PolarCoordsData`](/docs/kcl/types/PolarCoordsData) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Properties |  | ||||||
|  |  | ||||||
| | Property | Type | Description | Required | |  | ||||||
| |----------|------|-------------|----------| |  | ||||||
| | `angle` |`number`| The angle of the line (in degrees). | No | |  | ||||||
| | `length` |`number`| The length of the line. | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- | ---- | ||||||
|  | |||||||
| @ -1,23 +0,0 @@ | |||||||
| --- |  | ||||||
| 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,42 +0,0 @@ | |||||||
| --- |  | ||||||
| title: "Axis3dOrEdgeReference" |  | ||||||
| excerpt: "A 3D axis or tagged edge." |  | ||||||
| layout: manual |  | ||||||
| --- |  | ||||||
|  |  | ||||||
| A 3D axis or tagged edge. |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| **This schema accepts any of the following:** |  | ||||||
|  |  | ||||||
| 3D axis and origin. |  | ||||||
|  |  | ||||||
| [`AxisAndOrigin3d`](/docs/kcl/types/AxisAndOrigin3d) |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
| Tagged edge. |  | ||||||
|  |  | ||||||
| [`EdgeReference`](/docs/kcl/types/EdgeReference) |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -1,10 +1,10 @@ | |||||||
| --- | --- | ||||||
| title: "AxisAndOrigin2d" | title: "AxisAndOrigin" | ||||||
| excerpt: "A 2D axis and origin." | excerpt: "Axis and origin." | ||||||
| layout: manual | layout: manual | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| A 2D axis and origin. | Axis and origin. | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -1,105 +0,0 @@ | |||||||
| --- |  | ||||||
| title: "AxisAndOrigin3d" |  | ||||||
| excerpt: "A 3D axis and origin." |  | ||||||
| layout: manual |  | ||||||
| --- |  | ||||||
|  |  | ||||||
| A 3D axis and origin. |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| **This schema accepts exactly one of the following:** |  | ||||||
|  |  | ||||||
| X-axis. |  | ||||||
|  |  | ||||||
| **enum:** `X` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
| Y-axis. |  | ||||||
|  |  | ||||||
| **enum:** `Y` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
| Z-axis. |  | ||||||
|  |  | ||||||
| **enum:** `Z` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
| Flip the X-axis. |  | ||||||
|  |  | ||||||
| **enum:** `-X` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
| Flip the Y-axis. |  | ||||||
|  |  | ||||||
| **enum:** `-Y` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
| Flip the Z-axis. |  | ||||||
|  |  | ||||||
| **enum:** `-Z` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
|  |  | ||||||
| **Type:** `object` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Properties |  | ||||||
|  |  | ||||||
| | Property | Type | Description | Required | |  | ||||||
| |----------|------|-------------|----------| |  | ||||||
| | `custom` |`object`|  | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -1,19 +1,19 @@ | |||||||
| --- | --- | ||||||
| title: "Axis2dOrEdgeReference" | title: "AxisOrEdgeReference" | ||||||
| excerpt: "A 2D axis or tagged edge." | excerpt: "Axis or tagged edge." | ||||||
| layout: manual | layout: manual | ||||||
| --- | --- | ||||||
| 
 | 
 | ||||||
| A 2D axis or tagged edge. | Axis or tagged edge. | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| **This schema accepts any of the following:** | **This schema accepts any of the following:** | ||||||
| 
 | 
 | ||||||
| 2D axis and origin. | Axis and origin. | ||||||
| 
 | 
 | ||||||
| [`AxisAndOrigin2d`](/docs/kcl/types/AxisAndOrigin2d) | [`AxisAndOrigin`](/docs/kcl/types/AxisAndOrigin) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -1,23 +0,0 @@ | |||||||
| --- |  | ||||||
| title: "CircleThreePointData" |  | ||||||
| excerpt: "Data for drawing a 3-point circle" |  | ||||||
| layout: manual |  | ||||||
| --- |  | ||||||
|  |  | ||||||
| Data for drawing a 3-point circle |  | ||||||
|  |  | ||||||
| **Type:** `object` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Properties |  | ||||||
|  |  | ||||||
| | Property | Type | Description | Required | |  | ||||||
| |----------|------|-------------|----------| |  | ||||||
| | `p1` |`[number, number]`| Point one for circle derivation. | No | |  | ||||||
| | `p2` |`[number, number]`| Point two for circle derivation. | No | |  | ||||||
| | `p3` |`[number, number]`| Point three for circle derivation. | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -1,25 +0,0 @@ | |||||||
| --- |  | ||||||
| title: "Helix" |  | ||||||
| excerpt: "A helix." |  | ||||||
| layout: manual |  | ||||||
| --- |  | ||||||
|  |  | ||||||
| A helix. |  | ||||||
|  |  | ||||||
| **Type:** `object` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Properties |  | ||||||
|  |  | ||||||
| | Property | Type | Description | Required | |  | ||||||
| |----------|------|-------------|----------| |  | ||||||
| | `value` |`string`| The id of the helix. | No | |  | ||||||
| | `revolutions` |`number`| Number of revolutions. | No | |  | ||||||
| | `angleStart` |`number`| Start angle (in degrees). | No | |  | ||||||
| | `ccw` |`boolean`| Is the helix rotation counter clockwise? | No | |  | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -1,10 +1,10 @@ | |||||||
| --- | --- | ||||||
| title: "HelixData" | title: "HelixData" | ||||||
| excerpt: "Data for a helix." | excerpt: "Data for helices." | ||||||
| layout: manual | layout: manual | ||||||
| --- | --- | ||||||
|  |  | ||||||
| Data for a helix. | Data for helices. | ||||||
|  |  | ||||||
| **Type:** `object` | **Type:** `object` | ||||||
|  |  | ||||||
| @ -19,8 +19,6 @@ Data for a helix. | |||||||
| | `revolutions` |`number`| Number of revolutions. | No | | | `revolutions` |`number`| Number of revolutions. | No | | ||||||
| | `angleStart` |`number`| Start angle (in degrees). | No | | | `angleStart` |`number`| Start angle (in degrees). | No | | ||||||
| | `ccw` |`boolean`| Is the helix rotation counter clockwise? The default is `false`. | No | | | `ccw` |`boolean`| Is the helix rotation counter clockwise? The default is `false`. | No | | ||||||
| | `length` |`number`| Length of the helix. | No | | | `length` |`number`| Length of the helix. If this argument is not provided, the height of the solid is used. | No | | ||||||
| | `radius` |`number`| Radius of the helix. | No | |  | ||||||
| | `axis` |[`Axis3dOrEdgeReference`](/docs/kcl/types/Axis3dOrEdgeReference)| Axis to use as mirror. | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,24 +0,0 @@ | |||||||
| --- |  | ||||||
| title: "HelixRevolutionsData" |  | ||||||
| excerpt: "Data for helix revolutions." |  | ||||||
| layout: manual |  | ||||||
| --- |  | ||||||
|  |  | ||||||
| Data for helix revolutions. |  | ||||||
|  |  | ||||||
| **Type:** `object` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Properties |  | ||||||
|  |  | ||||||
| | Property | Type | Description | Required | |  | ||||||
| |----------|------|-------------|----------| |  | ||||||
| | `revolutions` |`number`| Number of revolutions. | No | |  | ||||||
| | `angleStart` |`number`| Start angle (in degrees). | No | |  | ||||||
| | `ccw` |`boolean`| Is the helix rotation counter clockwise? The default is `false`. | No | |  | ||||||
| | `length` |`number`| Length of the helix. If this argument is not provided, the height of the solid is used. | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -1,25 +0,0 @@ | |||||||
| --- |  | ||||||
| title: "HelixValue" |  | ||||||
| excerpt: "A helix." |  | ||||||
| layout: manual |  | ||||||
| --- |  | ||||||
|  |  | ||||||
| A helix. |  | ||||||
|  |  | ||||||
| **Type:** `object` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Properties |  | ||||||
|  |  | ||||||
| | Property | Type | Description | Required | |  | ||||||
| |----------|------|-------------|----------| |  | ||||||
| | `value` |`string`| The id of the helix. | No | |  | ||||||
| | `revolutions` |`number`| Number of revolutions. | No | |  | ||||||
| | `angleStart` |`number`| Start angle (in degrees). | No | |  | ||||||
| | `ccw` |`boolean`| Is the helix rotation counter clockwise? | No | |  | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -12,10 +12,5 @@ KCL value for an optional parameter which was not given an argument. (remember, | |||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Properties |  | ||||||
|  |  | ||||||
| | Property | Type | Description | Required | |  | ||||||
| |----------|------|-------------|----------| |  | ||||||
| | `digest` |`[, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`, `integer`]`|  | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -285,27 +285,6 @@ An solid is a collection of extrude surfaces. | |||||||
| | `value` |`[` [`Solid`](/docs/kcl/types/Solid) `]`|  | No | | | `value` |`[` [`Solid`](/docs/kcl/types/Solid) `]`|  | No | | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
| A helix. |  | ||||||
|  |  | ||||||
| **Type:** `object` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Properties |  | ||||||
|  |  | ||||||
| | Property | Type | Description | Required | |  | ||||||
| |----------|------|-------------|----------| |  | ||||||
| | `type` |enum: [`Helix`](/docs/kcl/types/Helix)|  | No | |  | ||||||
| | `value` |`string`| The id of the helix. | No | |  | ||||||
| | `revolutions` |`number`| Number of revolutions. | No | |  | ||||||
| | `angleStart` |`number`| Start angle (in degrees). | No | |  | ||||||
| | `ccw` |`boolean`| Is the helix rotation counter clockwise? | No | |  | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- | ---- | ||||||
| Data for an imported geometry. | Data for an imported geometry. | ||||||
|  |  | ||||||
| @ -350,23 +329,6 @@ Data for an imported geometry. | |||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Properties |  | ||||||
|  |  | ||||||
| | Property | Type | Description | Required | |  | ||||||
| |----------|------|-------------|----------| |  | ||||||
| | `type` |enum: `Module`|  | No | |  | ||||||
| | `value` |[`ModuleId`](/docs/kcl/types/ModuleId)| Any KCL value. | No | |  | ||||||
| | `__meta` |`[` [`Metadata`](/docs/kcl/types/Metadata) `]`|  | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
|  |  | ||||||
| **Type:** `object` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Properties | ## Properties | ||||||
|  |  | ||||||
| | Property | Type | Description | Required | | | Property | Type | Description | Required | | ||||||
|  | |||||||
| @ -16,6 +16,6 @@ Data for a mirror. | |||||||
|  |  | ||||||
| | Property | Type | Description | Required | | | Property | Type | Description | Required | | ||||||
| |----------|------|-------------|----------| | |----------|------|-------------|----------| | ||||||
| | `axis` |[`Axis2dOrEdgeReference`](/docs/kcl/types/Axis2dOrEdgeReference)| Axis to use as mirror. | No | | | `axis` |[`AxisOrEdgeReference`](/docs/kcl/types/AxisOrEdgeReference)| Axis to use as mirror. | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,16 +0,0 @@ | |||||||
| --- |  | ||||||
| title: "ModuleId" |  | ||||||
| excerpt: "Identifier of a source file.  Uses a u32 to keep the size small." |  | ||||||
| layout: manual |  | ||||||
| --- |  | ||||||
|  |  | ||||||
| Identifier of a source file.  Uses a u32 to keep the size small. |  | ||||||
|  |  | ||||||
| **Type:** `integer` (`uint32`) |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -17,7 +17,7 @@ Data for revolution surfaces. | |||||||
| | Property | Type | Description | Required | | | Property | Type | Description | Required | | ||||||
| |----------|------|-------------|----------| | |----------|------|-------------|----------| | ||||||
| | `angle` |`number` (**maximum:** 360.0) (**minimum:** -360.0)| Angle to revolve (in degrees). Default is 360. | No | | | `angle` |`number` (**maximum:** 360.0) (**minimum:** -360.0)| Angle to revolve (in degrees). Default is 360. | No | | ||||||
| | `axis` |[`Axis2dOrEdgeReference`](/docs/kcl/types/Axis2dOrEdgeReference)| Axis of revolution. | No | | | `axis` |[`AxisOrEdgeReference`](/docs/kcl/types/AxisOrEdgeReference)| Axis of revolution. | No | | ||||||
| | `tolerance` |`number`| Tolerance for the revolve operation. | No | | | `tolerance` |`number`| Tolerance for the revolve operation. | No | | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,23 +0,0 @@ | |||||||
| --- |  | ||||||
| title: "SweepData" |  | ||||||
| excerpt: "Data for a sweep." |  | ||||||
| layout: manual |  | ||||||
| --- |  | ||||||
|  |  | ||||||
| Data for a sweep. |  | ||||||
|  |  | ||||||
| **Type:** `object` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Properties |  | ||||||
|  |  | ||||||
| | Property | Type | Description | Required | |  | ||||||
| |----------|------|-------------|----------| |  | ||||||
| | `path` |[`SweepPath`](/docs/kcl/types/SweepPath)| The path to sweep along. | No | |  | ||||||
| | `sectional` |`boolean`| If true, the sweep will be broken up into sub-sweeps (extrusions, revolves, sweeps) based on the trajectory path components. | No | |  | ||||||
| | `tolerance` |`number`| Tolerance for the sweep operation. | No | |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -1,42 +0,0 @@ | |||||||
| --- |  | ||||||
| title: "SweepPath" |  | ||||||
| excerpt: "A path to sweep along." |  | ||||||
| layout: manual |  | ||||||
| --- |  | ||||||
|  |  | ||||||
| A path to sweep along. |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| **This schema accepts any of the following:** |  | ||||||
|  |  | ||||||
| A path to sweep along. |  | ||||||
|  |  | ||||||
| [`Sketch`](/docs/kcl/types/Sketch) |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
| A path to sweep along. |  | ||||||
|  |  | ||||||
| [`Helix`](/docs/kcl/types/Helix) |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ---- |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -1,11 +1,22 @@ | |||||||
| import { test, expect } from './zoo-test' | import { test, expect } from '@playwright/test' | ||||||
|  |  | ||||||
|  | import { setupElectron, tearDown } from './test-utils' | ||||||
|  |  | ||||||
|  | test.afterEach(async ({ page }, testInfo) => { | ||||||
|  |   await tearDown(page, testInfo) | ||||||
|  | }) | ||||||
|  |  | ||||||
| test.describe('Electron app header tests', () => { | test.describe('Electron app header tests', () => { | ||||||
|   test( |   test( | ||||||
|     'Open Command Palette button has correct shortcut', |     'Open Command Palette button has correct shortcut', | ||||||
|     { tag: '@electron' }, |     { tag: '@electron' }, | ||||||
|     async ({ page }, testInfo) => { |     async ({ browserName }, testInfo) => { | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |       const { electronApp, page } = await setupElectron({ | ||||||
|  |         testInfo, | ||||||
|  |         folderSetupFn: async () => {}, | ||||||
|  |       }) | ||||||
|  |  | ||||||
|  |       await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|       // No space before the shortcut since it checks textContent. |       // No space before the shortcut since it checks textContent. | ||||||
|       let text |       let text | ||||||
| @ -23,14 +34,21 @@ test.describe('Electron app header tests', () => { | |||||||
|       const commandsButton = page.getByRole('button', { name: 'Commands' }) |       const commandsButton = page.getByRole('button', { name: 'Commands' }) | ||||||
|       await expect(commandsButton).toBeVisible() |       await expect(commandsButton).toBeVisible() | ||||||
|       await expect(commandsButton).toHaveText(text) |       await expect(commandsButton).toHaveText(text) | ||||||
|  |  | ||||||
|  |       await electronApp.close() | ||||||
|     } |     } | ||||||
|   ) |   ) | ||||||
|  |  | ||||||
|   test( |   test( | ||||||
|     'User settings has correct shortcut', |     'User settings has correct shortcut', | ||||||
|     { tag: '@electron' }, |     { tag: '@electron' }, | ||||||
|     async ({ page }, testInfo) => { |     async ({ browserName }, testInfo) => { | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |       const { electronApp, page } = await setupElectron({ | ||||||
|  |         testInfo, | ||||||
|  |         folderSetupFn: async () => {}, | ||||||
|  |       }) | ||||||
|  |  | ||||||
|  |       await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|       // Open the user sidebar menu. |       // Open the user sidebar menu. | ||||||
|       await page.getByTestId('user-sidebar-toggle').click() |       await page.getByTestId('user-sidebar-toggle').click() | ||||||
| @ -41,6 +59,8 @@ test.describe('Electron app header tests', () => { | |||||||
|       const userSettingsButton = page.getByTestId('user-settings') |       const userSettingsButton = page.getByTestId('user-settings') | ||||||
|       await expect(userSettingsButton).toBeVisible() |       await expect(userSettingsButton).toBeVisible() | ||||||
|       await expect(userSettingsButton).toHaveText(text) |       await expect(userSettingsButton).toHaveText(text) | ||||||
|  |  | ||||||
|  |       await electronApp.close() | ||||||
|     } |     } | ||||||
|   ) |   ) | ||||||
| }) | }) | ||||||
|  | |||||||
| @ -1,26 +1,29 @@ | |||||||
| import { test, expect, Page } from './zoo-test' | import { test, expect, Page } from '@playwright/test' | ||||||
| import { | import { | ||||||
|   getUtils, |   getUtils, | ||||||
|   TEST_COLORS, |   TEST_COLORS, | ||||||
|  |   setup, | ||||||
|  |   tearDown, | ||||||
|   commonPoints, |   commonPoints, | ||||||
|   PERSIST_MODELING_CONTEXT, |   PERSIST_MODELING_CONTEXT, | ||||||
| } from './test-utils' | } from './test-utils' | ||||||
| import { HomePageFixture } from './fixtures/homePageFixture' |  | ||||||
|  | test.beforeEach(async ({ context, page }, testInfo) => { | ||||||
|  |   await setup(context, page, testInfo) | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | test.afterEach(async ({ page }, testInfo) => { | ||||||
|  |   await tearDown(page, testInfo) | ||||||
|  | }) | ||||||
|  |  | ||||||
| test.setTimeout(120000) | test.setTimeout(120000) | ||||||
|  |  | ||||||
| async function doBasicSketch( | async function doBasicSketch(page: Page, openPanes: string[]) { | ||||||
|   page: Page, |  | ||||||
|   homePage: HomePageFixture, |  | ||||||
|   openPanes: string[] |  | ||||||
| ) { |  | ||||||
|   const u = await getUtils(page) |   const u = await getUtils(page) | ||||||
|   await page.setBodyDimensions({ width: 1200, height: 500 }) |   await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|   const PUR = 400 / 37.5 //pixeltoUnitRatio |   const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|  |  | ||||||
|   await homePage.goToModelingScene() |   await u.waitForAuthSkipAppStart() | ||||||
|   await u.waitForPageLoad() |  | ||||||
|   await page.waitForTimeout(1000) |  | ||||||
|   await u.openDebugPanel() |   await u.openDebugPanel() | ||||||
|  |  | ||||||
|   // If we have the code pane open, we should see the code. |   // If we have the code pane open, we should see the code. | ||||||
| @ -145,11 +148,13 @@ async function doBasicSketch( | |||||||
| } | } | ||||||
|  |  | ||||||
| test.describe('Basic sketch', () => { | test.describe('Basic sketch', () => { | ||||||
|   test.fixme('code pane open at start', async ({ page, homePage }) => { |   test('code pane open at start', { tag: ['@skipWin'] }, async ({ page }) => { | ||||||
|     await doBasicSketch(page, homePage, ['code']) |     // Skip on windows it is being weird. | ||||||
|  |     test.skip(process.platform === 'win32', 'Skip on windows') | ||||||
|  |     await doBasicSketch(page, ['code']) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('code pane closed at start', async ({ page, homePage }) => { |   test('code pane closed at start', async ({ page }) => { | ||||||
|     // Load the app with the code panes |     // Load the app with the code panes | ||||||
|     await page.addInitScript(async (persistModelingContext) => { |     await page.addInitScript(async (persistModelingContext) => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
| @ -157,6 +162,6 @@ test.describe('Basic sketch', () => { | |||||||
|         JSON.stringify({ openPanes: [] }) |         JSON.stringify({ openPanes: [] }) | ||||||
|       ) |       ) | ||||||
|     }, PERSIST_MODELING_CONTEXT) |     }, PERSIST_MODELING_CONTEXT) | ||||||
|     await doBasicSketch(page, homePage, []) |     await doBasicSketch(page, []) | ||||||
|   }) |   }) | ||||||
| }) | }) | ||||||
|  | |||||||
| @ -1,21 +1,27 @@ | |||||||
| import { test, expect, Page } from './zoo-test' | import { test, expect } from '@playwright/test' | ||||||
| import { HomePageFixture } from './fixtures/homePageFixture' | import { getUtils, setup, tearDown } from './test-utils' | ||||||
| import { getUtils } from './test-utils' |  | ||||||
| import { EngineCommand } from 'lang/std/artifactGraph' | import { EngineCommand } from 'lang/std/artifactGraph' | ||||||
| import { uuidv4 } from 'lib/utils' | 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', () => { | test.describe('Can create sketches on all planes and their back sides', () => { | ||||||
|   const sketchOnPlaneAndBackSideTest = async ( |   const sketchOnPlaneAndBackSideTest = async ( | ||||||
|     page: Page, |     page: any, | ||||||
|     homePage: HomePageFixture, |  | ||||||
|     plane: string, |     plane: string, | ||||||
|     clickCoords: { x: number; y: number } |     clickCoords: { x: number; y: number } | ||||||
|   ) => { |   ) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     const PUR = 400 / 37.5 //pixeltoUnitRatio |     const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|     await u.openDebugPanel() |     await u.openDebugPanel() | ||||||
|  |  | ||||||
|     const coord = |     const coord = | ||||||
| @ -77,39 +83,32 @@ test.describe('Can create sketches on all planes and their back sides', () => { | |||||||
|     await u.clearCommandLogs() |     await u.clearCommandLogs() | ||||||
|     await u.removeCurrentCode() |     await u.removeCurrentCode() | ||||||
|   } |   } | ||||||
|   test('XY', async ({ page, homePage }) => { |   test('XY', async ({ page }) => { | ||||||
|     await sketchOnPlaneAndBackSideTest( |     await sketchOnPlaneAndBackSideTest( | ||||||
|       page, |       page, | ||||||
|       homePage, |  | ||||||
|       'XY', |       'XY', | ||||||
|       { x: 600, y: 388 } // red plane |       { 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. |       // { 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, homePage }) => { |   test('YZ', async ({ page }) => { | ||||||
|     await sketchOnPlaneAndBackSideTest(page, homePage, 'YZ', { x: 700, y: 250 }) // green plane |     await sketchOnPlaneAndBackSideTest(page, 'YZ', { x: 700, y: 250 }) // green plane | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('XZ', async ({ page, homePage }) => { |   test('XZ', async ({ page }) => { | ||||||
|     await sketchOnPlaneAndBackSideTest(page, homePage, '-XZ', { x: 700, y: 80 }) // blue plane |     await sketchOnPlaneAndBackSideTest(page, '-XZ', { x: 700, y: 80 }) // blue plane | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('-XY', async ({ page, homePage }) => { |   test('-XY', async ({ page }) => { | ||||||
|     await sketchOnPlaneAndBackSideTest(page, homePage, '-XY', { |     await sketchOnPlaneAndBackSideTest(page, '-XY', { x: 600, y: 118 }) // back of red plane | ||||||
|       x: 600, |  | ||||||
|       y: 118, |  | ||||||
|     }) // back of red plane |  | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('-YZ', async ({ page, homePage }) => { |   test('-YZ', async ({ page }) => { | ||||||
|     await sketchOnPlaneAndBackSideTest(page, homePage, '-YZ', { |     await sketchOnPlaneAndBackSideTest(page, '-YZ', { x: 700, y: 219 }) // back of green plane | ||||||
|       x: 700, |  | ||||||
|       y: 219, |  | ||||||
|     }) // back of green plan |  | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('-XZ', async ({ page, homePage }) => { |   test('-XZ', async ({ page }) => { | ||||||
|     await sketchOnPlaneAndBackSideTest(page, homePage, 'XZ', { x: 700, y: 427 }) // back of blue plane |     await sketchOnPlaneAndBackSideTest(page, 'XZ', { x: 700, y: 427 }) // back of blue plane | ||||||
|   }) |   }) | ||||||
| }) | }) | ||||||
|  | |||||||
| @ -1,15 +1,28 @@ | |||||||
| import { test, expect } from './zoo-test' | import { test, expect } from '@playwright/test' | ||||||
|  |  | ||||||
| import { getUtils, executorInputPath } from './test-utils' | import { | ||||||
|  |   getUtils, | ||||||
|  |   setup, | ||||||
|  |   setupElectron, | ||||||
|  |   tearDown, | ||||||
|  |   executorInputPath, | ||||||
|  | } from './test-utils' | ||||||
| import { join } from 'path' | import { join } from 'path' | ||||||
| import { bracket } from 'lib/exampleKcl' | import { bracket } from 'lib/exampleKcl' | ||||||
| import { TEST_CODE_LONG_WITH_ERROR_OUT_OF_VIEW } from './storageStates' | import { TEST_CODE_LONG_WITH_ERROR_OUT_OF_VIEW } from './storageStates' | ||||||
| import fsp from 'fs/promises' | 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.describe('Code pane and errors', () => { | ||||||
|   test('Typing KCL errors induces a badge on the code pane button', async ({ |   test('Typing KCL errors induces a badge on the code pane button', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|  |  | ||||||
| @ -18,18 +31,18 @@ test.describe('Code pane and errors', () => { | |||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `// Extruded Triangle |         `// Extruded Triangle | ||||||
|   sketch001 = startSketchOn('XZ') | sketch001 = startSketchOn('XZ') | ||||||
|     |> startProfileAt([0, 0], %) |   |> startProfileAt([0, 0], %) | ||||||
|     |> line([10, 0], %) |   |> line([10, 0], %) | ||||||
|     |> line([-5, 10], %) |   |> line([-5, 10], %) | ||||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) |   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|     |> close(%) |   |> close(%) | ||||||
|   extrude001 = extrude(5, sketch001)` | extrude001 = extrude(5, sketch001)` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     // wait for execution done |     // wait for execution done | ||||||
|     await u.openDebugPanel() |     await u.openDebugPanel() | ||||||
| @ -49,11 +62,11 @@ test.describe('Code pane and errors', () => { | |||||||
|     await expect(codePaneButtonHolder).toContainText('notification') |     await expect(codePaneButtonHolder).toContainText('notification') | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test.skip('Opening and closing the code pane will consistently show error diagnostics', async ({ |   test('Opening and closing the code pane will consistently show error diagnostics', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|     editor, |  | ||||||
|   }) => { |   }) => { | ||||||
|  |     await page.goto('http://localhost:3000') | ||||||
|  |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|  |  | ||||||
|     // Load the app with the working starter code |     // Load the app with the working starter code | ||||||
| @ -61,8 +74,8 @@ test.describe('Code pane and errors', () => { | |||||||
|       localStorage.setItem('persistCode', code) |       localStorage.setItem('persistCode', code) | ||||||
|     }, bracket) |     }, bracket) | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 900 }) |     await page.setViewportSize({ width: 1200, height: 900 }) | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     // wait for execution done |     // wait for execution done | ||||||
|     await u.openDebugPanel() |     await u.openDebugPanel() | ||||||
| @ -78,9 +91,8 @@ test.describe('Code pane and errors', () => { | |||||||
|     await expect(codePaneButtonHolder).not.toContainText('notification') |     await expect(codePaneButtonHolder).not.toContainText('notification') | ||||||
|  |  | ||||||
|     // Delete a character to break the KCL |     // Delete a character to break the KCL | ||||||
|     await editor.openPane() |     await u.openKclCodePanel() | ||||||
|     await editor.scrollToText('thickness, bracketLeg1Sketch)') |     await page.getByText('thickness, bracketLeg1Sketch)').click() | ||||||
|     await page.getByText('extrude(thickness, bracketLeg1Sketch)').click() |  | ||||||
|     await page.keyboard.press('Backspace') |     await page.keyboard.press('Backspace') | ||||||
|  |  | ||||||
|     // Ensure that a badge appears on the button |     // Ensure that a badge appears on the button | ||||||
| @ -104,10 +116,7 @@ test.describe('Code pane and errors', () => { | |||||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() |     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||||
|  |  | ||||||
|     // Open the code pane |     // Open the code pane | ||||||
|     await editor.openPane() |     await u.openKclCodePanel() | ||||||
|  |  | ||||||
|     // Go to our problematic code again (missing closing paren!) |  | ||||||
|     await editor.scrollToText('extrude(thickness, bracketLeg1Sketch') |  | ||||||
|  |  | ||||||
|     // Ensure that a badge appears on the button |     // Ensure that a badge appears on the button | ||||||
|     await expect(codePaneButtonHolder).toContainText('notification') |     await expect(codePaneButtonHolder).toContainText('notification') | ||||||
| @ -120,58 +129,59 @@ test.describe('Code pane and errors', () => { | |||||||
|     await expect(page.locator('.cm-tooltip').first()).toBeVisible() |     await expect(page.locator('.cm-tooltip').first()).toBeVisible() | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test.fixme( |   test('When error is not in view you can click the badge to scroll to it', async ({ | ||||||
|     '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) |  | ||||||
|  |  | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |  | ||||||
|       await homePage.goToModelingScene() |  | ||||||
|  |  | ||||||
|       await page.waitForTimeout(1000) |  | ||||||
|  |  | ||||||
|       // 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() |  | ||||||
|  |  | ||||||
|       // 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() |  | ||||||
|  |  | ||||||
|       // 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, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|  |     const u = await getUtils(page) | ||||||
|  |  | ||||||
|     // Load the app with the working starter code |     // Load the app with the working starter code | ||||||
|     await context.addInitScript((code) => { |     await page.addInitScript((code) => { | ||||||
|       localStorage.setItem('persistCode', code) |       localStorage.setItem('persistCode', code) | ||||||
|     }, TEST_CODE_LONG_WITH_ERROR_OUT_OF_VIEW) |     }, TEST_CODE_LONG_WITH_ERROR_OUT_OF_VIEW) | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|  |     await page.waitForTimeout(1000) | ||||||
|  |  | ||||||
|  |     // 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() | ||||||
|  |  | ||||||
|  |     // 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() | ||||||
|  |  | ||||||
|  |     // 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() | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   test('When error is not in view WITH LINTS you can click the badge to scroll to it', async ({ | ||||||
|  |     page, | ||||||
|  |   }) => { | ||||||
|  |     const u = await getUtils(page) | ||||||
|  |  | ||||||
|  |     // 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.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await page.waitForTimeout(1000) |     await page.waitForTimeout(1000) | ||||||
|  |  | ||||||
| @ -231,29 +241,32 @@ test.describe('Code pane and errors', () => { | |||||||
| test( | test( | ||||||
|   'Opening multiple panes persists when switching projects', |   'Opening multiple panes persists when switching projects', | ||||||
|   { tag: '@electron' }, |   { tag: '@electron' }, | ||||||
|   async ({ context, page }, testInfo) => { |   async ({ browserName }, testInfo) => { | ||||||
|     // Setup multiple projects. |     // Setup multiple projects. | ||||||
|     await context.folderSetupFn(async (dir) => { |     const { electronApp, page } = await setupElectron({ | ||||||
|       const routerTemplateDir = join(dir, 'router-template-slate') |       testInfo, | ||||||
|       const bracketDir = join(dir, 'bracket') |       folderSetupFn: async (dir) => { | ||||||
|       await Promise.all([ |         const routerTemplateDir = join(dir, 'router-template-slate') | ||||||
|         fsp.mkdir(routerTemplateDir, { recursive: true }), |         const bracketDir = join(dir, 'bracket') | ||||||
|         fsp.mkdir(bracketDir, { recursive: true }), |         await Promise.all([ | ||||||
|       ]) |           fsp.mkdir(routerTemplateDir, { recursive: true }), | ||||||
|       await Promise.all([ |           fsp.mkdir(bracketDir, { recursive: true }), | ||||||
|         fsp.copyFile( |         ]) | ||||||
|           executorInputPath('router-template-slate.kcl'), |         await Promise.all([ | ||||||
|           join(routerTemplateDir, 'main.kcl') |           fsp.copyFile( | ||||||
|         ), |             executorInputPath('router-template-slate.kcl'), | ||||||
|         fsp.copyFile( |             join(routerTemplateDir, 'main.kcl') | ||||||
|           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), |           ), | ||||||
|           join(bracketDir, 'main.kcl') |           fsp.copyFile( | ||||||
|         ), |             executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||||
|       ]) |             join(bracketDir, 'main.kcl') | ||||||
|  |           ), | ||||||
|  |         ]) | ||||||
|  |       }, | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await test.step('Opening the bracket project should load', async () => { |     await test.step('Opening the bracket project should load', async () => { | ||||||
|       await expect(page.getByText('bracket')).toBeVisible() |       await expect(page.getByText('bracket')).toBeVisible() | ||||||
| @ -296,21 +309,30 @@ test( | |||||||
|       await expect(page.locator('#variables-pane')).toBeVisible() |       await expect(page.locator('#variables-pane')).toBeVisible() | ||||||
|       await expect(page.locator('#logs-pane')).toBeVisible() |       await expect(page.locator('#logs-pane')).toBeVisible() | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|  |     await electronApp.close() | ||||||
|   } |   } | ||||||
| ) | ) | ||||||
|  |  | ||||||
| test( | test( | ||||||
|   'external change of file contents are reflected in editor', |   'external change of file contents are reflected in editor', | ||||||
|   { tag: '@electron' }, |   { tag: '@electron' }, | ||||||
|   async ({ context, page }, testInfo) => { |   async ({ browserName }, testInfo) => { | ||||||
|     const PROJECT_DIR_NAME = 'lee-was-here' |     const PROJECT_DIR_NAME = 'lee-was-here' | ||||||
|     const { dir: projectsDir } = await context.folderSetupFn(async (dir) => { |     const { | ||||||
|       const aProjectDir = join(dir, PROJECT_DIR_NAME) |       electronApp, | ||||||
|       await fsp.mkdir(aProjectDir, { recursive: true }) |       page, | ||||||
|  |       dir: projectsDir, | ||||||
|  |     } = await setupElectron({ | ||||||
|  |       testInfo, | ||||||
|  |       folderSetupFn: async (dir) => { | ||||||
|  |         const aProjectDir = join(dir, PROJECT_DIR_NAME) | ||||||
|  |         await fsp.mkdir(aProjectDir, { recursive: true }) | ||||||
|  |       }, | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await test.step('Open the project', async () => { |     await test.step('Open the project', async () => { | ||||||
|       await expect(page.getByText(PROJECT_DIR_NAME)).toBeVisible() |       await expect(page.getByText(PROJECT_DIR_NAME)).toBeVisible() | ||||||
| @ -329,5 +351,7 @@ test( | |||||||
|       ) |       ) | ||||||
|       await u.editorTextMatches(content) |       await u.editorTextMatches(content) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|  |     await electronApp.close() | ||||||
|   } |   } | ||||||
| ) | ) | ||||||
|  | |||||||
| @ -1,30 +1,37 @@ | |||||||
| import { test, expect } from './zoo-test' | import { test, expect } from '@playwright/test' | ||||||
|  |  | ||||||
| import { getUtils } from './test-utils' | import { getUtils, setup, tearDown } from './test-utils' | ||||||
| import { KCL_DEFAULT_LENGTH } from 'lib/constants' | 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.describe('Command bar tests', () => { | ||||||
|   test('Extrude from command bar selects extrude line after', async ({ |   test('Extrude from command bar selects extrude line after', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `sketch001 = startSketchOn('XY') |         `sketch001 = startSketchOn('XY') | ||||||
|   |> startProfileAt([-10, -10], %) |     |> startProfileAt([-10, -10], %) | ||||||
|   |> line([20, 0], %) |     |> line([20, 0], %) | ||||||
|   |> line([0, 20], %) |     |> line([0, 20], %) | ||||||
|   |> xLine(-20, %) |     |> xLine(-20, %) | ||||||
|   |> close(%) |     |> close(%) | ||||||
|     ` |       ` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await u.openDebugPanel() |     await u.openDebugPanel() | ||||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') |     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||||
| @ -45,25 +52,24 @@ test.describe('Command bar tests', () => { | |||||||
|     ) |     ) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   // TODO: fix this test after the electron migration |   test('Fillet from command bar', async ({ page }) => { | ||||||
|   test.fixme('Fillet from command bar', async ({ page, homePage }) => { |  | ||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `sketch001 = startSketchOn('XY') |         `sketch001 = startSketchOn('XY') | ||||||
|     |> startProfileAt([-5, -5], %) |   |> startProfileAt([-5, -5], %) | ||||||
|     |> line([0, 10], %) |   |> line([0, 10], %) | ||||||
|     |> line([10, 0], %) |   |> line([10, 0], %) | ||||||
|     |> line([0, -10], %) |   |> line([0, -10], %) | ||||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) |   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|     |> close(%) |   |> close(%) | ||||||
|   extrude001 = extrude(-10, sketch001)` | extrude001 = extrude(-10, sketch001)` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|     await u.openDebugPanel() |     await u.openDebugPanel() | ||||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') |     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||||
|     await u.closeDebugPanel() |     await u.closeDebugPanel() | ||||||
| @ -87,10 +93,10 @@ test.describe('Command bar tests', () => { | |||||||
|  |  | ||||||
|   test('Command bar can change a setting, and switch back and forth between arguments', async ({ |   test('Command bar can change a setting, and switch back and forth between arguments', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     const u = await getUtils(page) | ||||||
|     await homePage.goToModelingScene() |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     const commandBarButton = page.getByRole('button', { name: 'Commands' }) |     const commandBarButton = page.getByRole('button', { name: 'Commands' }) | ||||||
|     const cmdSearchBar = page.getByPlaceholder('Search commands') |     const cmdSearchBar = page.getByPlaceholder('Search commands') | ||||||
| @ -147,7 +153,7 @@ test.describe('Command bar tests', () => { | |||||||
|     // Check that the visibility changed |     // Check that the visibility changed | ||||||
|     await expect(paneSelector).not.toBeVisible() |     await expect(paneSelector).not.toBeVisible() | ||||||
|  |  | ||||||
|     commandOptionInput = page.locator('[id="option-input"]') |     commandOptionInput = page.getByPlaceholder('off') | ||||||
|  |  | ||||||
|     // Test case for https://github.com/KittyCAD/modeling-app/issues/2882 |     // Test case for https://github.com/KittyCAD/modeling-app/issues/2882 | ||||||
|     await commandBarButton.click() |     await commandBarButton.click() | ||||||
| @ -168,10 +174,10 @@ test.describe('Command bar tests', () => { | |||||||
|  |  | ||||||
|   test('Command bar keybinding works from code editor and can change a setting', async ({ |   test('Command bar keybinding works from code editor and can change a setting', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     const u = await getUtils(page) | ||||||
|     await homePage.goToModelingScene() |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await expect( |     await expect( | ||||||
|       page.getByRole('button', { name: 'Start Sketch' }) |       page.getByRole('button', { name: 'Start Sketch' }) | ||||||
| @ -215,25 +221,25 @@ test.describe('Command bar tests', () => { | |||||||
|     await expect(page.locator('body')).not.toHaveClass(`body-bg dark`) |     await expect(page.locator('body')).not.toHaveClass(`body-bg dark`) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('Can extrude from the command bar', async ({ page, homePage }) => { |   test('Can extrude from the command bar', async ({ page }) => { | ||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `distance = sqrt(20) |         `distance = sqrt(20) | ||||||
|     sketch001 = startSketchOn('XZ') |       sketch001 = startSketchOn('XZ') | ||||||
|     |> startProfileAt([-6.95, 10.98], %) |       |> startProfileAt([-6.95, 10.98], %) | ||||||
|     |> line([25.1, 0.41], %) |       |> line([25.1, 0.41], %) | ||||||
|     |> line([0.73, -20.93], %) |       |> line([0.73, -20.93], %) | ||||||
|     |> line([-23.44, 0.52], %) |       |> line([-23.44, 0.52], %) | ||||||
|     |> close(%) |       |> close(%) | ||||||
|         ` |           ` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     // Make sure the stream is up |     // Make sure the stream is up | ||||||
|     await u.openDebugPanel() |     await u.openDebugPanel() | ||||||
| @ -287,19 +293,26 @@ test.describe('Command bar tests', () => { | |||||||
|     await continueButton.click() |     await continueButton.click() | ||||||
|     await submitButton.click() |     await submitButton.click() | ||||||
|  |  | ||||||
|  |     // Check that the code was updated | ||||||
|     await u.waitForCmdReceive('extrude') |     await u.waitForCmdReceive('extrude') | ||||||
|  |     // Unfortunately this indentation seems to matter for the test | ||||||
|     await expect(page.locator('.cm-content')).toContainText( |     await expect(page.locator('.cm-content')).toHaveText( | ||||||
|       'extrude001 = extrude(distance001, sketch001)' |       `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 | ||||||
|     ) |     ) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('Can switch between sketch tools via command bar', async ({ |   test('Can switch between sketch tools via command bar', async ({ page }) => { | ||||||
|     page, |     const u = await getUtils(page) | ||||||
|     homePage, |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|   }) => { |     await u.waitForAuthSkipAppStart() | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |  | ||||||
|     await homePage.goToModelingScene() |  | ||||||
|  |  | ||||||
|     const sketchButton = page.getByRole('button', { name: 'Start Sketch' }) |     const sketchButton = page.getByRole('button', { name: 'Start Sketch' }) | ||||||
|     const cmdBarButton = page.getByRole('button', { name: 'Commands' }) |     const cmdBarButton = page.getByRole('button', { name: 'Commands' }) | ||||||
|  | |||||||
| @ -1,16 +1,23 @@ | |||||||
| import { test, expect } from './zoo-test' | import { test, expect } from '@playwright/test' | ||||||
| import { getUtils } from './test-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) | ||||||
|  | }) | ||||||
| test.describe('Copilot ghost text', () => { | test.describe('Copilot ghost text', () => { | ||||||
|   // eslint-disable-next-line jest/valid-title |   // eslint-disable-next-line jest/valid-title | ||||||
|   test.skip(true, 'Needs to get covered again') |   test.skip(true, 'Needs to get covered again') | ||||||
|  |  | ||||||
|   test('completes code in empty file', async ({ page, homePage }) => { |   test('completes code in empty file', async ({ page }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
| @ -45,13 +52,12 @@ test.describe('Copilot ghost text', () => { | |||||||
|  |  | ||||||
|   test.skip('copilot disabled in sketch mode no select plane', async ({ |   test.skip('copilot disabled in sketch mode no select plane', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
| @ -95,13 +101,12 @@ test.describe('Copilot ghost text', () => { | |||||||
|  |  | ||||||
|   test('copilot disabled in sketch mode after selecting plane', async ({ |   test('copilot disabled in sketch mode after selecting plane', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
| @ -179,12 +184,12 @@ test.describe('Copilot ghost text', () => { | |||||||
|     await expect(page.locator('.cm-ghostText')).not.toBeVisible() |     await expect(page.locator('.cm-ghostText')).not.toBeVisible() | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('ArrowUp in code rejects the suggestion', async ({ page, homePage }) => { |   test('ArrowUp in code rejects the suggestion', async ({ page }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
| @ -207,15 +212,12 @@ test.describe('Copilot ghost text', () => { | |||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('ArrowDown in code rejects the suggestion', async ({ |   test('ArrowDown in code rejects the suggestion', async ({ page }) => { | ||||||
|     page, |  | ||||||
|     homePage, |  | ||||||
|   }) => { |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
| @ -238,15 +240,12 @@ test.describe('Copilot ghost text', () => { | |||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('ArrowLeft in code rejects the suggestion', async ({ |   test('ArrowLeft in code rejects the suggestion', async ({ page }) => { | ||||||
|     page, |  | ||||||
|     homePage, |  | ||||||
|   }) => { |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
| @ -269,15 +268,12 @@ test.describe('Copilot ghost text', () => { | |||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('ArrowRight in code rejects the suggestion', async ({ |   test('ArrowRight in code rejects the suggestion', async ({ page }) => { | ||||||
|     page, |  | ||||||
|     homePage, |  | ||||||
|   }) => { |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
| @ -300,12 +296,12 @@ test.describe('Copilot ghost text', () => { | |||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('Enter in code scoots it down', async ({ page, homePage }) => { |   test('Enter in code scoots it down', async ({ page }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
| @ -330,15 +326,12 @@ test.describe('Copilot ghost text', () => { | |||||||
|     ) |     ) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('Ctrl+shift+z in code rejects the suggestion', async ({ |   test('Ctrl+shift+z in code rejects the suggestion', async ({ page }) => { | ||||||
|     page, |  | ||||||
|     homePage, |  | ||||||
|   }) => { |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
| @ -367,13 +360,12 @@ test.describe('Copilot ghost text', () => { | |||||||
|  |  | ||||||
|   test('Ctrl+z in code rejects the suggestion and undos the last code', async ({ |   test('Ctrl+z in code rejects the suggestion and undos the last code', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     // const PUR = 400 / 37.5 //pixeltoUnitRatio |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await page.waitForTimeout(800) |     await page.waitForTimeout(800) | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
| @ -428,107 +420,98 @@ test.describe('Copilot ghost text', () => { | |||||||
|     await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() |     await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() | ||||||
|  |  | ||||||
|     // TODO when we make codemirror a widget, we can test this. |     // 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 ({ |   test('delete in code rejects the suggestion', async ({ page }) => { | ||||||
|       page, |     const u = await getUtils(page) | ||||||
|       homePage, |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     }) => { |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|       const u = await getUtils(page) |  | ||||||
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio |  | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |  | ||||||
|  |  | ||||||
|       await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|       await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|       await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
|  |  | ||||||
|       await expect(page.locator('.cm-ghostText')).not.toBeVisible() |     await expect(page.locator('.cm-ghostText')).not.toBeVisible() | ||||||
|       await page.waitForTimeout(500) |     await page.waitForTimeout(500) | ||||||
|       await page.keyboard.press('Enter') |     await page.keyboard.press('Enter') | ||||||
|       await page.keyboard.press('Enter') |     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-ghostText').first()).toBeVisible() | ||||||
|       await expect(page.locator('.cm-content')).toHaveText( |     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, %)` |       `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( |     await expect(page.locator('.cm-ghostText').first()).toHaveText( | ||||||
|         `fn cube = (pos, scale) => {` |       `fn cube = (pos, scale) => {` | ||||||
|       ) |     ) | ||||||
|  |  | ||||||
|       // Going elsewhere in the code should hide the ghost text. |     // Going elsewhere in the code should hide the ghost text. | ||||||
|       await page.keyboard.press('Delete') |     await page.keyboard.press('Delete') | ||||||
|       await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() |     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 ({ |   test('backspace in code rejects the suggestion', async ({ page }) => { | ||||||
|       page, |     const u = await getUtils(page) | ||||||
|       homePage, |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     }) => { |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|       const u = await getUtils(page) |  | ||||||
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio |  | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |  | ||||||
|  |  | ||||||
|       await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|       await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|       await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
|  |  | ||||||
|       await expect(page.locator('.cm-ghostText')).not.toBeVisible() |     await expect(page.locator('.cm-ghostText')).not.toBeVisible() | ||||||
|       await page.waitForTimeout(500) |     await page.waitForTimeout(500) | ||||||
|       await page.keyboard.press('Enter') |     await page.keyboard.press('Enter') | ||||||
|       await page.keyboard.press('Enter') |     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-ghostText').first()).toBeVisible() | ||||||
|       await expect(page.locator('.cm-content')).toHaveText( |     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, %)` |       `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( |     await expect(page.locator('.cm-ghostText').first()).toHaveText( | ||||||
|         `fn cube = (pos, scale) => {` |       `fn cube = (pos, scale) => {` | ||||||
|       ) |     ) | ||||||
|  |  | ||||||
|       // Going elsewhere in the code should hide the ghost text. |     // Going elsewhere in the code should hide the ghost text. | ||||||
|       await page.keyboard.press('Backspace') |     await page.keyboard.press('Backspace') | ||||||
|       await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() |     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 ({ |   test('focus outside code pane rejects the suggestion', async ({ page }) => { | ||||||
|       page, |     const u = await getUtils(page) | ||||||
|       homePage, |     // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|     }) => { |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|       const u = await getUtils(page) |  | ||||||
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio |  | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |  | ||||||
|  |  | ||||||
|       await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|       await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|       await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
|  |  | ||||||
|       await expect(page.locator('.cm-ghostText')).not.toBeVisible() |     await expect(page.locator('.cm-ghostText')).not.toBeVisible() | ||||||
|       await page.waitForTimeout(500) |     await page.waitForTimeout(500) | ||||||
|       await page.keyboard.press('Enter') |     await page.keyboard.press('Enter') | ||||||
|       await expect(page.locator('.cm-ghostText').first()).toBeVisible() |     await expect(page.locator('.cm-ghostText').first()).toBeVisible() | ||||||
|       await expect(page.locator('.cm-content')).toHaveText( |     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, %)` |       `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( |     await expect(page.locator('.cm-ghostText').first()).toHaveText( | ||||||
|         `fn cube = (pos, scale) => {` |       `fn cube = (pos, scale) => {` | ||||||
|       ) |     ) | ||||||
|  |  | ||||||
|       // Going outside the editor should hide the ghost text. |     // Going outside the editor should hide the ghost text. | ||||||
|       await page.mouse.move(0, 0) |     await page.mouse.move(0, 0) | ||||||
|       await page |     await page | ||||||
|         .getByRole('button', { name: 'Start Sketch' }) |       .getByRole('button', { name: 'Start Sketch' }) | ||||||
|         .waitFor({ state: 'visible' }) |       .waitFor({ state: 'visible' }) | ||||||
|       await page.getByRole('button', { name: 'Start Sketch' }).click() |     await page.getByRole('button', { name: 'Start Sketch' }).click() | ||||||
|       await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() |     await expect(page.locator('.cm-ghostText').first()).not.toBeVisible() | ||||||
|  |  | ||||||
|       await expect(page.locator('.cm-content')).toHaveText(``) |     await expect(page.locator('.cm-content')).toHaveText(``) | ||||||
|     }) |  | ||||||
|   }) |   }) | ||||||
| }) | }) | ||||||
|  | |||||||
| @ -1,6 +1,14 @@ | |||||||
| import { test, expect } from './zoo-test' | import { test, expect } from '@playwright/test' | ||||||
|  |  | ||||||
| import { getUtils } from './test-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) | ||||||
|  | }) | ||||||
|  |  | ||||||
| function countNewlines(input: string): number { | function countNewlines(input: string): number { | ||||||
|   let count = 0 |   let count = 0 | ||||||
| @ -16,14 +24,13 @@ test.describe('Debug pane', () => { | |||||||
|   test('Artifact IDs in the artifact graph are stable across code edits', async ({ |   test('Artifact IDs in the artifact graph are stable across code edits', async ({ | ||||||
|     page, |     page, | ||||||
|     context, |     context, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     const code = `sketch001 = startSketchOn('XZ') |     const code = `sketch001 = startSketchOn('XZ') | ||||||
|     |> startProfileAt([0, 0], %) |   |> startProfileAt([0, 0], %) | ||||||
|   |> line([1, 1], %) | |> line([1, 1], %) | ||||||
|   ` | ` | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     const tree = page.getByTestId('debug-feature-tree') |     const tree = page.getByTestId('debug-feature-tree') | ||||||
|     const segment = tree.locator('li', { |     const segment = tree.locator('li', { | ||||||
| @ -32,7 +39,7 @@ test.describe('Debug pane', () => { | |||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await test.step('Test setup', async () => { |     await test.step('Test setup', async () => { | ||||||
|       await homePage.goToModelingScene() |       await u.waitForAuthSkipAppStart() | ||||||
|       await u.openKclCodePanel() |       await u.openKclCodePanel() | ||||||
|       await u.openDebugPanel() |       await u.openDebugPanel() | ||||||
|       // Set the code in the code editor. |       // Set the code in the code editor. | ||||||
|  | |||||||
| @ -1,31 +1,39 @@ | |||||||
| import { test, expect } from './zoo-test' | import { test, expect } from '@playwright/test' | ||||||
| import path from 'path' | import { join } from 'path' | ||||||
| import { | import { | ||||||
|   getUtils, |   getUtils, | ||||||
|  |   setupElectron, | ||||||
|  |   tearDown, | ||||||
|   executorInputPath, |   executorInputPath, | ||||||
|   getPlaywrightDownloadDir, |  | ||||||
| } from './test-utils' | } from './test-utils' | ||||||
| import fsp from 'fs/promises' | import fsp from 'fs/promises' | ||||||
|  |  | ||||||
|  | test.afterEach(async ({ page }, testInfo) => { | ||||||
|  |   await tearDown(page, testInfo) | ||||||
|  | }) | ||||||
|  |  | ||||||
| test( | test( | ||||||
|   'export works on the first try', |   'export works on the first try', | ||||||
|   { tag: '@electron' }, |   { tag: '@electron' }, | ||||||
|   async ({ page, context }, testInfo) => { |   async ({ browserName }, testInfo) => { | ||||||
|     await context.folderSetupFn(async (dir) => { |     const { electronApp, page } = await setupElectron({ | ||||||
|       const bracketDir = path.join(dir, 'bracket') |       testInfo, | ||||||
|       await Promise.all([fsp.mkdir(bracketDir, { recursive: true })]) |       folderSetupFn: async (dir) => { | ||||||
|       await Promise.all([ |         const bracketDir = join(dir, 'bracket') | ||||||
|         fsp.copyFile( |         await Promise.all([fsp.mkdir(bracketDir, { recursive: true })]) | ||||||
|           executorInputPath('router-template-slate.kcl'), |         await Promise.all([ | ||||||
|           path.join(bracketDir, 'other.kcl') |           fsp.copyFile( | ||||||
|         ), |             executorInputPath('router-template-slate.kcl'), | ||||||
|         fsp.copyFile( |             join(bracketDir, 'other.kcl') | ||||||
|           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), |           ), | ||||||
|           path.join(bracketDir, 'main.kcl') |           fsp.copyFile( | ||||||
|         ), |             executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||||
|       ]) |             join(bracketDir, 'main.kcl') | ||||||
|  |           ), | ||||||
|  |         ]) | ||||||
|  |       }, | ||||||
|     }) |     }) | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     page.on('console', console.log) |     page.on('console', console.log) | ||||||
|  |  | ||||||
| @ -85,16 +93,12 @@ test( | |||||||
|       await expect(successToastMessage).toBeVisible() |       await expect(successToastMessage).toBeVisible() | ||||||
|       await expect(exportingToastMessage).not.toBeVisible() |       await expect(exportingToastMessage).not.toBeVisible() | ||||||
|  |  | ||||||
|       const firstFileFullPath = path.resolve( |  | ||||||
|         getPlaywrightDownloadDir(page), |  | ||||||
|         exportFileName |  | ||||||
|       ) |  | ||||||
|       await test.step('Check the export size', async () => { |       await test.step('Check the export size', async () => { | ||||||
|         await expect |         await expect | ||||||
|           .poll( |           .poll( | ||||||
|             async () => { |             async () => { | ||||||
|               try { |               try { | ||||||
|                 const outputGltf = await fsp.readFile(firstFileFullPath) |                 const outputGltf = await fsp.readFile(exportFileName) | ||||||
|                 return outputGltf.byteLength |                 return outputGltf.byteLength | ||||||
|               } catch (e) { |               } catch (e) { | ||||||
|                 return 0 |                 return 0 | ||||||
| @ -103,6 +107,9 @@ test( | |||||||
|             { timeout: 15_000 } |             { timeout: 15_000 } | ||||||
|           ) |           ) | ||||||
|           .toBeGreaterThan(300_000) |           .toBeGreaterThan(300_000) | ||||||
|  |  | ||||||
|  |         // clean up exported file | ||||||
|  |         await fsp.rm(exportFileName) | ||||||
|       }) |       }) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
| @ -163,16 +170,12 @@ test( | |||||||
|           expect(exportingToastMessage).not.toBeVisible(), |           expect(exportingToastMessage).not.toBeVisible(), | ||||||
|         ])) |         ])) | ||||||
|  |  | ||||||
|       const secondFileFullPath = path.resolve( |  | ||||||
|         getPlaywrightDownloadDir(page), |  | ||||||
|         exportFileName |  | ||||||
|       ) |  | ||||||
|       await test.step('Check the export size', async () => { |       await test.step('Check the export size', async () => { | ||||||
|         await expect |         await expect | ||||||
|           .poll( |           .poll( | ||||||
|             async () => { |             async () => { | ||||||
|               try { |               try { | ||||||
|                 const outputGltf = await fsp.readFile(secondFileFullPath) |                 const outputGltf = await fsp.readFile(exportFileName) | ||||||
|                 return outputGltf.byteLength |                 return outputGltf.byteLength | ||||||
|               } catch (e) { |               } catch (e) { | ||||||
|                 return 0 |                 return 0 | ||||||
| @ -181,7 +184,13 @@ test( | |||||||
|             { timeout: 15_000 } |             { timeout: 15_000 } | ||||||
|           ) |           ) | ||||||
|           .toBeGreaterThan(100_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 './zoo-test' | import { test, expect } from '@playwright/test' | ||||||
| import fsp from 'fs/promises' | import fsp from 'fs/promises' | ||||||
| import { uuidv4 } from 'lib/utils' | import { uuidv4 } from 'lib/utils' | ||||||
| import { | import { | ||||||
| @ -6,27 +6,37 @@ import { | |||||||
|   darkModePlaneColorXZ, |   darkModePlaneColorXZ, | ||||||
|   executorInputPath, |   executorInputPath, | ||||||
|   getUtils, |   getUtils, | ||||||
|  |   setup, | ||||||
|  |   setupElectron, | ||||||
|  |   tearDown, | ||||||
| } from './test-utils' | } from './test-utils' | ||||||
|  |  | ||||||
| import { join } from 'path' | import { join } from 'path' | ||||||
|  |  | ||||||
| test.describe('Editor tests', () => { | test.beforeEach(async ({ context, page }, testInfo) => { | ||||||
|   test('can comment out code with ctrl+/', async ({ page, homePage }) => { |   await setup(context, page, testInfo) | ||||||
|     const u = await getUtils(page) | }) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |  | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() | test.afterEach(async ({ page }, testInfo) => { | ||||||
|  |   await tearDown(page, testInfo) | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | test.describe('Editor tests', () => { | ||||||
|  |   test('can comment out code with ctrl+/', async ({ page }) => { | ||||||
|  |     const u = await getUtils(page) | ||||||
|  |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|  |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     // check no error to begin with |     // check no error to begin with | ||||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() |     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await page.keyboard.type(`sketch001 = startSketchOn('XY') |     await page.keyboard.type(`sketch001 = startSketchOn('XY') | ||||||
|     |> startProfileAt([-10, -10], %) |   |> startProfileAt([-10, -10], %) | ||||||
|     |> line([20, 0], %) |   |> line([20, 0], %) | ||||||
|     |> line([0, 20], %) |   |> line([0, 20], %) | ||||||
|     |> line([-20, 0], %) |   |> line([-20, 0], %) | ||||||
|     |> close(%)`) |   |> close(%)`) | ||||||
|  |  | ||||||
|     await page.keyboard.down('ControlOrMeta') |     await page.keyboard.down('ControlOrMeta') | ||||||
|     await page.keyboard.press('/') |     await page.keyboard.press('/') | ||||||
| @ -34,11 +44,11 @@ test.describe('Editor tests', () => { | |||||||
|  |  | ||||||
|     await expect(page.locator('.cm-content')) |     await expect(page.locator('.cm-content')) | ||||||
|       .toHaveText(`sketch001 = startSketchOn('XY') |       .toHaveText(`sketch001 = startSketchOn('XY') | ||||||
|   |> startProfileAt([-10, -10], %) |     |> startProfileAt([-10, -10], %) | ||||||
|   |> line([20, 0], %) |     |> line([20, 0], %) | ||||||
|   |> line([0, 20], %) |     |> line([0, 20], %) | ||||||
|   |> line([-20, 0], %) |     |> line([-20, 0], %) | ||||||
|   // |> close(%)`) |     // |> close(%)`) | ||||||
|  |  | ||||||
|     // uncomment the code |     // uncomment the code | ||||||
|     await page.keyboard.down('ControlOrMeta') |     await page.keyboard.down('ControlOrMeta') | ||||||
| @ -47,165 +57,61 @@ test.describe('Editor tests', () => { | |||||||
|  |  | ||||||
|     await expect(page.locator('.cm-content')) |     await expect(page.locator('.cm-content')) | ||||||
|       .toHaveText(`sketch001 = startSketchOn('XY') |       .toHaveText(`sketch001 = startSketchOn('XY') | ||||||
|   |> startProfileAt([-10, -10], %) |     |> startProfileAt([-10, -10], %) | ||||||
|   |> line([20, 0], %) |     |> line([20, 0], %) | ||||||
|   |> line([0, 20], %) |     |> line([0, 20], %) | ||||||
|   |> line([-20, 0], %) |     |> line([-20, 0], %) | ||||||
|   |> close(%)`) |     |> close(%)`) | ||||||
|   }) |  | ||||||
|  |  | ||||||
|   test('ensure we use the cache, and do not re-execute', async ({ |  | ||||||
|     homePage, |  | ||||||
|     page, |  | ||||||
|   }) => { |  | ||||||
|     const u = await getUtils(page) |  | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |  | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |  | ||||||
|     await u.waitForPageLoad() |  | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |  | ||||||
|     await page.keyboard.type(`sketch001 = startSketchOn('XY') |  | ||||||
|   |> startProfileAt([-10, -10], %) |  | ||||||
|   |> line([20, 0], %) |  | ||||||
|   |> line([0, 20], %) |  | ||||||
|   |> line([-20, 0], %) |  | ||||||
|   |> close(%)`) |  | ||||||
|  |  | ||||||
|     // Ensure we execute the first time. |  | ||||||
|     await u.openDebugPanel() |  | ||||||
|     await expect( |  | ||||||
|       page.locator('[data-receive-command-type="scene_clear_all"]') |  | ||||||
|     ).toHaveCount(1) |  | ||||||
|     await expect( |  | ||||||
|       page.locator('[data-message-type="execution-done"]') |  | ||||||
|     ).toHaveCount(2) |  | ||||||
|  |  | ||||||
|     // Add whitespace to the end of the code. |  | ||||||
|     await u.codeLocator.click() |  | ||||||
|     await page.keyboard.press('ArrowUp') |  | ||||||
|     await page.keyboard.press('ArrowUp') |  | ||||||
|     await page.keyboard.press('ArrowUp') |  | ||||||
|     await page.keyboard.press('ArrowUp') |  | ||||||
|     await page.keyboard.press('Home') |  | ||||||
|     await page.keyboard.type('    ') |  | ||||||
|     await page.keyboard.press('Enter') |  | ||||||
|     await page.keyboard.type('    ') |  | ||||||
|  |  | ||||||
|     // Ensure we don't execute the second time. |  | ||||||
|     await u.openDebugPanel() |  | ||||||
|     // Make sure we didn't clear the scene. |  | ||||||
|     await expect( |  | ||||||
|       page.locator('[data-message-type="execution-done"]') |  | ||||||
|     ).toHaveCount(3) |  | ||||||
|     await expect( |  | ||||||
|       page.locator('[data-receive-command-type="scene_clear_all"]') |  | ||||||
|     ).toHaveCount(1) |  | ||||||
|   }) |  | ||||||
|  |  | ||||||
|   test('ensure we use the cache, and do not clear on append', async ({ |  | ||||||
|     homePage, |  | ||||||
|     page, |  | ||||||
|   }) => { |  | ||||||
|     const u = await getUtils(page) |  | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |  | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |  | ||||||
|     await u.waitForPageLoad() |  | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |  | ||||||
|     await page.keyboard.type(`sketch001 = startSketchOn('XY') |  | ||||||
|   |> startProfileAt([-10, -10], %) |  | ||||||
|   |> line([20, 0], %) |  | ||||||
|   |> line([0, 20], %) |  | ||||||
|   |> line([-20, 0], %) |  | ||||||
|   |> close(%)`) |  | ||||||
|  |  | ||||||
|     // Ensure we execute the first time. |  | ||||||
|     await u.openDebugPanel() |  | ||||||
|     await expect( |  | ||||||
|       page.locator('[data-receive-command-type="scene_clear_all"]') |  | ||||||
|     ).toHaveCount(1) |  | ||||||
|     await expect( |  | ||||||
|       page.locator('[data-message-type="execution-done"]') |  | ||||||
|     ).toHaveCount(2) |  | ||||||
|  |  | ||||||
|     // Add whitespace to the end of the code. |  | ||||||
|     await u.codeLocator.click() |  | ||||||
|     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('End') |  | ||||||
|     await page.keyboard.press('Enter') |  | ||||||
|     await page.keyboard.press('Enter') |  | ||||||
|     await page.keyboard.type('const x = 1') |  | ||||||
|     await page.keyboard.press('Enter') |  | ||||||
|  |  | ||||||
|     await u.openDebugPanel() |  | ||||||
|     await expect( |  | ||||||
|       page.locator('[data-message-type="execution-done"]') |  | ||||||
|     ).toHaveCount(3) |  | ||||||
|     await expect( |  | ||||||
|       page.locator('[data-receive-command-type="scene_clear_all"]') |  | ||||||
|     ).toHaveCount(1) |  | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('if you click the format button it formats your code', async ({ |   test('if you click the format button it formats your code', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     // check no error to begin with |     // check no error to begin with | ||||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() |     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await page.keyboard.type(`sketch001 = startSketchOn('XY') |     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], %) |   |> startProfileAt([-10, -10], %) | ||||||
|   |> line([20, 0], %) |   |> line([20, 0], %) | ||||||
|   |> line([0, 20], %) |   |> line([0, 20], %) | ||||||
|   |> line([-20, 0], %) |   |> line([-20, 0], %) | ||||||
|   |> close(%)`) |   |> 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 ({ |   test('if you click the format button it formats your code and executes so lints are still there', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     // check no error to begin with |     // check no error to begin with | ||||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() |     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||||
|  |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await page.keyboard.type(`sketch_001 = startSketchOn('XY') |     await page.keyboard.type(`sketch_001 = startSketchOn('XY') | ||||||
|     |> startProfileAt([-10, -10], %) |   |> startProfileAt([-10, -10], %) | ||||||
|     |> line([20, 0], %) |   |> line([20, 0], %) | ||||||
|     |> line([0, 20], %) |   |> line([0, 20], %) | ||||||
|     |> line([-20, 0], %) |   |> line([-20, 0], %) | ||||||
|     |> close(%)`) |   |> close(%)`) | ||||||
|  |  | ||||||
|     await u.openDebugPanel() |     await u.openDebugPanel() | ||||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') |     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||||
| @ -229,11 +135,11 @@ test.describe('Editor tests', () => { | |||||||
|  |  | ||||||
|     await expect(page.locator('.cm-content')) |     await expect(page.locator('.cm-content')) | ||||||
|       .toHaveText(`sketch_001 = startSketchOn('XY') |       .toHaveText(`sketch_001 = startSketchOn('XY') | ||||||
|   |> startProfileAt([-10, -10], %) |     |> startProfileAt([-10, -10], %) | ||||||
|   |> line([20, 0], %) |     |> line([20, 0], %) | ||||||
|   |> line([0, 20], %) |     |> line([0, 20], %) | ||||||
|   |> line([-20, 0], %) |     |> line([-20, 0], %) | ||||||
|   |> close(%)`) |     |> close(%)`) | ||||||
|  |  | ||||||
|     // error in guter |     // error in guter | ||||||
|     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible() |     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible() | ||||||
| @ -245,27 +151,29 @@ test.describe('Editor tests', () => { | |||||||
|     ).toBeVisible() |     ).toBeVisible() | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('fold gutters work', async ({ page, homePage }) => { |   test('fold gutters work', async ({ page }) => { | ||||||
|  |     const u = await getUtils(page) | ||||||
|  |  | ||||||
|     const fullCode = `sketch001 = startSketchOn('XY') |     const fullCode = `sketch001 = startSketchOn('XY') | ||||||
|    |> startProfileAt([-10, -10], %) |      |> startProfileAt([-10, -10], %) | ||||||
|    |> line([20, 0], %) |      |> line([20, 0], %) | ||||||
|    |> line([0, 20], %) |      |> line([0, 20], %) | ||||||
|    |> line([-20, 0], %) |      |> line([-20, 0], %) | ||||||
|    |> close(%)` |      |> close(%)` | ||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `sketch001 = startSketchOn('XY') |         `sketch001 = startSketchOn('XY') | ||||||
|    |> startProfileAt([-10, -10], %) |      |> startProfileAt([-10, -10], %) | ||||||
|    |> line([20, 0], %) |      |> line([20, 0], %) | ||||||
|    |> line([0, 20], %) |      |> line([0, 20], %) | ||||||
|    |> line([-20, 0], %) |      |> line([-20, 0], %) | ||||||
|    |> close(%)` |      |> close(%)` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     // TODO: Jess needs to fix this but you have to mod the code to get them to show |     // TODO: Jess needs to fix this but you have to mod the code to get them to show | ||||||
|     // up, its an annoying codemirror thing. |     // up, its an annoying codemirror thing. | ||||||
| @ -316,25 +224,22 @@ test.describe('Editor tests', () => { | |||||||
|     await expect(foldGutterFoldLine).not.toBeVisible() |     await expect(foldGutterFoldLine).not.toBeVisible() | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('hover over functions shows function description', async ({ |   test('hover over functions shows function description', async ({ page }) => { | ||||||
|     page, |  | ||||||
|     homePage, |  | ||||||
|   }) => { |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `sketch001 = startSketchOn('XY') |         `sketch001 = startSketchOn('XY') | ||||||
|     |> startProfileAt([-10, -10], %) |   |> startProfileAt([-10, -10], %) | ||||||
|     |> line([20, 0], %) |   |> line([20, 0], %) | ||||||
|     |> line([0, 20], %) |   |> line([0, 20], %) | ||||||
|     |> line([-20, 0], %) |   |> line([-20, 0], %) | ||||||
|     |> close(%)` |   |> close(%)` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     // check no error to begin with |     // check no error to begin with | ||||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() |     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||||
| @ -363,24 +268,23 @@ test.describe('Editor tests', () => { | |||||||
|  |  | ||||||
|   test('if you use the format keyboard binding it formats your code', async ({ |   test('if you use the format keyboard binding it formats your code', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `sketch001 = startSketchOn('XY') |         `sketch001 = startSketchOn('XY') | ||||||
|     |> startProfileAt([-10, -10], %) |   |> startProfileAt([-10, -10], %) | ||||||
|     |> line([20, 0], %) |   |> line([20, 0], %) | ||||||
|     |> line([0, 20], %) |   |> line([0, 20], %) | ||||||
|     |> line([-20, 0], %) |   |> line([-20, 0], %) | ||||||
|     |> close(%)` |   |> close(%)` | ||||||
|       ) |       ) | ||||||
|       localStorage.setItem('disableAxis', 'true') |       localStorage.setItem('disableAxis', 'true') | ||||||
|     }) |     }) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     // check no error to begin with |     // check no error to begin with | ||||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() |     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||||
| @ -397,33 +301,32 @@ test.describe('Editor tests', () => { | |||||||
|  |  | ||||||
|     await expect(page.locator('.cm-content')) |     await expect(page.locator('.cm-content')) | ||||||
|       .toHaveText(`sketch001 = startSketchOn('XY') |       .toHaveText(`sketch001 = startSketchOn('XY') | ||||||
|   |> startProfileAt([-10, -10], %) |     |> startProfileAt([-10, -10], %) | ||||||
|   |> line([20, 0], %) |     |> line([20, 0], %) | ||||||
|   |> line([0, 20], %) |     |> line([0, 20], %) | ||||||
|   |> line([-20, 0], %) |     |> line([-20, 0], %) | ||||||
|   |> close(%)`) |     |> close(%)`) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('if you use the format keyboard binding it formats your code and executes so lints are shown', async ({ |   test('if you use the format keyboard binding it formats your code and executes so lints are shown', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `sketch_001 = startSketchOn('XY') |         `sketch_001 = startSketchOn('XY') | ||||||
|     |> startProfileAt([-10, -10], %) |   |> startProfileAt([-10, -10], %) | ||||||
|     |> line([20, 0], %) |   |> line([20, 0], %) | ||||||
|     |> line([0, 20], %) |   |> line([0, 20], %) | ||||||
|     |> line([-20, 0], %) |   |> line([-20, 0], %) | ||||||
|     |> close(%)` |   |> close(%)` | ||||||
|       ) |       ) | ||||||
|       localStorage.setItem('disableAxis', 'true') |       localStorage.setItem('disableAxis', 'true') | ||||||
|     }) |     }) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     await u.openDebugPanel() |     await u.openDebugPanel() | ||||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') |     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||||
| @ -450,11 +353,11 @@ test.describe('Editor tests', () => { | |||||||
|  |  | ||||||
|     await expect(page.locator('.cm-content')) |     await expect(page.locator('.cm-content')) | ||||||
|       .toHaveText(`sketch_001 = startSketchOn('XY') |       .toHaveText(`sketch_001 = startSketchOn('XY') | ||||||
|   |> startProfileAt([-10, -10], %) |     |> startProfileAt([-10, -10], %) | ||||||
|   |> line([20, 0], %) |     |> line([20, 0], %) | ||||||
|   |> line([0, 20], %) |     |> line([0, 20], %) | ||||||
|   |> line([-20, 0], %) |     |> line([-20, 0], %) | ||||||
|   |> close(%)`) |     |> close(%)`) | ||||||
|  |  | ||||||
|     // error in guter |     // error in guter | ||||||
|     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible() |     await expect(page.locator('.cm-lint-marker-info').first()).toBeVisible() | ||||||
| @ -466,14 +369,11 @@ test.describe('Editor tests', () => { | |||||||
|     ).toBeVisible() |     ).toBeVisible() | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('if you write kcl with lint errors you get lints', async ({ |   test('if you write kcl with lint errors you get lints', async ({ page }) => { | ||||||
|     page, |  | ||||||
|     homePage, |  | ||||||
|   }) => { |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     // check no error to begin with |     // check no error to begin with | ||||||
|     await expect(page.locator('.cm-lint-marker-info')).not.toBeVisible() |     await expect(page.locator('.cm-lint-marker-info')).not.toBeVisible() | ||||||
| @ -509,26 +409,23 @@ test.describe('Editor tests', () => { | |||||||
|     await expect(page.locator('.cm-lint-marker-info')).not.toBeVisible() |     await expect(page.locator('.cm-lint-marker-info')).not.toBeVisible() | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('if you fixup kcl errors you clear lints', async ({ |   test('if you fixup kcl errors you clear lints', async ({ page }) => { | ||||||
|     page, |  | ||||||
|     homePage, |  | ||||||
|   }) => { |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `sketch001 = startSketchOn('XZ') |         `sketch001 = startSketchOn('XZ') | ||||||
|     |> startProfileAt([3.29, 7.86], %) |   |> startProfileAt([3.29, 7.86], %) | ||||||
|     |> line([2.48, 2.44], %) |   |> line([2.48, 2.44], %) | ||||||
|     |> line([2.66, 1.17], %) |   |> line([2.66, 1.17], %) | ||||||
|     |> close(%) |   |> close(%) | ||||||
|     ` |   ` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     // check no error to begin with |     // check no error to begin with | ||||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() |     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||||
| @ -550,27 +447,22 @@ test.describe('Editor tests', () => { | |||||||
|     ).not.toBeVisible() |     ).not.toBeVisible() | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('if you write invalid kcl you get inlined errors', async ({ |   test('if you write invalid kcl you get inlined errors', async ({ page }) => { | ||||||
|     page, |  | ||||||
|     homePage, |  | ||||||
|   }) => { |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|     // check no error to begin with |     // check no error to begin with | ||||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() |     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||||
|  |  | ||||||
|     /* add the following code to the editor (~ error is not a valid line) |     /* add the following code to the editor ($ error is not a valid line) | ||||||
|       * the old check here used $ but this is for tags so it changed meaning. |       $ error | ||||||
|       * hopefully ~ doesn't change meaning |       topAng = 30 | ||||||
|     ~ error |       bottomAng = 25 | ||||||
|     const topAng = 30 |      */ | ||||||
|     const bottomAng = 25 |  | ||||||
|    */ |  | ||||||
|     await u.codeLocator.click() |     await u.codeLocator.click() | ||||||
|     await page.keyboard.type('~ error') |     await page.keyboard.type('$ error') | ||||||
|  |  | ||||||
|     // press arrows to clear autocomplete |     // press arrows to clear autocomplete | ||||||
|     await page.keyboard.press('ArrowLeft') |     await page.keyboard.press('ArrowLeft') | ||||||
| @ -582,17 +474,17 @@ test.describe('Editor tests', () => { | |||||||
|     await page.keyboard.type('bottomAng = 25') |     await page.keyboard.type('bottomAng = 25') | ||||||
|     await page.keyboard.press('Enter') |     await page.keyboard.press('Enter') | ||||||
|  |  | ||||||
|     // error in guter |     // error in gutter | ||||||
|     await expect(page.locator('.cm-lint-marker-error')).toBeVisible() |     await expect(page.locator('.cm-lint-marker-error')).toBeVisible() | ||||||
|  |  | ||||||
|     // error text on hover |     // error text on hover | ||||||
|     await page.hover('.cm-lint-marker-error') |     await page.hover('.cm-lint-marker-error') | ||||||
|     await expect( |     await expect( | ||||||
|       page.getByText("found unknown token '~'").first() |       page.getByText('Tag names must not be empty').first() | ||||||
|     ).toBeVisible() |     ).toBeVisible() | ||||||
|  |  | ||||||
|     // select the line that's causing the error and delete it |     // select the line that's causing the error and delete it | ||||||
|     await page.getByText('~ error').click() |     await page.getByText('$ error').click() | ||||||
|     await page.keyboard.press('End') |     await page.keyboard.press('End') | ||||||
|     await page.keyboard.down('Shift') |     await page.keyboard.down('Shift') | ||||||
|     await page.keyboard.press('Home') |     await page.keyboard.press('Home') | ||||||
| @ -628,108 +520,106 @@ test.describe('Editor tests', () => { | |||||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() |     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test.fixme( |   // TODO currently multiple source ranges are not supported | ||||||
|     'error with 2 source ranges gets 2 diagnostics', |   test.skip('error with 2 source ranges gets 2 diagnostics', async ({ | ||||||
|     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 }) |  | ||||||
|  |  | ||||||
|       await homePage.goToModelingScene() |  | ||||||
|       await u.waitForPageLoad() |  | ||||||
|       await page.waitForTimeout(1000) |  | ||||||
|  |  | ||||||
|       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() |  | ||||||
|  |  | ||||||
|       // 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() |  | ||||||
|  |  | ||||||
|       // 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, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     await context.addInitScript(async () => { |     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.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|  |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|  |     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() | ||||||
|  |  | ||||||
|  |     // 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() | ||||||
|  |  | ||||||
|  |     // 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 ({ | ||||||
|  |     page, | ||||||
|  |   }) => { | ||||||
|  |     const u = await getUtils(page) | ||||||
|  |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `box = startSketchOn('XY') |         `box = startSketchOn('XY') | ||||||
|     |> startProfileAt([0, 0], %) |   |> startProfileAt([0, 0], %) | ||||||
|     |> line([0, 10], %) |   |> line([0, 10], %) | ||||||
|     |> line([10, 0], %) |   |> line([10, 0], %) | ||||||
|     |> line([0, -10], %, $revolveAxis) |   |> line([0, -10], %, $revolveAxis) | ||||||
|     |> close(%) |   |> close(%) | ||||||
|     |> extrude(10, %) |   |> extrude(10, %) | ||||||
|    |  | ||||||
|     sketch001 = startSketchOn(box, revolveAxis) |   sketch001 = startSketchOn(box, revolveAxis) | ||||||
|     |> startProfileAt([5, 10], %) |   |> startProfileAt([5, 10], %) | ||||||
|     |> line([0, -10], %) |   |> line([0, -10], %) | ||||||
|     |> line([2, 0], %) |   |> line([2, 0], %) | ||||||
|     |> line([0, -10], %) |   |> line([0, -10], %) | ||||||
|     |> close(%) |   |> close(%) | ||||||
|     |> revolve({ |   |> revolve({ | ||||||
|     axis: revolveAxis, |   axis = revolveAxis, | ||||||
|     angle: 90 |   angle = 90 | ||||||
|     }, %) |   }, %) | ||||||
|     ` |       ` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await page.goto('/') | ||||||
|  |     await u.waitForPageLoad() | ||||||
|  |  | ||||||
|     await expect(page.locator('.cm-lint-marker-error')).toBeVisible() |     await expect(page.locator('.cm-lint-marker-error')).toBeVisible() | ||||||
|  |  | ||||||
| @ -740,15 +630,12 @@ test.describe('Editor tests', () => { | |||||||
|     await expect(page.getByText(searchText)).toBeVisible() |     await expect(page.getByText(searchText)).toBeVisible() | ||||||
|   }) |   }) | ||||||
|   test.describe('Autocomplete works', () => { |   test.describe('Autocomplete works', () => { | ||||||
|     test('with enter/click to accept the completion', async ({ |     test('with enter/click to accept the completion', async ({ page }) => { | ||||||
|       page, |  | ||||||
|       homePage, |  | ||||||
|     }) => { |  | ||||||
|       const u = await getUtils(page) |       const u = await getUtils(page) | ||||||
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio |       // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |       await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|       await homePage.goToModelingScene() |       await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|       // tests clicking on an option, selection the first option |       // tests clicking on an option, selection the first option | ||||||
|       // and arrowing down to an option |       // and arrowing down to an option | ||||||
| @ -810,19 +697,19 @@ test.describe('Editor tests', () => { | |||||||
|  |  | ||||||
|       await expect(page.locator('.cm-content')) |       await expect(page.locator('.cm-content')) | ||||||
|         .toHaveText(`sketch001 = startSketchOn('XZ') |         .toHaveText(`sketch001 = startSketchOn('XZ') | ||||||
|         |> startProfileAt([3.14, 12], %) |     |> startProfileAt([3.14, 12], %) | ||||||
|         |> xLine(5, %) // lin`) |     |> xLine(5, %) // lin`) | ||||||
|  |  | ||||||
|       // expect there to be no KCL errors |       // expect there to be no KCL errors | ||||||
|       await expect(page.locator('.cm-lint-marker-error')).toHaveCount(0) |       await expect(page.locator('.cm-lint-marker-error')).toHaveCount(0) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     test('with tab to accept the completion', async ({ page, homePage }) => { |     test('with tab to accept the completion', async ({ page }) => { | ||||||
|       const u = await getUtils(page) |       const u = await getUtils(page) | ||||||
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio |       // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |       await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|       await homePage.goToModelingScene() |       await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|       // this test might be brittle as we add and remove functions |       // this test might be brittle as we add and remove functions | ||||||
|       // but should also be easy to update. |       // but should also be easy to update. | ||||||
| @ -884,30 +771,26 @@ test.describe('Editor tests', () => { | |||||||
|  |  | ||||||
|       await expect(page.locator('.cm-content')) |       await expect(page.locator('.cm-content')) | ||||||
|         .toHaveText(`sketch001 = startSketchOn('XZ') |         .toHaveText(`sketch001 = startSketchOn('XZ') | ||||||
|         |> startProfileAt([3.14, 12], %) |     |> startProfileAt([3.14, 12], %) | ||||||
|         |> xLine(5, %) // lin`) |     |> xLine(5, %) // lin`) | ||||||
|     }) |     }) | ||||||
|   }) |   }) | ||||||
|   test('Can undo a click and point extrude with ctrl+z', async ({ |   test('Can undo a click and point extrude with ctrl+z', async ({ page }) => { | ||||||
|     page, |  | ||||||
|     context, |  | ||||||
|     homePage, |  | ||||||
|   }) => { |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await context.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `sketch001 = startSketchOn('XZ') |         `sketch001 = startSketchOn('XZ') | ||||||
|   |> startProfileAt([4.61, -14.01], %) |     |> startProfileAt([4.61, -14.01], %) | ||||||
|   |> line([12.73, -0.09], %) |     |> line([12.73, -0.09], %) | ||||||
|   |> tangentialArcTo([24.95, -5.38], %) |     |> tangentialArcTo([24.95, -5.38], %) | ||||||
|   |> close(%)` |     |> close(%)` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|     await expect( |     await expect( | ||||||
|       page.getByRole('button', { name: 'Start Sketch' }) |       page.getByRole('button', { name: 'Start Sketch' }) | ||||||
|     ).not.toBeDisabled() |     ).not.toBeDisabled() | ||||||
| @ -960,32 +843,29 @@ test.describe('Editor tests', () => { | |||||||
|     await page.waitForTimeout(100) |     await page.waitForTimeout(100) | ||||||
|     await expect(page.locator('.cm-content')) |     await expect(page.locator('.cm-content')) | ||||||
|       .toHaveText(`sketch001 = startSketchOn('XZ') |       .toHaveText(`sketch001 = startSketchOn('XZ') | ||||||
|   |> startProfileAt([4.61, -14.01], %) |     |> startProfileAt([4.61, -14.01], %) | ||||||
|   |> line([12.73, -0.09], %) |     |> line([12.73, -0.09], %) | ||||||
|   |> tangentialArcTo([24.95, -5.38], %) |     |> tangentialArcTo([24.95, -5.38], %) | ||||||
|   |> close(%)`) |     |> close(%)`) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('Can undo a sketch modification with ctrl+z', async ({ |   test('Can undo a sketch modification with ctrl+z', async ({ page }) => { | ||||||
|     page, |  | ||||||
|     homePage, |  | ||||||
|   }) => { |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `sketch001 = startSketchOn('XZ') |         `sketch001 = startSketchOn('XZ') | ||||||
|   |> startProfileAt([4.61, -10.01], %) |     |> startProfileAt([4.61, -10.01], %) | ||||||
|   |> line([12.73, -0.09], %) |     |> line([12.73, -0.09], %) | ||||||
|   |> tangentialArcTo([24.95, -0.38], %) |     |> tangentialArcTo([24.95, -0.38], %) | ||||||
|   |> close(%) |     |> close(%) | ||||||
|   |> extrude(5, %)` |     |> extrude(5, %)` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|     await expect( |     await expect( | ||||||
|       page.getByRole('button', { name: 'Start Sketch' }) |       page.getByRole('button', { name: 'Start Sketch' }) | ||||||
|     ).not.toBeDisabled() |     ).not.toBeDisabled() | ||||||
| @ -1012,7 +892,7 @@ test.describe('Editor tests', () => { | |||||||
|     }) |     }) | ||||||
|     await page.waitForTimeout(100) |     await page.waitForTimeout(100) | ||||||
|  |  | ||||||
|     const startPX = [1200 / 2, 500 / 2] |     const startPX = [665, 397] | ||||||
|  |  | ||||||
|     const dragPX = 40 |     const dragPX = 40 | ||||||
|  |  | ||||||
| @ -1026,9 +906,9 @@ test.describe('Editor tests', () => { | |||||||
|  |  | ||||||
|     await expect(page.getByTestId('segment-overlay')).toHaveCount(2) |     await expect(page.getByTestId('segment-overlay')).toHaveCount(2) | ||||||
|  |  | ||||||
|     // drag startProfileAt handle |     // drag startProfieAt handle | ||||||
|     await page.dragAndDrop('#stream', '#stream', { |     await page.dragAndDrop('#stream', '#stream', { | ||||||
|       sourcePosition: { x: startPX[0] + 68, y: startPX[1] + 147 }, |       sourcePosition: { x: startPX[0], y: startPX[1] }, | ||||||
|       targetPosition: { x: startPX[0] + dragPX, y: startPX[1] + dragPX }, |       targetPosition: { x: startPX[0] + dragPX, y: startPX[1] + dragPX }, | ||||||
|     }) |     }) | ||||||
|     await page.waitForTimeout(100) |     await page.waitForTimeout(100) | ||||||
| @ -1066,12 +946,12 @@ test.describe('Editor tests', () => { | |||||||
|     // expect the code to have changed |     // expect the code to have changed | ||||||
|     await expect(page.locator('.cm-content')) |     await expect(page.locator('.cm-content')) | ||||||
|       .toHaveText(`sketch001 = startSketchOn('XZ') |       .toHaveText(`sketch001 = startSketchOn('XZ') | ||||||
|     |> startProfileAt([2.71, -2.71], %) |   |> startProfileAt([7.12, -12.68], %) | ||||||
|     |> line([15.4, -2.78], %) |   |> line([15.39, -2.78], %) | ||||||
|     |> tangentialArcTo([27.6, -3.05], %) |   |> tangentialArcTo([27.6, -3.05], %) | ||||||
|     |> close(%) |   |> close(%) | ||||||
|     |> extrude(5, %) |   |> extrude(5, %) | ||||||
|   `) | `) | ||||||
|  |  | ||||||
|     // Hit undo |     // Hit undo | ||||||
|     await page.keyboard.down('Control') |     await page.keyboard.down('Control') | ||||||
| @ -1080,11 +960,11 @@ test.describe('Editor tests', () => { | |||||||
|  |  | ||||||
|     await expect(page.locator('.cm-content')) |     await expect(page.locator('.cm-content')) | ||||||
|       .toHaveText(`sketch001 = startSketchOn('XZ') |       .toHaveText(`sketch001 = startSketchOn('XZ') | ||||||
|     |> startProfileAt([2.71, -2.71], %) |   |> startProfileAt([7.12, -12.68], %) | ||||||
|     |> line([15.4, -2.78], %) |   |> line([15.39, -2.78], %) | ||||||
|     |> tangentialArcTo([24.95, -0.38], %) |   |> tangentialArcTo([24.95, -0.38], %) | ||||||
|     |> close(%) |   |> close(%) | ||||||
|     |> extrude(5, %)`) |   |> extrude(5, %)`) | ||||||
|  |  | ||||||
|     // Hit undo again. |     // Hit undo again. | ||||||
|     await page.keyboard.down('Control') |     await page.keyboard.down('Control') | ||||||
| @ -1093,12 +973,12 @@ test.describe('Editor tests', () => { | |||||||
|  |  | ||||||
|     await expect(page.locator('.cm-content')) |     await expect(page.locator('.cm-content')) | ||||||
|       .toHaveText(`sketch001 = startSketchOn('XZ') |       .toHaveText(`sketch001 = startSketchOn('XZ') | ||||||
|     |> startProfileAt([2.71, -2.71], %) |   |> startProfileAt([7.12, -12.68], %) | ||||||
|     |> line([12.73, -0.09], %) |   |> line([12.73, -0.09], %) | ||||||
|     |> tangentialArcTo([24.95, -0.38], %) |   |> tangentialArcTo([24.95, -0.38], %) | ||||||
|     |> close(%) |   |> close(%) | ||||||
|     |> extrude(5, %) |   |> extrude(5, %) | ||||||
|   `) | `) | ||||||
|  |  | ||||||
|     // Hit undo again. |     // Hit undo again. | ||||||
|     await page.keyboard.down('Control') |     await page.keyboard.down('Control') | ||||||
| @ -1108,29 +988,31 @@ test.describe('Editor tests', () => { | |||||||
|     await page.waitForTimeout(100) |     await page.waitForTimeout(100) | ||||||
|     await expect(page.locator('.cm-content')) |     await expect(page.locator('.cm-content')) | ||||||
|       .toHaveText(`sketch001 = startSketchOn('XZ') |       .toHaveText(`sketch001 = startSketchOn('XZ') | ||||||
|   |> startProfileAt([4.61, -10.01], %) |     |> startProfileAt([4.61, -10.01], %) | ||||||
|   |> line([12.73, -0.09], %) |     |> line([12.73, -0.09], %) | ||||||
|   |> tangentialArcTo([24.95, -0.38], %) |     |> tangentialArcTo([24.95, -0.38], %) | ||||||
|   |> close(%) |     |> close(%) | ||||||
|   |> extrude(5, %)`) |     |> extrude(5, %)`) | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test.fixme( |   test.fixme( | ||||||
|     `Can use the import stdlib function on a local OBJ file`, |     `Can use the import stdlib function on a local OBJ file`, | ||||||
|     { tag: '@electron' }, |     { tag: '@electron' }, | ||||||
|     async ({ page, context }, testInfo) => { |     async ({ browserName }, testInfo) => { | ||||||
|       await context.folderSetupFn(async (dir) => { |       const { electronApp, page } = await setupElectron({ | ||||||
|         const bracketDir = join(dir, 'cube') |         testInfo, | ||||||
|         await fsp.mkdir(bracketDir, { recursive: true }) |         folderSetupFn: async (dir) => { | ||||||
|         await fsp.copyFile( |           const bracketDir = join(dir, 'cube') | ||||||
|           executorInputPath('cube.obj'), |           await fsp.mkdir(bracketDir, { recursive: true }) | ||||||
|           join(bracketDir, 'cube.obj') |           await fsp.copyFile( | ||||||
|         ) |             executorInputPath('cube.obj'), | ||||||
|         await fsp.writeFile(join(bracketDir, 'main.kcl'), '') |             join(bracketDir, 'cube.obj') | ||||||
|  |           ) | ||||||
|  |           await fsp.writeFile(join(bracketDir, 'main.kcl'), '') | ||||||
|  |         }, | ||||||
|       }) |       }) | ||||||
|  |  | ||||||
|       const viewportSize = { width: 1200, height: 500 } |       const viewportSize = { width: 1200, height: 500 } | ||||||
|       await page.setBodyDimensions(viewportSize) |       await page.setViewportSize(viewportSize) | ||||||
|  |  | ||||||
|       // Locators and constants |       // Locators and constants | ||||||
|       const u = await getUtils(page) |       const u = await getUtils(page) | ||||||
| @ -1188,6 +1070,8 @@ test.describe('Editor tests', () => { | |||||||
|           }) |           }) | ||||||
|           .toBeGreaterThan(15) |           .toBeGreaterThan(15) | ||||||
|       }) |       }) | ||||||
|  |  | ||||||
|  |       await electronApp.close() | ||||||
|     } |     } | ||||||
|   ) |   ) | ||||||
| }) | }) | ||||||
|  | |||||||
| @ -1,127 +0,0 @@ | |||||||
| import { test, expect } from './zoo-test' |  | ||||||
| import * as fsp from 'fs/promises' |  | ||||||
| import { join } from 'path' |  | ||||||
|  |  | ||||||
| const FEATURE_TREE_EXAMPLE_CODE = `export fn timesFive(x) { |  | ||||||
|   return 5 * x |  | ||||||
| } |  | ||||||
| export fn triangle() { |  | ||||||
|   return startSketchOn('XZ') |  | ||||||
|     |> startProfileAt([0, 0], %) |  | ||||||
|     |> xLine(10, %) |  | ||||||
|     |> line([-10, -5], %) |  | ||||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) |  | ||||||
|     |> close(%) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| length001 = timesFive(1) * 5 |  | ||||||
| sketch001 = startSketchOn('XZ') |  | ||||||
|   |> startProfileAt([20, 10], %) |  | ||||||
|   |> line([10, 10], %) |  | ||||||
|   |> angledLine([-45, length001], %) |  | ||||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) |  | ||||||
|   |> close(%) |  | ||||||
| revolve001 = revolve({ axis = "X" }, sketch001) |  | ||||||
| triangle() |  | ||||||
|   |> extrude(30, %) |  | ||||||
| plane001 = offsetPlane('XY', 10) |  | ||||||
| sketch002 = startSketchOn(plane001) |  | ||||||
|   |> startProfileAt([-20, 0], %) |  | ||||||
|   |> line([5, -15], %) |  | ||||||
|   |> xLine(-10, %) |  | ||||||
|   |> lineTo([-40, 0], %) |  | ||||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) |  | ||||||
|   |> close(%) |  | ||||||
| extrude001 = extrude(10, sketch002) |  | ||||||
| ` |  | ||||||
|  |  | ||||||
| test.describe('Feature Tree pane', () => { |  | ||||||
|   test( |  | ||||||
|     'User can go to definition and go to function definition', |  | ||||||
|     { tag: '@electron' }, |  | ||||||
|     async ({ context, homePage, scene, editor, toolbar }) => { |  | ||||||
|       await context.folderSetupFn(async (dir) => { |  | ||||||
|         const bracketDir = join(dir, 'test-sample') |  | ||||||
|         await fsp.mkdir(bracketDir, { recursive: true }) |  | ||||||
|         await fsp.writeFile( |  | ||||||
|           join(bracketDir, 'main.kcl'), |  | ||||||
|           FEATURE_TREE_EXAMPLE_CODE, |  | ||||||
|           'utf-8' |  | ||||||
|         ) |  | ||||||
|       }) |  | ||||||
|  |  | ||||||
|       await test.step('setup test', async () => { |  | ||||||
|         await homePage.expectState({ |  | ||||||
|           projectCards: [ |  | ||||||
|             { |  | ||||||
|               title: 'test-sample', |  | ||||||
|               fileCount: 1, |  | ||||||
|             }, |  | ||||||
|           ], |  | ||||||
|           sortBy: 'last-modified-desc', |  | ||||||
|         }) |  | ||||||
|         await homePage.openProject('test-sample') |  | ||||||
|         await scene.waitForExecutionDone() |  | ||||||
|         await editor.closePane() |  | ||||||
|         await toolbar.openFeatureTreePane() |  | ||||||
|       }) |  | ||||||
|  |  | ||||||
|       async function testViewSource({ |  | ||||||
|         operationName, |  | ||||||
|         operationIndex, |  | ||||||
|         expectedActiveLine, |  | ||||||
|       }: { |  | ||||||
|         operationName: string |  | ||||||
|         operationIndex: number |  | ||||||
|         expectedActiveLine: string |  | ||||||
|       }) { |  | ||||||
|         await test.step(`Go to definition of the ${operationName}`, async () => { |  | ||||||
|           await toolbar.viewSourceOnOperation(operationName, operationIndex) |  | ||||||
|           await editor.expectState({ |  | ||||||
|             highlightedCode: '', |  | ||||||
|             diagnostics: [], |  | ||||||
|             activeLines: [expectedActiveLine], |  | ||||||
|           }) |  | ||||||
|           await expect( |  | ||||||
|             editor.activeLine.first(), |  | ||||||
|             `${operationName} code should be scrolled into view` |  | ||||||
|           ).toBeVisible() |  | ||||||
|         }) |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       await testViewSource({ |  | ||||||
|         operationName: 'Offset Plane', |  | ||||||
|         operationIndex: 0, |  | ||||||
|         expectedActiveLine: "plane001 = offsetPlane('XY', 10)", |  | ||||||
|       }) |  | ||||||
|       await testViewSource({ |  | ||||||
|         operationName: 'Extrude', |  | ||||||
|         operationIndex: 1, |  | ||||||
|         expectedActiveLine: 'extrude001 = extrude(10, sketch002)', |  | ||||||
|       }) |  | ||||||
|       await testViewSource({ |  | ||||||
|         operationName: 'Revolve', |  | ||||||
|         operationIndex: 0, |  | ||||||
|         expectedActiveLine: 'revolve001 = revolve({ axis = "X" }, sketch001)', |  | ||||||
|       }) |  | ||||||
|       await testViewSource({ |  | ||||||
|         operationName: 'Triangle', |  | ||||||
|         operationIndex: 0, |  | ||||||
|         expectedActiveLine: 'triangle()', |  | ||||||
|       }) |  | ||||||
|  |  | ||||||
|       await test.step('Go to definition on the triangle function', async () => { |  | ||||||
|         await toolbar.goToDefinitionOnOperation('Triangle', 0) |  | ||||||
|         await editor.expectState({ |  | ||||||
|           highlightedCode: '', |  | ||||||
|           diagnostics: [], |  | ||||||
|           activeLines: ['export fn triangle() {'], |  | ||||||
|         }) |  | ||||||
|         await expect( |  | ||||||
|           editor.activeLine.first(), |  | ||||||
|           'Triangle function definition should be scrolled into view' |  | ||||||
|         ).toBeVisible() |  | ||||||
|       }) |  | ||||||
|     } |  | ||||||
|   ) |  | ||||||
| }) |  | ||||||
| @ -1,4 +1,4 @@ | |||||||
| import type { Page, Locator } from '@playwright/test' | import type { Page } from '@playwright/test' | ||||||
| import { expect } from '@playwright/test' | import { expect } from '@playwright/test' | ||||||
|  |  | ||||||
| type CmdBarSerialised = | type CmdBarSerialised = | ||||||
| @ -26,11 +26,9 @@ type CmdBarSerialised = | |||||||
|  |  | ||||||
| export class CmdBarFixture { | export class CmdBarFixture { | ||||||
|   public page: Page |   public page: Page | ||||||
|   cmdBarOpenBtn!: Locator |  | ||||||
|  |  | ||||||
|   constructor(page: Page) { |   constructor(page: Page) { | ||||||
|     this.page = page |     this.page = page | ||||||
|     this.cmdBarOpenBtn = page.getByTestId('command-bar-open-button') |  | ||||||
|   } |   } | ||||||
|   reConstruct = (page: Page) => { |   reConstruct = (page: Page) => { | ||||||
|     this.page = page |     this.page = page | ||||||
| @ -118,21 +116,4 @@ export class CmdBarFixture { | |||||||
|       await this.page.keyboard.press('Enter') |       await this.page.keyboard.press('Enter') | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   openCmdBar = async (selectCmd?: 'promptToEdit') => { |  | ||||||
|     // TODO why does this button not work in electron tests? |  | ||||||
|     // await this.cmdBarOpenBtn.click() |  | ||||||
|     await this.page.keyboard.down('ControlOrMeta') |  | ||||||
|     await this.page.keyboard.press('KeyK') |  | ||||||
|     await this.page.keyboard.up('ControlOrMeta') |  | ||||||
|     await expect(this.page.getByPlaceholder('Search commands')).toBeVisible() |  | ||||||
|     if (selectCmd === 'promptToEdit') { |  | ||||||
|       const promptEditCommand = this.page.getByText( |  | ||||||
|         'Use Zoo AI to edit your kcl' |  | ||||||
|       ) |  | ||||||
|       await expect(promptEditCommand.first()).toBeVisible() |  | ||||||
|       await promptEditCommand.first().scrollIntoViewIfNeeded() |  | ||||||
|       await promptEditCommand.first().click() |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -20,7 +20,7 @@ export class EditorFixture { | |||||||
|   private diagnosticsTooltip!: Locator |   private diagnosticsTooltip!: Locator | ||||||
|   private diagnosticsGutterIcon!: Locator |   private diagnosticsGutterIcon!: Locator | ||||||
|   private codeContent!: Locator |   private codeContent!: Locator | ||||||
|   public activeLine!: Locator |   private activeLine!: Locator | ||||||
|  |  | ||||||
|   constructor(page: Page) { |   constructor(page: Page) { | ||||||
|     this.page = page |     this.page = page | ||||||
| @ -29,7 +29,7 @@ export class EditorFixture { | |||||||
|   reConstruct = (page: Page) => { |   reConstruct = (page: Page) => { | ||||||
|     this.page = page |     this.page = page | ||||||
|  |  | ||||||
|     this.codeContent = page.locator('.cm-content[data-language="kcl"]') |     this.codeContent = page.locator('.cm-content') | ||||||
|     this.diagnosticsTooltip = page.locator('.cm-tooltip-lint') |     this.diagnosticsTooltip = page.locator('.cm-tooltip-lint') | ||||||
|     this.diagnosticsGutterIcon = page.locator('.cm-lint-marker-error') |     this.diagnosticsGutterIcon = page.locator('.cm-lint-marker-error') | ||||||
|     this.activeLine = this.page.locator('.cm-activeLine') |     this.activeLine = this.page.locator('.cm-activeLine') | ||||||
| @ -54,13 +54,13 @@ export class EditorFixture { | |||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       if (!shouldNormalise) { |       if (!shouldNormalise) { | ||||||
|         const expectStart = expect.poll(() => this.codeContent.textContent()) |         const expectStart = expect(this.codeContent) | ||||||
|         if (not) { |         if (not) { | ||||||
|           const result = await expectStart.not.toContain(code) |           const result = await expectStart.not.toContainText(code, { timeout }) | ||||||
|           await resetPane() |           await resetPane() | ||||||
|           return result |           return result | ||||||
|         } |         } | ||||||
|         const result = await expectStart.toContain(code) |         const result = await expectStart.toContainText(code, { timeout }) | ||||||
|         await resetPane() |         await resetPane() | ||||||
|         return result |         return result | ||||||
|       } |       } | ||||||
| @ -147,28 +147,4 @@ export class EditorFixture { | |||||||
|   openPane() { |   openPane() { | ||||||
|     return openPane(this.page, this.paneButtonTestId) |     return openPane(this.page, this.paneButtonTestId) | ||||||
|   } |   } | ||||||
|   scrollToText(text: string, placeCursor?: boolean) { |  | ||||||
|     return this.page.evaluate( |  | ||||||
|       (args: { text: string; placeCursor?: boolean }) => { |  | ||||||
|         // error TS2339: Property 'docView' does not exist on type 'EditorView'. |  | ||||||
|         // Except it does so :shrug: |  | ||||||
|         // @ts-ignore |  | ||||||
|         let index = window.editorManager._editorView?.docView.view.state.doc |  | ||||||
|           .toString() |  | ||||||
|           .indexOf(args.text) |  | ||||||
|         window.editorManager._editorView?.focus() |  | ||||||
|         window.editorManager._editorView?.dispatch({ |  | ||||||
|           selection: window.EditorSelection.create([ |  | ||||||
|             window.EditorSelection.cursor(index), |  | ||||||
|           ]), |  | ||||||
|           effects: [ |  | ||||||
|             window.EditorView.scrollIntoView( |  | ||||||
|               window.EditorSelection.range(index, index + 1) |  | ||||||
|             ), |  | ||||||
|           ], |  | ||||||
|         }) |  | ||||||
|       }, |  | ||||||
|       { text, placeCursor } |  | ||||||
|     ) |  | ||||||
|   } |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,11 +1,11 @@ | |||||||
| import type { | import type { | ||||||
|   BrowserContext, |   BrowserContext, | ||||||
|   ElectronApplication, |   ElectronApplication, | ||||||
|   TestInfo, |  | ||||||
|   Page, |   Page, | ||||||
|  |   TestInfo, | ||||||
| } from '@playwright/test' | } from '@playwright/test' | ||||||
|  | import { test as base } from '@playwright/test' | ||||||
| import { getUtils, setup, setupElectron } from '../test-utils' | import { getUtils, setup, setupElectron, tearDown } from '../test-utils' | ||||||
| import fsp from 'fs/promises' | import fsp from 'fs/promises' | ||||||
| import { join } from 'path' | import { join } from 'path' | ||||||
| import { CmdBarFixture } from './cmdBarFixture' | import { CmdBarFixture } from './cmdBarFixture' | ||||||
| @ -20,13 +20,11 @@ export class AuthenticatedApp { | |||||||
|   public readonly page: Page |   public readonly page: Page | ||||||
|   public readonly context: BrowserContext |   public readonly context: BrowserContext | ||||||
|   public readonly testInfo: TestInfo |   public readonly testInfo: TestInfo | ||||||
|   public readonly viewPortSize = { width: 1200, height: 500 } |   public readonly viewPortSize = { width: 1000, height: 500 } | ||||||
|   public electronApp: undefined | ElectronApplication |  | ||||||
|   public dir: string = '' |  | ||||||
|  |  | ||||||
|   constructor(context: BrowserContext, page: Page, testInfo: TestInfo) { |   constructor(context: BrowserContext, page: Page, testInfo: TestInfo) { | ||||||
|     this.context = context |  | ||||||
|     this.page = page |     this.page = page | ||||||
|  |     this.context = context | ||||||
|     this.testInfo = testInfo |     this.testInfo = testInfo | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @ -51,7 +49,9 @@ export class AuthenticatedApp { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| export interface Fixtures { | interface Fixtures { | ||||||
|  |   app: AuthenticatedApp | ||||||
|  |   tronApp: AuthenticatedTronApp | ||||||
|   cmdBar: CmdBarFixture |   cmdBar: CmdBarFixture | ||||||
|   editor: EditorFixture |   editor: EditorFixture | ||||||
|   toolbar: ToolbarFixture |   toolbar: ToolbarFixture | ||||||
| @ -61,11 +61,9 @@ export interface Fixtures { | |||||||
| export class AuthenticatedTronApp { | export class AuthenticatedTronApp { | ||||||
|   public readonly _page: Page |   public readonly _page: Page | ||||||
|   public page: Page |   public page: Page | ||||||
|   public context: BrowserContext |   public readonly context: BrowserContext | ||||||
|   public readonly testInfo: TestInfo |   public readonly testInfo: TestInfo | ||||||
|   public electronApp: ElectronApplication | undefined |   public electronApp?: ElectronApplication | ||||||
|   public readonly viewPortSize = { width: 1200, height: 500 } |  | ||||||
|   public dir: string = '' |  | ||||||
|  |  | ||||||
|   constructor(context: BrowserContext, page: Page, testInfo: TestInfo) { |   constructor(context: BrowserContext, page: Page, testInfo: TestInfo) { | ||||||
|     this._page = page |     this._page = page | ||||||
| @ -81,22 +79,15 @@ export class AuthenticatedTronApp { | |||||||
|       appSettings?: Partial<SaveSettingsPayload> |       appSettings?: Partial<SaveSettingsPayload> | ||||||
|     } = { fixtures: {} } |     } = { fixtures: {} } | ||||||
|   ) { |   ) { | ||||||
|     const { electronApp, page, context, dir } = await setupElectron({ |     const { electronApp, page } = await setupElectron({ | ||||||
|       testInfo: this.testInfo, |       testInfo: this.testInfo, | ||||||
|       folderSetupFn: arg.folderSetupFn, |       folderSetupFn: arg.folderSetupFn, | ||||||
|       cleanProjectDir: arg.cleanProjectDir, |       cleanProjectDir: arg.cleanProjectDir, | ||||||
|       appSettings: arg.appSettings, |       appSettings: arg.appSettings, | ||||||
|     }) |     }) | ||||||
|     this.page = page |     this.page = page | ||||||
|     this.context = context |  | ||||||
|     this.electronApp = electronApp |     this.electronApp = electronApp | ||||||
|     this.dir = dir |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     // 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)) { |     for (const key of unsafeTypedKeys(arg.fixtures)) { | ||||||
|       const fixture = arg.fixtures[key] |       const fixture = arg.fixtures[key] | ||||||
| @ -119,20 +110,32 @@ export class AuthenticatedTronApp { | |||||||
|     }) |     }) | ||||||
| } | } | ||||||
|  |  | ||||||
| export const fixtures = { | export const test = base.extend<Fixtures>({ | ||||||
|   cmdBar: async ({ page }: { page: Page }, use: any) => { |   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) => { | ||||||
|     await use(new CmdBarFixture(page)) |     await use(new CmdBarFixture(page)) | ||||||
|   }, |   }, | ||||||
|   editor: async ({ page }: { page: Page }, use: any) => { |   editor: async ({ page }, use) => { | ||||||
|     await use(new EditorFixture(page)) |     await use(new EditorFixture(page)) | ||||||
|   }, |   }, | ||||||
|   toolbar: async ({ page }: { page: Page }, use: any) => { |   toolbar: async ({ page }, use) => { | ||||||
|     await use(new ToolbarFixture(page)) |     await use(new ToolbarFixture(page)) | ||||||
|   }, |   }, | ||||||
|   scene: async ({ page }: { page: Page }, use: any) => { |   scene: async ({ page }, use) => { | ||||||
|     await use(new SceneFixture(page)) |     await use(new SceneFixture(page)) | ||||||
|   }, |   }, | ||||||
|   homePage: async ({ page }: { page: Page }, use: any) => { |   homePage: async ({ page }, use) => { | ||||||
|     await use(new HomePageFixture(page)) |     await use(new HomePageFixture(page)) | ||||||
|   }, |   }, | ||||||
| } | }) | ||||||
|  |  | ||||||
|  | test.afterEach(async ({ page }, testInfo) => { | ||||||
|  |   await tearDown(page, testInfo) | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | export { expect } from '@playwright/test' | ||||||
|  | |||||||
| @ -14,14 +14,10 @@ interface HomePageState { | |||||||
| export class HomePageFixture { | export class HomePageFixture { | ||||||
|   public page: Page |   public page: Page | ||||||
|  |  | ||||||
|   projectSection!: Locator |  | ||||||
|   projectCard!: Locator |   projectCard!: Locator | ||||||
|   projectCardTitle!: Locator |   projectCardTitle!: Locator | ||||||
|   projectCardFile!: Locator |   projectCardFile!: Locator | ||||||
|   projectCardFolder!: Locator |   projectCardFolder!: Locator | ||||||
|   projectButtonNew!: Locator |  | ||||||
|   projectButtonContinue!: Locator |  | ||||||
|   projectTextName!: Locator |  | ||||||
|   sortByDateBtn!: Locator |   sortByDateBtn!: Locator | ||||||
|   sortByNameBtn!: Locator |   sortByNameBtn!: Locator | ||||||
|  |  | ||||||
| @ -32,19 +28,11 @@ export class HomePageFixture { | |||||||
|   reConstruct = (page: Page) => { |   reConstruct = (page: Page) => { | ||||||
|     this.page = page |     this.page = page | ||||||
|  |  | ||||||
|     this.projectSection = this.page.getByTestId('home-section') |  | ||||||
|  |  | ||||||
|     this.projectCard = this.page.getByTestId('project-link') |     this.projectCard = this.page.getByTestId('project-link') | ||||||
|     this.projectCardTitle = this.page.getByTestId('project-title') |     this.projectCardTitle = this.page.getByTestId('project-title') | ||||||
|     this.projectCardFile = this.page.getByTestId('project-file-count') |     this.projectCardFile = this.page.getByTestId('project-file-count') | ||||||
|     this.projectCardFolder = this.page.getByTestId('project-folder-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.sortByDateBtn = this.page.getByTestId('home-sort-by-modified') | ||||||
|     this.sortByNameBtn = this.page.getByTestId('home-sort-by-name') |     this.sortByNameBtn = this.page.getByTestId('home-sort-by-name') | ||||||
|   } |   } | ||||||
| @ -103,25 +91,10 @@ export class HomePageFixture { | |||||||
|       .toEqual(expectedState) |       .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) => { |   openProject = async (projectTitle: string) => { | ||||||
|     const projectCard = this.projectCard.locator( |     const projectCard = this.projectCard.locator( | ||||||
|       this.page.getByText(projectTitle) |       this.page.getByText(projectTitle) | ||||||
|     ) |     ) | ||||||
|     await projectCard.click() |     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,9 +53,8 @@ export class SceneFixture { | |||||||
|  |  | ||||||
|   expectState = async (expected: SceneSerialised) => { |   expectState = async (expected: SceneSerialised) => { | ||||||
|     return expect |     return expect | ||||||
|       .poll(async () => await this._serialiseScene(), { |       .poll(() => this._serialiseScene(), { | ||||||
|         intervals: [1_000, 2_000, 10_000], |         message: `Expected scene state to match`, | ||||||
|         timeout: 60000, |  | ||||||
|       }) |       }) | ||||||
|       .toEqual(expected) |       .toEqual(expected) | ||||||
|   } |   } | ||||||
| @ -188,10 +187,7 @@ export class SceneFixture { | |||||||
|         type: 'default_camera_get_settings', |         type: 'default_camera_get_settings', | ||||||
|       }, |       }, | ||||||
|     }) |     }) | ||||||
|     await this.page |     await this.waitForExecutionDone() | ||||||
|       .locator(`[data-receive-command-type="default_camera_get_settings"]`) |  | ||||||
|       .first() |  | ||||||
|       .waitFor() |  | ||||||
|     const position = await Promise.all([ |     const position = await Promise.all([ | ||||||
|       this.page.getByTestId('cam-x-position').inputValue().then(Number), |       this.page.getByTestId('cam-x-position').inputValue().then(Number), | ||||||
|       this.page.getByTestId('cam-y-position').inputValue().then(Number), |       this.page.getByTestId('cam-y-position').inputValue().then(Number), | ||||||
| @ -218,7 +214,23 @@ export class SceneFixture { | |||||||
|     coords: { x: number; y: number }, |     coords: { x: number; y: number }, | ||||||
|     diff: number |     diff: number | ||||||
|   ) => { |   ) => { | ||||||
|     await expectPixelColor(this.page, colour, coords, diff) |     let finalValue = colour | ||||||
|  |     await expect | ||||||
|  |       .poll(async () => { | ||||||
|  |         const pixel = (await getPixelRGBs(this.page)(coords, 1))[0] | ||||||
|  |         if (!pixel) return null | ||||||
|  |         finalValue = pixel | ||||||
|  |         return pixel.every( | ||||||
|  |           (channel, index) => Math.abs(channel - colour[index]) < diff | ||||||
|  |         ) | ||||||
|  |       }) | ||||||
|  |       .toBeTruthy() | ||||||
|  |       .catch((cause) => { | ||||||
|  |         throw new Error( | ||||||
|  |           `ExpectPixelColor: expecting ${colour} got ${finalValue}`, | ||||||
|  |           { cause } | ||||||
|  |         ) | ||||||
|  |       }) | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   get gizmo() { |   get gizmo() { | ||||||
| @ -226,7 +238,6 @@ export class SceneFixture { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   async clickGizmoMenuItem(name: string) { |   async clickGizmoMenuItem(name: string) { | ||||||
|     await this.gizmo.hover() |  | ||||||
|     await this.gizmo.click({ button: 'right' }) |     await this.gizmo.click({ button: 'right' }) | ||||||
|     const buttonToTest = this.page.getByRole('button', { |     const buttonToTest = this.page.getByRole('button', { | ||||||
|       name: name, |       name: name, | ||||||
| @ -235,28 +246,3 @@ export class SceneFixture { | |||||||
|     await buttonToTest.click() |     await buttonToTest.click() | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| export async function expectPixelColor( |  | ||||||
|   page: Page, |  | ||||||
|   colour: [number, number, number], |  | ||||||
|   coords: { x: number; y: number }, |  | ||||||
|   diff: number |  | ||||||
| ) { |  | ||||||
|   let finalValue = colour |  | ||||||
|   await expect |  | ||||||
|     .poll(async () => { |  | ||||||
|       const pixel = (await getPixelRGBs(page)(coords, 1))[0] |  | ||||||
|       if (!pixel) return null |  | ||||||
|       finalValue = pixel |  | ||||||
|       return pixel.every( |  | ||||||
|         (channel, index) => Math.abs(channel - colour[index]) < diff |  | ||||||
|       ) |  | ||||||
|     }) |  | ||||||
|     .toBeTruthy() |  | ||||||
|     .catch((cause) => { |  | ||||||
|       throw new Error( |  | ||||||
|         `ExpectPixelColor: expecting ${colour} got ${finalValue}`, |  | ||||||
|         { cause } |  | ||||||
|       ) |  | ||||||
|     }) |  | ||||||
| } |  | ||||||
|  | |||||||
| @ -1,13 +1,6 @@ | |||||||
| import type { Page, Locator } from '@playwright/test' | import type { Page, Locator } from '@playwright/test' | ||||||
| import { expect } from '../zoo-test' | import { expect } from './fixtureSetup' | ||||||
| import { | import { doAndWaitForImageDiff } from '../test-utils' | ||||||
|   checkIfPaneIsOpen, |  | ||||||
|   closePane, |  | ||||||
|   doAndWaitForImageDiff, |  | ||||||
|   openPane, |  | ||||||
| } from '../test-utils' |  | ||||||
| import { SidebarType } from 'components/ModelingSidebar/ModelingPanes' |  | ||||||
| import { SIDEBAR_BUTTON_SUFFIX } from 'lib/constants' |  | ||||||
|  |  | ||||||
| export class ToolbarFixture { | export class ToolbarFixture { | ||||||
|   public page: Page |   public page: Page | ||||||
| @ -27,10 +20,6 @@ export class ToolbarFixture { | |||||||
|   filePane!: Locator |   filePane!: Locator | ||||||
|   exeIndicator!: Locator |   exeIndicator!: Locator | ||||||
|   treeInputField!: Locator |   treeInputField!: Locator | ||||||
|   /** The sidebar button for the Feature Tree pane */ |  | ||||||
|   featureTreeId = 'feature-tree' as const |  | ||||||
|   /** The pane element for the Feature Tree */ |  | ||||||
|   featureTreePane!: Locator |  | ||||||
|  |  | ||||||
|   constructor(page: Page) { |   constructor(page: Page) { | ||||||
|     this.page = page |     this.page = page | ||||||
| @ -52,7 +41,6 @@ export class ToolbarFixture { | |||||||
|     this.treeInputField = page.getByTestId('tree-input-field') |     this.treeInputField = page.getByTestId('tree-input-field') | ||||||
|  |  | ||||||
|     this.filePane = page.locator('#files-pane') |     this.filePane = page.locator('#files-pane') | ||||||
|     this.featureTreePane = page.locator('#feature-tree-pane') |  | ||||||
|     this.fileCreateToast = page.getByText('Successfully created') |     this.fileCreateToast = page.getByText('Successfully created') | ||||||
|     this.exeIndicator = page.getByTestId('model-state-indicator-execution-done') |     this.exeIndicator = page.getByTestId('model-state-indicator-execution-done') | ||||||
|   } |   } | ||||||
| @ -103,76 +91,4 @@ export class ToolbarFixture { | |||||||
|       await expect(this.exeIndicator).toBeVisible({ timeout: 15_000 }) |       await expect(this.exeIndicator).toBeVisible({ timeout: 15_000 }) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   async closePane(paneId: SidebarType) { |  | ||||||
|     return closePane(this.page, paneId + SIDEBAR_BUTTON_SUFFIX) |  | ||||||
|   } |  | ||||||
|   async openPane(paneId: SidebarType) { |  | ||||||
|     return openPane(this.page, paneId + SIDEBAR_BUTTON_SUFFIX) |  | ||||||
|   } |  | ||||||
|   async checkIfPaneIsOpen(paneId: SidebarType) { |  | ||||||
|     return checkIfPaneIsOpen(this.page, paneId + SIDEBAR_BUTTON_SUFFIX) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   async openFeatureTreePane() { |  | ||||||
|     return this.openPane(this.featureTreeId) |  | ||||||
|   } |  | ||||||
|   async closeFeatureTreePane() { |  | ||||||
|     await this.closePane(this.featureTreeId) |  | ||||||
|   } |  | ||||||
|   async checkIfFeatureTreePaneIsOpen() { |  | ||||||
|     return this.checkIfPaneIsOpen(this.featureTreeId) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Get a specific operation button from the Feature Tree pane |  | ||||||
|    */ |  | ||||||
|   async getFeatureTreeOperation(operationName: string, operationIndex: number) { |  | ||||||
|     await this.openFeatureTreePane() |  | ||||||
|     await expect(this.featureTreePane).toBeVisible() |  | ||||||
|     return this.featureTreePane |  | ||||||
|       .getByRole('button', { |  | ||||||
|         name: operationName, |  | ||||||
|       }) |  | ||||||
|       .nth(operationIndex) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * View source on a specific operation in the Feature Tree pane. |  | ||||||
|    * @param operationName The name of the operation type |  | ||||||
|    * @param operationIndex The index out of operations of this type |  | ||||||
|    */ |  | ||||||
|   async viewSourceOnOperation(operationName: string, operationIndex: number) { |  | ||||||
|     const operationButton = await this.getFeatureTreeOperation( |  | ||||||
|       operationName, |  | ||||||
|       operationIndex |  | ||||||
|     ) |  | ||||||
|     const viewSourceMenuButton = this.page.getByRole('button', { |  | ||||||
|       name: 'View KCL source code', |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     await operationButton.click({ button: 'right' }) |  | ||||||
|     await expect(viewSourceMenuButton).toBeVisible() |  | ||||||
|     await viewSourceMenuButton.click() |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Go to definition on a specific operation in the Feature Tree pane |  | ||||||
|    */ |  | ||||||
|   async goToDefinitionOnOperation( |  | ||||||
|     operationName: string, |  | ||||||
|     operationIndex: number |  | ||||||
|   ) { |  | ||||||
|     const operationButton = await this.getFeatureTreeOperation( |  | ||||||
|       operationName, |  | ||||||
|       operationIndex |  | ||||||
|     ) |  | ||||||
|     const goToDefinitionMenuButton = this.page.getByRole('button', { |  | ||||||
|       name: 'View function definition', |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     await operationButton.click({ button: 'right' }) |  | ||||||
|     await expect(goToDefinitionMenuButton).toBeVisible() |  | ||||||
|     await goToDefinitionMenuButton.click() |  | ||||||
|   } |  | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,22 +1,29 @@ | |||||||
| import { test, expect } from './zoo-test' | import { test, expect } from '@playwright/test' | ||||||
| import { executorInputPath } from './test-utils' | import { setupElectron, tearDown, executorInputPath } from './test-utils' | ||||||
| import { join } from 'path' | import { join } from 'path' | ||||||
| import fsp from 'fs/promises' | import fsp from 'fs/promises' | ||||||
|  |  | ||||||
|  | test.afterEach(async ({ page }, testInfo) => { | ||||||
|  |   await tearDown(page, testInfo) | ||||||
|  | }) | ||||||
|  |  | ||||||
| test( | test( | ||||||
|   'When machine-api server not found butt is disabled and shows the reason', |   'When machine-api server not found butt is disabled and shows the reason', | ||||||
|   { tag: '@electron' }, |   { tag: '@electron' }, | ||||||
|   async ({ context, page }, testInfo) => { |   async ({ browserName }, testInfo) => { | ||||||
|     await context.folderSetupFn(async (dir) => { |     const { electronApp, page } = await setupElectron({ | ||||||
|       const bracketDir = join(dir, 'bracket') |       testInfo, | ||||||
|       await fsp.mkdir(bracketDir, { recursive: true }) |       folderSetupFn: async (dir) => { | ||||||
|       await fsp.copyFile( |         const bracketDir = join(dir, 'bracket') | ||||||
|         executorInputPath('focusrite_scarlett_mounting_braket.kcl'), |         await fsp.mkdir(bracketDir, { recursive: true }) | ||||||
|         join(bracketDir, 'main.kcl') |         await fsp.copyFile( | ||||||
|       ) |           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||||
|  |           join(bracketDir, 'main.kcl') | ||||||
|  |         ) | ||||||
|  |       }, | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     await expect(page.getByText('bracket')).toBeVisible() |     await expect(page.getByText('bracket')).toBeVisible() | ||||||
|  |  | ||||||
| @ -40,23 +47,28 @@ test( | |||||||
|     // that the machine-api server is not found |     // that the machine-api server is not found | ||||||
|     await makeButton.hover() |     await makeButton.hover() | ||||||
|     await expect(page.getByText(notFoundText).first()).toBeVisible() |     await expect(page.getByText(notFoundText).first()).toBeVisible() | ||||||
|  |  | ||||||
|  |     await electronApp.close() | ||||||
|   } |   } | ||||||
| ) | ) | ||||||
|  |  | ||||||
| test( | test( | ||||||
|   'When machine-api server not found home screen & project status shows the reason', |   'When machine-api server not found home screen & project status shows the reason', | ||||||
|   { tag: '@electron' }, |   { tag: '@electron' }, | ||||||
|   async ({ context, page }, testInfo) => { |   async ({ browserName }, testInfo) => { | ||||||
|     await context.folderSetupFn(async (dir) => { |     const { electronApp, page } = await setupElectron({ | ||||||
|       const bracketDir = join(dir, 'bracket') |       testInfo, | ||||||
|       await fsp.mkdir(bracketDir, { recursive: true }) |       folderSetupFn: async (dir) => { | ||||||
|       await fsp.copyFile( |         const bracketDir = join(dir, 'bracket') | ||||||
|         executorInputPath('focusrite_scarlett_mounting_braket.kcl'), |         await fsp.mkdir(bracketDir, { recursive: true }) | ||||||
|         join(bracketDir, 'main.kcl') |         await fsp.copyFile( | ||||||
|       ) |           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||||
|  |           join(bracketDir, 'main.kcl') | ||||||
|  |         ) | ||||||
|  |       }, | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1200, height: 500 }) |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|     const notFoundText = 'Machine API server was not discovered' |     const notFoundText = 'Machine API server was not discovered' | ||||||
|  |  | ||||||
| @ -79,5 +91,7 @@ test( | |||||||
|  |  | ||||||
|     await networkMachineToggle.hover() |     await networkMachineToggle.hover() | ||||||
|     await expect(page.getByText(notFoundText).nth(1)).toBeVisible() |     await expect(page.getByText(notFoundText).nth(1)).toBeVisible() | ||||||
|  |  | ||||||
|  |     await electronApp.close() | ||||||
|   } |   } | ||||||
| ) | ) | ||||||
|  | |||||||
| @ -1,12 +0,0 @@ | |||||||
| // 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,78 +1,79 @@ | |||||||
| import { test, expect } from './zoo-test' | import { test, expect } from '@playwright/test' | ||||||
| import { join } from 'path' | import { join } from 'path' | ||||||
| import fsp from 'fs/promises' | import fsp from 'fs/promises' | ||||||
| import { getUtils, executorInputPath, createProject } from './test-utils' | import { | ||||||
|  |   getUtils, | ||||||
|  |   setup, | ||||||
|  |   setupElectron, | ||||||
|  |   tearDown, | ||||||
|  |   executorInputPath, | ||||||
|  |   createProject, | ||||||
|  | } from './test-utils' | ||||||
| import { bracket } from 'lib/exampleKcl' | import { bracket } from 'lib/exampleKcl' | ||||||
| import { onboardingPaths } from 'routes/Onboarding/paths' | import { onboardingPaths } from 'routes/Onboarding/paths' | ||||||
| import { | import { | ||||||
|   TEST_SETTINGS_KEY, |   TEST_SETTINGS_KEY, | ||||||
|   TEST_SETTINGS_ONBOARDING_START, |   TEST_SETTINGS_ONBOARDING_START, | ||||||
|   TEST_SETTINGS_ONBOARDING_EXPORT, |   TEST_SETTINGS_ONBOARDING_EXPORT, | ||||||
|  |   TEST_SETTINGS_ONBOARDING_PARAMETRIC_MODELING, | ||||||
|   TEST_SETTINGS_ONBOARDING_USER_MENU, |   TEST_SETTINGS_ONBOARDING_USER_MENU, | ||||||
| } from './storageStates' | } from './storageStates' | ||||||
| import * as TOML from '@iarna/toml' | import * as TOML from '@iarna/toml' | ||||||
| import { expectPixelColor } from './fixtures/sceneFixture' |  | ||||||
|  |  | ||||||
| // Because onboarding relies on an app setting we need to set it as incompletel | test.beforeEach(async ({ context, page }, testInfo) => { | ||||||
| // for all these tests. |   if (testInfo.tags.includes('@electron')) { | ||||||
|  |     return | ||||||
|  |   } | ||||||
|  |   await setup(context, page) | ||||||
|  | }) | ||||||
|  |  | ||||||
|  | test.afterEach(async ({ page }, testInfo) => { | ||||||
|  |   await tearDown(page, testInfo) | ||||||
|  | }) | ||||||
|  |  | ||||||
| test.describe('Onboarding tests', () => { | test.describe('Onboarding tests', () => { | ||||||
|   test( |   test('Onboarding code is shown in the editor', async ({ page }) => { | ||||||
|     'Onboarding code is shown in the editor', |     const u = await getUtils(page) | ||||||
|     { |  | ||||||
|       appSettings: { |     // Override beforeEach test setup | ||||||
|         app: { |     await page.addInitScript( | ||||||
|           onboardingStatus: 'incomplete', |       async ({ settingsKey }) => { | ||||||
|         }, |         // Give no initial code, so that the onboarding start is shown immediately | ||||||
|  |         localStorage.removeItem('persistCode') | ||||||
|  |         localStorage.removeItem(settingsKey) | ||||||
|       }, |       }, | ||||||
|       cleanProjectDir: true, |       { settingsKey: TEST_SETTINGS_KEY } | ||||||
|     }, |     ) | ||||||
|     async ({ context, page, homePage }) => { |  | ||||||
|       const u = await getUtils(page) |  | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |  | ||||||
|       await homePage.goToModelingScene() |  | ||||||
|  |  | ||||||
|       // Test that the onboarding pane loaded |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|       await expect( |  | ||||||
|         page.getByText('Welcome to Modeling App! This') |  | ||||||
|       ).toBeVisible() |  | ||||||
|  |  | ||||||
|       // Test that the onboarding pane loaded |     await u.waitForAuthSkipAppStart() | ||||||
|       await expect( |  | ||||||
|         page.getByText('Welcome to Modeling App! This') |  | ||||||
|       ).toBeVisible() |  | ||||||
|  |  | ||||||
|       // *and* that the code is shown in the editor |     // Test that the onboarding pane loaded | ||||||
|       await expect(page.locator('.cm-content')).toContainText( |     await expect(page.getByText('Welcome to Modeling App! This')).toBeVisible() | ||||||
|         '// Shelf Bracket' |  | ||||||
|       ) |  | ||||||
|  |  | ||||||
|       // Make sure the model loaded |     // *and* that the code is shown in the editor | ||||||
|       const XYPlanePoint = { x: 774, y: 116 } as const |     await expect(page.locator('.cm-content')).toContainText('// Shelf Bracket') | ||||||
|       const modelColor: [number, number, number] = [45, 45, 45] |   }) | ||||||
|       await page.mouse.move(XYPlanePoint.x, XYPlanePoint.y) |  | ||||||
|       expect(await u.getGreatestPixDiff(XYPlanePoint, modelColor)).toBeLessThan( |  | ||||||
|         8 |  | ||||||
|       ) |  | ||||||
|     } |  | ||||||
|   ) |  | ||||||
|  |  | ||||||
|   test( |   test( | ||||||
|     'Desktop: fresh onboarding executes and loads', |     'Desktop: fresh onboarding executes and loads', | ||||||
|     { |     { tag: '@electron' }, | ||||||
|       tag: '@electron', |     async ({ browserName: _ }, testInfo) => { | ||||||
|       appSettings: { |       const { electronApp, page } = await setupElectron({ | ||||||
|         app: { |         testInfo, | ||||||
|           onboardingStatus: 'incomplete', |         appSettings: { | ||||||
|  |           app: { | ||||||
|  |             onboardingStatus: 'incomplete', | ||||||
|  |           }, | ||||||
|         }, |         }, | ||||||
|       }, |         cleanProjectDir: true, | ||||||
|       cleanProjectDir: true, |       }) | ||||||
|     }, |  | ||||||
|     async ({ page, homePage }, testInfo) => { |  | ||||||
|       const u = await getUtils(page) |       const u = await getUtils(page) | ||||||
|  |  | ||||||
|       const viewportSize = { width: 1200, height: 500 } |       const viewportSize = { width: 1200, height: 500 } | ||||||
|       await page.setBodyDimensions(viewportSize) |       await page.setViewportSize(viewportSize) | ||||||
|  |  | ||||||
|       await test.step(`Create a project and open to the onboarding`, async () => { |       await test.step(`Create a project and open to the onboarding`, async () => { | ||||||
|         await createProject({ name: 'project-link', page }) |         await createProject({ name: 'project-link', page }) | ||||||
| @ -91,370 +92,321 @@ test.describe('Onboarding tests', () => { | |||||||
|         await expect(page.locator('.cm-content')).toContainText( |         await expect(page.locator('.cm-content')).toContainText( | ||||||
|           '// Shelf Bracket' |           '// Shelf Bracket' | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         // TODO: jess make less shit |  | ||||||
|         // Make sure the model loaded |  | ||||||
|         //const XYPlanePoint = { x: 986, y: 522 } as const |  | ||||||
|         //const modelColor: [number, number, number] = [76, 76, 76] |  | ||||||
|         //await page.mouse.move(XYPlanePoint.x, XYPlanePoint.y) |  | ||||||
|  |  | ||||||
|         //await expectPixelColor(page, modelColor, XYPlanePoint, 8) |  | ||||||
|       }) |       }) | ||||||
|  |  | ||||||
|  |       await electronApp.close() | ||||||
|     } |     } | ||||||
|   ) |   ) | ||||||
|  |  | ||||||
|   test( |   test('Code resets after confirmation', async ({ page }) => { | ||||||
|     'Code resets after confirmation', |     const initialCode = `sketch001 = startSketchOn('XZ')` | ||||||
|     { |  | ||||||
|       appSettings: { |  | ||||||
|         app: { |  | ||||||
|           onboardingStatus: 'incomplete', |  | ||||||
|         }, |  | ||||||
|       }, |  | ||||||
|       cleanProjectDir: true, |  | ||||||
|     }, |  | ||||||
|     async ({ context, page, homePage }) => { |  | ||||||
|       const initialCode = `sketch001 = startSketchOn('XZ')` |  | ||||||
|  |  | ||||||
|       // Load the page up with some code so we see the confirmation warning |     // Load the page up with some code so we see the confirmation warning | ||||||
|       // when we go to replay onboarding |     // when we go to replay onboarding | ||||||
|       await context.addInitScript((code) => { |     await page.addInitScript((code) => { | ||||||
|         localStorage.setItem('persistCode', code) |       localStorage.setItem('persistCode', code) | ||||||
|       }, initialCode) |     }, initialCode) | ||||||
|  |  | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |     const u = await getUtils(page) | ||||||
|       await homePage.goToModelingScene() |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|       // Replay the onboarding |     // Replay the onboarding | ||||||
|       await page.getByRole('link', { name: 'Settings' }).last().click() |     await page.getByRole('link', { name: 'Settings' }).last().click() | ||||||
|       const replayButton = page.getByRole('button', { |     const replayButton = page.getByRole('button', { name: 'Replay onboarding' }) | ||||||
|         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') | ||||||
|       }) |       }) | ||||||
|       await expect(replayButton).toBeVisible() |     ).toContain('// Shelf Bracket') | ||||||
|       await replayButton.click() |   }) | ||||||
|  |  | ||||||
|       // Ensure we see the warning, and that the code has not yet updated |   test('Click through each onboarding step', async ({ page }) => { | ||||||
|       await expect(page.getByText('Would you like to create')).toBeVisible() |     const u = await getUtils(page) | ||||||
|       await expect(page.locator('.cm-content')).toHaveText(initialCode) |  | ||||||
|  |  | ||||||
|       const nextButton = page.getByTestId('onboarding-next') |     // Override beforeEach test setup | ||||||
|       await nextButton.hover() |     await page.addInitScript( | ||||||
|       await nextButton.click() |       async ({ settingsKey, settings }) => { | ||||||
|  |         // Give no initial code, so that the onboarding start is shown immediately | ||||||
|       // Ensure we see the introduction and that the code has been reset |         localStorage.setItem('persistCode', '') | ||||||
|       await expect(page.getByText('Welcome to Modeling App!')).toBeVisible() |         localStorage.setItem(settingsKey, settings) | ||||||
|       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 isn't the case on desktop. It gets |  | ||||||
|       // saved to the file system, which we have other tests for. |  | ||||||
|     } |  | ||||||
|   ) |  | ||||||
|  |  | ||||||
|   test( |  | ||||||
|     'Click through each onboarding step', |  | ||||||
|     { |  | ||||||
|       appSettings: { |  | ||||||
|         app: { |  | ||||||
|           onboardingStatus: 'incomplete', |  | ||||||
|         }, |  | ||||||
|       }, |       }, | ||||||
|     }, |       { | ||||||
|     async ({ context, page, homePage }) => { |         settingsKey: TEST_SETTINGS_KEY, | ||||||
|       // Override beforeEach test setup |         settings: TOML.stringify({ settings: TEST_SETTINGS_ONBOARDING_START }), | ||||||
|       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() |  | ||||||
|       } |       } | ||||||
|  |     ) | ||||||
|  |  | ||||||
|       // Finish the onboarding |     await page.setViewportSize({ width: 1200, height: 1080 }) | ||||||
|       await nextButton.hover() |  | ||||||
|  |     await u.waitForAuthSkipAppStart() | ||||||
|  |  | ||||||
|  |     // 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 expect(nextButton).toBeVisible() | ||||||
|       await nextButton.click() |       await nextButton.click() | ||||||
|  |  | ||||||
|       // Test that the onboarding pane is gone |  | ||||||
|       await expect(page.getByTestId('onboarding-content')).not.toBeVisible() |  | ||||||
|       await expect.poll(() => page.url()).not.toContain('/onboarding') |  | ||||||
|     } |     } | ||||||
|   ) |  | ||||||
|  |  | ||||||
|   test( |     // Finish the onboarding | ||||||
|     'Onboarding redirects and code updating', |     await expect(nextButton).toBeVisible() | ||||||
|     { |     await nextButton.click() | ||||||
|       appSettings: { |  | ||||||
|         app: { |     // Test that the onboarding pane is gone | ||||||
|           onboardingStatus: '/export', |     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) | ||||||
|       }, |       }, | ||||||
|       cleanProjectDir: true, |       { | ||||||
|     }, |         settingsKey: TEST_SETTINGS_KEY, | ||||||
|     async ({ context, page, homePage }) => { |         settings: TOML.stringify({ settings: TEST_SETTINGS_ONBOARDING_EXPORT }), | ||||||
|       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) |  | ||||||
|       } |       } | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|  |     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.fixme( | test( | ||||||
|   'Restarting onboarding on desktop takes one attempt', |   'Restarting onboarding on desktop takes one attempt', | ||||||
|   { |   { tag: '@electron' }, | ||||||
|     appSettings: { |   async ({ browser: _ }, testInfo) => { | ||||||
|       app: { |     const { electronApp, page } = await setupElectron({ | ||||||
|         onboardingStatus: 'dismissed', |       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') | ||||||
|  |         ) | ||||||
|       }, |       }, | ||||||
|     }, |  | ||||||
|     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 |     // Our constants | ||||||
| @ -466,8 +418,9 @@ test.fixme( | |||||||
|     const restartOnboardingButton = page.getByRole('button', { |     const restartOnboardingButton = page.getByRole('button', { | ||||||
|       name: 'Reset onboarding', |       name: 'Reset onboarding', | ||||||
|     }) |     }) | ||||||
|     const nextButton = page.getByTestId('onboarding-next') |     const restartConfirmationButton = page.getByRole('button', { | ||||||
|  |       name: 'Make a new project', | ||||||
|  |     }) | ||||||
|     const tutorialProjectIndicator = page |     const tutorialProjectIndicator = page | ||||||
|       .getByTestId('project-sidebar-toggle') |       .getByTestId('project-sidebar-toggle') | ||||||
|       .filter({ hasText: 'Tutorial Project 00' }) |       .filter({ hasText: 'Tutorial Project 00' }) | ||||||
| @ -486,7 +439,7 @@ test.fixme( | |||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await test.step('Navigate into project', async () => { |     await test.step('Navigate into project', async () => { | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |       await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|  |  | ||||||
|       page.on('console', console.log) |       page.on('console', console.log) | ||||||
|  |  | ||||||
| @ -502,22 +455,14 @@ test.fixme( | |||||||
|       await helpMenuButton.click() |       await helpMenuButton.click() | ||||||
|       await restartOnboardingButton.click() |       await restartOnboardingButton.click() | ||||||
|  |  | ||||||
|       await nextButton.hover() |       await expect(restartConfirmationButton).toBeVisible() | ||||||
|       await nextButton.click() |       await restartConfirmationButton.click() | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await test.step('Confirm that the onboarding has restarted', async () => { |     await test.step('Confirm that the onboarding has restarted', async () => { | ||||||
|       await expect(tutorialProjectIndicator).toBeVisible() |       await expect(tutorialProjectIndicator).toBeVisible() | ||||||
|       await expect(tutorialModalText).toBeVisible() |       await expect(tutorialModalText).toBeVisible() | ||||||
|       // Make sure the model loaded |  | ||||||
|       const XYPlanePoint = { x: 988, y: 523 } as const |  | ||||||
|       const modelColor: [number, number, number] = [76, 76, 76] |  | ||||||
|  |  | ||||||
|       await page.mouse.move(XYPlanePoint.x, XYPlanePoint.y) |  | ||||||
|       await expectPixelColor(page, modelColor, XYPlanePoint, 8) |  | ||||||
|       await tutorialDismissButton.click() |       await tutorialDismissButton.click() | ||||||
|       // Make sure model still there. |  | ||||||
|       await expectPixelColor(page, modelColor, XYPlanePoint, 8) |  | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await test.step('Clear code and restart onboarding from settings', async () => { |     await test.step('Clear code and restart onboarding from settings', async () => { | ||||||
| @ -535,9 +480,11 @@ test.fixme( | |||||||
|  |  | ||||||
|       await restartOnboardingSettingsButton.click() |       await restartOnboardingSettingsButton.click() | ||||||
|       // Since the code is empty, we should not see the confirmation dialog |       // Since the code is empty, we should not see the confirmation dialog | ||||||
|       await expect(nextButton).not.toBeVisible() |       await expect(restartConfirmationButton).not.toBeVisible() | ||||||
|       await expect(tutorialProjectIndicator).toBeVisible() |       await expect(tutorialProjectIndicator).toBeVisible() | ||||||
|       await expect(tutorialModalText).toBeVisible() |       await expect(tutorialModalText).toBeVisible() | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|  |     await electronApp.close() | ||||||
|   } |   } | ||||||
| ) | ) | ||||||
|  | |||||||
| @ -1,51 +1,25 @@ | |||||||
| import { test, expect, Page } from './zoo-test' | import { test, expect, AuthenticatedApp } from './fixtures/fixtureSetup' | ||||||
| import { EditorFixture } from './fixtures/editorFixture' | import { EditorFixture } from './fixtures/editorFixture' | ||||||
| import { SceneFixture } from './fixtures/sceneFixture' | import { SceneFixture } from './fixtures/sceneFixture' | ||||||
| import { ToolbarFixture } from './fixtures/toolbarFixture' | 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 file is for testing point an click code gen functionality that's not sketch mode related | ||||||
|  |  | ||||||
| test('verify extruding circle works', async ({ | test( | ||||||
|   context, |   'verify extruding circle works', | ||||||
|   homePage, |   { tag: ['@skipWin'] }, | ||||||
|   cmdBar, |   async ({ app, cmdBar, editor, toolbar, scene }) => { | ||||||
|   editor, |     test.skip( | ||||||
|   toolbar, |       process.platform === 'win32', | ||||||
|   scene, |       'Fails on windows in CI, can not be replicated locally on windows.' | ||||||
| }) => { |     ) | ||||||
|   // TODO: fix this test on windows after the electron migration |     const file = await app.getInputFile('test-circle-extrude.kcl') | ||||||
|   test.skip(process.platform === 'win32', 'Skip on windows') |     await app.initialise(file) | ||||||
|   const file = await fs.readFile( |     const [clickCircle, moveToCircle] = scene.makeMouseHelpers(582, 217) | ||||||
|     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() |  | ||||||
|  |  | ||||||
|   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 test.step('because there is sweepable geometry, verify extrude is enable when nothing is selected', async () => { |       await expect(toolbar.extrudeButton).toBeEnabled() | ||||||
|     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 () => { |     await test.step('check code model connection works and that button is still enable once circle is selected ', async () => { | ||||||
| @ -53,7 +27,7 @@ test('verify extruding circle works', async ({ | |||||||
|       const circleSnippet = |       const circleSnippet = | ||||||
|         'circle({ center = [318.33, 168.1], radius = 182.8 }, %)' |         'circle({ center = [318.33, 168.1], radius = 182.8 }, %)' | ||||||
|       await editor.expectState({ |       await editor.expectState({ | ||||||
|         activeLines: ["constsketch002=startSketchOn('XZ')"], |         activeLines: [], | ||||||
|         highlightedCode: circleSnippet, |         highlightedCode: circleSnippet, | ||||||
|         diagnostics: [], |         diagnostics: [], | ||||||
|       }) |       }) | ||||||
| @ -66,42 +40,39 @@ test('verify extruding circle works', async ({ | |||||||
|       }) |       }) | ||||||
|       await expect(toolbar.extrudeButton).toBeEnabled() |       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 test.step('do extrude flow and check extrude code is added to editor', async () => { | ||||||
|     await toolbar.extrudeButton.click() |       await toolbar.extrudeButton.click() | ||||||
|  |  | ||||||
|     await cmdBar.expectState({ |       await cmdBar.expectState({ | ||||||
|       stage: 'arguments', |         stage: 'arguments', | ||||||
|       currentArgKey: 'distance', |         currentArgKey: 'distance', | ||||||
|       currentArgValue: '5', |         currentArgValue: '5', | ||||||
|       headerArguments: { Selection: '1 face', Distance: '' }, |         headerArguments: { Selection: '1 face', Distance: '' }, | ||||||
|       highlightedHeaderArg: 'distance', |         highlightedHeaderArg: 'distance', | ||||||
|       commandName: 'Extrude', |         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.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', () => { | test.describe('verify sketch on chamfer works', () => { | ||||||
|   // TODO: fix this test on windows after the electron migration |  | ||||||
|   test.skip(process.platform === 'win32', 'Skip on windows') |  | ||||||
|   const _sketchOnAChamfer = |   const _sketchOnAChamfer = | ||||||
|     ( |     ( | ||||||
|       page: Page, |       app: AuthenticatedApp, | ||||||
|       editor: EditorFixture, |       editor: EditorFixture, | ||||||
|       toolbar: ToolbarFixture, |       toolbar: ToolbarFixture, | ||||||
|       scene: SceneFixture |       scene: SceneFixture | ||||||
| @ -153,7 +124,7 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|         await toolbar.startSketchPlaneSelection() |         await toolbar.startSketchPlaneSelection() | ||||||
|         await clickChamfer() |         await clickChamfer() | ||||||
|         // timeout wait for engine animation is unavoidable |         // timeout wait for engine animation is unavoidable | ||||||
|         await page.waitForTimeout(1000) |         await app.page.waitForTimeout(600) | ||||||
|         await editor.expectEditor.toContain(afterChamferSelectSnippet) |         await editor.expectEditor.toContain(afterChamferSelectSnippet) | ||||||
|       }) |       }) | ||||||
|       await test.step('make sure a basic sketch can be added', async () => { |       await test.step('make sure a basic sketch can be added', async () => { | ||||||
| @ -164,9 +135,7 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|           pixelDiff: 50, |           pixelDiff: 50, | ||||||
|         }) |         }) | ||||||
|         await rectangle2ndClick() |         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 () => { |       await test.step('Clean up so that `_sketchOnAChamfer` util can be called again', async () => { | ||||||
| @ -181,35 +150,24 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|         }) |         }) | ||||||
|       }) |       }) | ||||||
|     } |     } | ||||||
|   test('works on all edge selections and can break up multi edges in a chamfer array', async ({ |   test( | ||||||
|     context, |     'works on all edge selections and can break up multi edges in a chamfer array', | ||||||
|     page, |     { tag: ['@skipWin'] }, | ||||||
|     homePage, |     async ({ app, editor, toolbar, scene }) => { | ||||||
|     editor, |       test.skip( | ||||||
|     toolbar, |         process.platform === 'win32', | ||||||
|     scene, |         'Fails on windows in CI, can not be replicated locally on windows.' | ||||||
|   }) => { |       ) | ||||||
|     const file = await fs.readFile( |       const file = await app.getInputFile('e2e-can-sketch-on-chamfer.kcl') | ||||||
|       path.resolve( |       await app.initialise(file) | ||||||
|         __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(page, editor, toolbar, scene) |       const sketchOnAChamfer = _sketchOnAChamfer(app, editor, toolbar, scene) | ||||||
|  |  | ||||||
|     await sketchOnAChamfer({ |       await sketchOnAChamfer({ | ||||||
|       clickCoords: { x: 570, y: 220 }, |         clickCoords: { x: 570, y: 220 }, | ||||||
|       cameraPos: { x: 16020, y: -2000, z: 10500 }, |         cameraPos: { x: 16020, y: -2000, z: 10500 }, | ||||||
|       cameraTarget: { x: -150, y: -4500, z: -80 }, |         cameraTarget: { x: -150, y: -4500, z: -80 }, | ||||||
|       beforeChamferSnippet: `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01) |         beforeChamferSnippet: `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01) | ||||||
|       chamfer({length = 30,tags = [ |       chamfer({length = 30,tags = [ | ||||||
|       seg01, |       seg01, | ||||||
|       getNextAdjacentEdge(yo), |       getNextAdjacentEdge(yo), | ||||||
| @ -217,9 +175,10 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|       getOppositeEdge(seg01) |       getOppositeEdge(seg01) | ||||||
|     ]}, %)`, |     ]}, %)`, | ||||||
|  |  | ||||||
|       afterChamferSelectSnippet: 'sketch002 = startSketchOn(extrude001, seg03)', |         afterChamferSelectSnippet: | ||||||
|       afterRectangle1stClickSnippet: 'startProfileAt([205.96, 254.59], %)', |           'sketch002 = startSketchOn(extrude001, seg03)', | ||||||
|       afterRectangle2ndClickSnippet: `angledLine([0, 11.39], %, $rectangleSegmentA002) |         afterRectangle1stClickSnippet: 'startProfileAt([205.96, 254.59], %)', | ||||||
|  |         afterRectangle2ndClickSnippet: `angledLine([0, 11.39], %, $rectangleSegmentA002) | ||||||
|     |> angledLine([ |     |> angledLine([ | ||||||
|          segAng(rectangleSegmentA002) - 90, |          segAng(rectangleSegmentA002) - 90, | ||||||
|          105.26 |          105.26 | ||||||
| @ -230,13 +189,13 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|        ], %, $rectangleSegmentC001) |        ], %, $rectangleSegmentC001) | ||||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) |     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|     |> close(%)`, |     |> close(%)`, | ||||||
|     }) |       }) | ||||||
|  |  | ||||||
|     await sketchOnAChamfer({ |       await sketchOnAChamfer({ | ||||||
|       clickCoords: { x: 690, y: 250 }, |         clickCoords: { x: 690, y: 250 }, | ||||||
|       cameraPos: { x: 16020, y: -2000, z: 10500 }, |         cameraPos: { x: 16020, y: -2000, z: 10500 }, | ||||||
|       cameraTarget: { x: -150, y: -4500, z: -80 }, |         cameraTarget: { x: -150, y: -4500, z: -80 }, | ||||||
|       beforeChamferSnippet: `angledLine([ |         beforeChamferSnippet: `angledLine([ | ||||||
|          segAng(rectangleSegmentA001) - 90, |          segAng(rectangleSegmentA001) - 90, | ||||||
|          217.26 |          217.26 | ||||||
|        ], %, $seg01)chamfer({ |        ], %, $seg01)chamfer({ | ||||||
| @ -248,9 +207,10 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|          ] |          ] | ||||||
|        }, %)`, |        }, %)`, | ||||||
|  |  | ||||||
|       afterChamferSelectSnippet: 'sketch003 = startSketchOn(extrude001, seg04)', |         afterChamferSelectSnippet: | ||||||
|       afterRectangle1stClickSnippet: 'startProfileAt([-209.64, 255.28], %)', |           'sketch003 = startSketchOn(extrude001, seg04)', | ||||||
|       afterRectangle2ndClickSnippet: `angledLine([0, 11.56], %, $rectangleSegmentA003) |         afterRectangle1stClickSnippet: 'startProfileAt([-209.64, 255.28], %)', | ||||||
|  |         afterRectangle2ndClickSnippet: `angledLine([0, 11.56], %, $rectangleSegmentA003) | ||||||
|     |> angledLine([ |     |> angledLine([ | ||||||
|          segAng(rectangleSegmentA003) - 90, |          segAng(rectangleSegmentA003) - 90, | ||||||
|          106.84 |          106.84 | ||||||
| @ -261,21 +221,22 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|        ], %, $rectangleSegmentC002) |        ], %, $rectangleSegmentC002) | ||||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) |     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|     |> close(%)`, |     |> close(%)`, | ||||||
|     }) |       }) | ||||||
|     await sketchOnAChamfer({ |       await sketchOnAChamfer({ | ||||||
|       clickCoords: { x: 677, y: 87 }, |         clickCoords: { x: 677, y: 87 }, | ||||||
|       cameraPos: { x: -6200, y: 1500, z: 6200 }, |         cameraPos: { x: -6200, y: 1500, z: 6200 }, | ||||||
|       cameraTarget: { x: 8300, y: 1100, z: 4800 }, |         cameraTarget: { x: 8300, y: 1100, z: 4800 }, | ||||||
|       beforeChamferSnippet: `angledLine([0, 268.43], %, $rectangleSegmentA001)chamfer({ |         beforeChamferSnippet: `angledLine([0, 268.43], %, $rectangleSegmentA001)chamfer({ | ||||||
|          length = 30, |          length = 30, | ||||||
|          tags = [ |          tags = [ | ||||||
|            getNextAdjacentEdge(yo), |            getNextAdjacentEdge(yo), | ||||||
|            getNextAdjacentEdge(seg02) |            getNextAdjacentEdge(seg02) | ||||||
|          ] |          ] | ||||||
|        }, %)`, |        }, %)`, | ||||||
|       afterChamferSelectSnippet: 'sketch003 = startSketchOn(extrude001, seg04)', |         afterChamferSelectSnippet: | ||||||
|       afterRectangle1stClickSnippet: 'startProfileAt([75.8, 317.2], %)', |           'sketch003 = startSketchOn(extrude001, seg04)', | ||||||
|       afterRectangle2ndClickSnippet: `angledLine([0, 11.56], %, $rectangleSegmentA003) |         afterRectangle1stClickSnippet: 'startProfileAt([-209.64, 255.28], %)', | ||||||
|  |         afterRectangle2ndClickSnippet: `angledLine([0, 11.56], %, $rectangleSegmentA003) | ||||||
|     |> angledLine([ |     |> angledLine([ | ||||||
|          segAng(rectangleSegmentA003) - 90, |          segAng(rectangleSegmentA003) - 90, | ||||||
|          106.84 |          106.84 | ||||||
| @ -286,19 +247,20 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|        ], %, $rectangleSegmentC002) |        ], %, $rectangleSegmentC002) | ||||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) |     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|     |> close(%)`, |     |> close(%)`, | ||||||
|     }) |       }) | ||||||
|     /// last one |       /// last one | ||||||
|     await sketchOnAChamfer({ |       await sketchOnAChamfer({ | ||||||
|       clickCoords: { x: 620, y: 300 }, |         clickCoords: { x: 620, y: 300 }, | ||||||
|       cameraPos: { x: -1100, y: -7700, z: 1600 }, |         cameraPos: { x: -1100, y: -7700, z: 1600 }, | ||||||
|       cameraTarget: { x: 1450, y: 670, z: 4000 }, |         cameraTarget: { x: 1450, y: 670, z: 4000 }, | ||||||
|       beforeChamferSnippet: `chamfer({ |         beforeChamferSnippet: `chamfer({ | ||||||
|          length = 30, |          length = 30, | ||||||
|          tags = [getNextAdjacentEdge(yo)] |          tags = [getNextAdjacentEdge(yo)] | ||||||
|        }, %)`, |        }, %)`, | ||||||
|       afterChamferSelectSnippet: 'sketch005 = startSketchOn(extrude001, seg06)', |         afterChamferSelectSnippet: | ||||||
|       afterRectangle1stClickSnippet: 'startProfileAt([-23.43, 19.69], %)', |           'sketch005 = startSketchOn(extrude001, seg06)', | ||||||
|       afterRectangle2ndClickSnippet: `angledLine([0, 9.1], %, $rectangleSegmentA005) |         afterRectangle1stClickSnippet: 'startProfileAt([-23.43, 19.69], %)', | ||||||
|  |         afterRectangle2ndClickSnippet: `angledLine([0, 9.1], %, $rectangleSegmentA005) | ||||||
|  |  | ||||||
|     |> angledLine([ |     |> angledLine([ | ||||||
|          segAng(rectangleSegmentA005) - 90, |          segAng(rectangleSegmentA005) - 90, | ||||||
| @ -310,11 +272,11 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|        ], %, $rectangleSegmentC004) |        ], %, $rectangleSegmentC004) | ||||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) |     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|     |> close(%)`, |     |> close(%)`, | ||||||
|     }) |       }) | ||||||
|  |  | ||||||
|     await test.step('verify at the end of the test that final code is what is expected', async () => { |       await test.step('verify at the end of the test that final code is what is expected', async () => { | ||||||
|       await editor.expectEditor.toContain( |         await editor.expectEditor.toContain( | ||||||
|         `sketch001 = startSketchOn('XZ') |           `sketch001 = startSketchOn('XZ') | ||||||
|  |  | ||||||
|       |> startProfileAt([75.8, 317.2], %) // [$startCapTag, $EndCapTag] |       |> startProfileAt([75.8, 317.2], %) // [$startCapTag, $EndCapTag] | ||||||
|       |> angledLine([0, 268.43], %, $rectangleSegmentA001) |       |> angledLine([0, 268.43], %, $rectangleSegmentA001) | ||||||
| @ -343,7 +305,7 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|            tags = [getNextAdjacentEdge(yo)] |            tags = [getNextAdjacentEdge(yo)] | ||||||
|          }, %, $seg06) |          }, %, $seg06) | ||||||
|     sketch005 = startSketchOn(extrude001, seg06) |     sketch005 = startSketchOn(extrude001, seg06) | ||||||
|       |> startProfileAt([-23.43,19.69], %) |       |> startProfileAt([-23.43, 19.69], %) | ||||||
|       |> angledLine([0, 9.1], %, $rectangleSegmentA005) |       |> angledLine([0, 9.1], %, $rectangleSegmentA005) | ||||||
|       |> angledLine([ |       |> angledLine([ | ||||||
|            segAng(rectangleSegmentA005) - 90, |            segAng(rectangleSegmentA005) - 90, | ||||||
| @ -356,7 +318,7 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|       |> lineTo([profileStartX(%), profileStartY(%)], %) |       |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|       |> close(%) |       |> close(%) | ||||||
|     sketch004 = startSketchOn(extrude001, seg05) |     sketch004 = startSketchOn(extrude001, seg05) | ||||||
|       |> startProfileAt([82.57,322.96], %) |       |> startProfileAt([82.57, 322.96], %) | ||||||
|       |> angledLine([0, 11.16], %, $rectangleSegmentA004) |       |> angledLine([0, 11.16], %, $rectangleSegmentA004) | ||||||
|       |> angledLine([ |       |> angledLine([ | ||||||
|            segAng(rectangleSegmentA004) - 90, |            segAng(rectangleSegmentA004) - 90, | ||||||
| @ -369,7 +331,7 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|       |> lineTo([profileStartX(%), profileStartY(%)], %) |       |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|       |> close(%) |       |> close(%) | ||||||
|     sketch003 = startSketchOn(extrude001, seg04) |     sketch003 = startSketchOn(extrude001, seg04) | ||||||
|       |> startProfileAt([-209.64,255.28], %) |       |> startProfileAt([-209.64, 255.28], %) | ||||||
|       |> angledLine([0, 11.56], %, $rectangleSegmentA003) |       |> angledLine([0, 11.56], %, $rectangleSegmentA003) | ||||||
|       |> angledLine([ |       |> angledLine([ | ||||||
|            segAng(rectangleSegmentA003) - 90, |            segAng(rectangleSegmentA003) - 90, | ||||||
| @ -382,7 +344,7 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|       |> lineTo([profileStartX(%), profileStartY(%)], %) |       |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|       |> close(%) |       |> close(%) | ||||||
|     sketch002 = startSketchOn(extrude001, seg03) |     sketch002 = startSketchOn(extrude001, seg03) | ||||||
|       |> startProfileAt([205.96,254.59], %) |       |> startProfileAt([205.96, 254.59], %) | ||||||
|       |> angledLine([0, 11.39], %, $rectangleSegmentA002) |       |> angledLine([0, 11.39], %, $rectangleSegmentA002) | ||||||
|       |> angledLine([ |       |> angledLine([ | ||||||
|            segAng(rectangleSegmentA002) - 90, |            segAng(rectangleSegmentA002) - 90, | ||||||
| @ -395,50 +357,43 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|       |> lineTo([profileStartX(%), profileStartY(%)], %) |       |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|       |> close(%) |       |> close(%) | ||||||
|     `, |     `, | ||||||
|         { shouldNormalise: true } |           { 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.' | ||||||
|       ) |       ) | ||||||
|     }) |       const file = await app.getInputFile( | ||||||
|   }) |         'e2e-can-sketch-on-chamfer-no-pipeExpr.kcl' | ||||||
|  |       ) | ||||||
|  |       await app.initialise(file) | ||||||
|  |  | ||||||
|   test('Works on chamfers that are non in a pipeExpression can break up multi edges in a chamfer array', async ({ |       const sketchOnAChamfer = _sketchOnAChamfer(app, editor, toolbar, scene) | ||||||
|     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() |  | ||||||
|  |  | ||||||
|     const sketchOnAChamfer = _sketchOnAChamfer(page, editor, toolbar, scene) |       await sketchOnAChamfer({ | ||||||
|  |         clickCoords: { x: 570, y: 220 }, | ||||||
|     await sketchOnAChamfer({ |         cameraPos: { x: 16020, y: -2000, z: 10500 }, | ||||||
|       clickCoords: { x: 570, y: 220 }, |         cameraTarget: { x: -150, y: -4500, z: -80 }, | ||||||
|       cameraPos: { x: 16020, y: -2000, z: 10500 }, |         beforeChamferSnippet: `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01) | ||||||
|       cameraTarget: { x: -150, y: -4500, z: -80 }, |  | ||||||
|       beforeChamferSnippet: `angledLine([segAng(rectangleSegmentA001)-90,217.26],%,$seg01) |  | ||||||
|       chamfer({length=30,tags=[ |       chamfer({length=30,tags=[ | ||||||
|       seg01, |       seg01, | ||||||
|       getNextAdjacentEdge(yo), |       getNextAdjacentEdge(yo), | ||||||
|       getNextAdjacentEdge(seg02), |       getNextAdjacentEdge(seg02), | ||||||
|       getOppositeEdge(seg01) |       getOppositeEdge(seg01) | ||||||
|     ]}, extrude001)`, |     ]}, extrude001)`, | ||||||
|       beforeChamferSnippetEnd: '}, extrude001)', |         beforeChamferSnippetEnd: '}, extrude001)', | ||||||
|       afterChamferSelectSnippet: 'sketch002 = startSketchOn(extrude001, seg03)', |         afterChamferSelectSnippet: | ||||||
|       afterRectangle1stClickSnippet: 'startProfileAt([205.96, 254.59], %)', |           'sketch002 = startSketchOn(extrude001, seg03)', | ||||||
|       afterRectangle2ndClickSnippet: `angledLine([0, 11.39], %, $rectangleSegmentA002) |         afterRectangle1stClickSnippet: 'startProfileAt([205.96, 254.59], %)', | ||||||
|  |         afterRectangle2ndClickSnippet: `angledLine([0, 11.39], %, $rectangleSegmentA002) | ||||||
|     |> angledLine([ |     |> angledLine([ | ||||||
|          segAng(rectangleSegmentA002) - 90, |          segAng(rectangleSegmentA002) - 90, | ||||||
|          105.26 |          105.26 | ||||||
| @ -449,9 +404,9 @@ test.describe('verify sketch on chamfer works', () => { | |||||||
|        ], %, $rectangleSegmentC001) |        ], %, $rectangleSegmentC001) | ||||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) |     |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|     |> close(%)`, |     |> close(%)`, | ||||||
|     }) |       }) | ||||||
|     await editor.expectEditor.toContain( |       await editor.expectEditor.toContain( | ||||||
|       `sketch001 = startSketchOn('XZ') |         `sketch001 = startSketchOn('XZ') | ||||||
|   |> startProfileAt([75.8, 317.2], %) |   |> startProfileAt([75.8, 317.2], %) | ||||||
|   |> angledLine([0, 268.43], %, $rectangleSegmentA001) |   |> angledLine([0, 268.43], %, $rectangleSegmentA001) | ||||||
|   |> angledLine([ |   |> angledLine([ | ||||||
| @ -491,56 +446,50 @@ sketch002 = startSketchOn(extrude001, seg03) | |||||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) |   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|   |> close(%) |   |> close(%) | ||||||
| `, | `, | ||||||
|       { shouldNormalise: true } |         { shouldNormalise: true } | ||||||
|     ) |       ) | ||||||
|   }) |     } | ||||||
|  |   ) | ||||||
| }) | }) | ||||||
|  |  | ||||||
| test(`Verify axis, origin, and horizontal snapping`, async ({ | test(`Verify axis, origin, and horizontal snapping`, async ({ | ||||||
|   page, |   app, | ||||||
|   homePage, |  | ||||||
|   editor, |   editor, | ||||||
|   toolbar, |   toolbar, | ||||||
|   scene, |   scene, | ||||||
| }) => { | }) => { | ||||||
|   const viewPortSize = { width: 1200, height: 500 } |  | ||||||
|  |  | ||||||
|   await page.setBodyDimensions(viewPortSize) |  | ||||||
|  |  | ||||||
|   await homePage.goToModelingScene() |  | ||||||
|  |  | ||||||
|   // Constants and locators |   // Constants and locators | ||||||
|   // These are mappings from screenspace to KCL coordinates, |   // These are mappings from screenspace to KCL coordinates, | ||||||
|   // until we merge in our coordinate system helpers |   // until we merge in our coordinate system helpers | ||||||
|   const xzPlane = [ |   const xzPlane = [ | ||||||
|     viewPortSize.width * 0.65, |     app.viewPortSize.width * 0.65, | ||||||
|     viewPortSize.height * 0.3, |     app.viewPortSize.height * 0.3, | ||||||
|   ] as const |   ] as const | ||||||
|   const originSloppy = { |   const originSloppy = { | ||||||
|     screen: [ |     screen: [ | ||||||
|       viewPortSize.width / 2 + 3, // 3px off the center of the screen |       app.viewPortSize.width / 2 + 3, // 3px off the center of the screen | ||||||
|       viewPortSize.height / 2, |       app.viewPortSize.height / 2, | ||||||
|     ], |     ], | ||||||
|     kcl: [0, 0], |     kcl: [0, 0], | ||||||
|   } as const |   } as const | ||||||
|   const xAxisSloppy = { |   const xAxisSloppy = { | ||||||
|     screen: [ |     screen: [ | ||||||
|       viewPortSize.width * 0.75, |       app.viewPortSize.width * 0.75, | ||||||
|       viewPortSize.height / 2 - 3, // 3px off the X-axis |       app.viewPortSize.height / 2 - 3, // 3px off the X-axis | ||||||
|     ], |     ], | ||||||
|     kcl: [20.34, 0], |     kcl: [16.95, 0], | ||||||
|   } as const |   } as const | ||||||
|   const offYAxis = { |   const offYAxis = { | ||||||
|     screen: [ |     screen: [ | ||||||
|       viewPortSize.width * 0.6, // Well off the Y-axis, out of snapping range |       app.viewPortSize.width * 0.6, // Well off the Y-axis, out of snapping range | ||||||
|       viewPortSize.height * 0.3, |       app.viewPortSize.height * 0.3, | ||||||
|     ], |     ], | ||||||
|     kcl: [8.14, 6.78], |     kcl: [6.78, 6.78], | ||||||
|   } as const |   } as const | ||||||
|   const yAxisSloppy = { |   const yAxisSloppy = { | ||||||
|     screen: [ |     screen: [ | ||||||
|       viewPortSize.width / 2 + 5, // 5px off the Y-axis |       app.viewPortSize.width / 2 + 5, // 5px off the Y-axis | ||||||
|       viewPortSize.height * 0.3, |       app.viewPortSize.height * 0.3, | ||||||
|     ], |     ], | ||||||
|     kcl: [0, 6.78], |     kcl: [0, 6.78], | ||||||
|   } as const |   } as const | ||||||
| @ -561,13 +510,15 @@ test(`Verify axis, origin, and horizontal snapping`, async ({ | |||||||
|     afterSegmentDraggedOnYAxis: `startProfileAt([${yAxisSloppy.kcl[0]}, ${yAxisSloppy.kcl[1]}], %)`, |     afterSegmentDraggedOnYAxis: `startProfileAt([${yAxisSloppy.kcl[0]}, ${yAxisSloppy.kcl[1]}], %)`, | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   await app.initialise() | ||||||
|  |  | ||||||
|   await test.step(`Start a sketch on the XZ plane`, async () => { |   await test.step(`Start a sketch on the XZ plane`, async () => { | ||||||
|     await editor.closePane() |     await editor.closePane() | ||||||
|     await toolbar.startSketchPlaneSelection() |     await toolbar.startSketchPlaneSelection() | ||||||
|     await moveToXzPlane() |     await moveToXzPlane() | ||||||
|     await clickOnXzPlane() |     await clickOnXzPlane() | ||||||
|     // timeout wait for engine animation is unavoidable |     // timeout wait for engine animation is unavoidable | ||||||
|     await page.waitForTimeout(600) |     await app.page.waitForTimeout(600) | ||||||
|     await editor.expectEditor.toContain(expectedCodeSnippets.sketchOnXzPlane) |     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 () => { |   await test.step(`Place a point a few pixels off the middle, verify it still snaps to 0,0`, async () => { | ||||||
| @ -602,15 +553,11 @@ test(`Verify axis, origin, and horizontal snapping`, async ({ | |||||||
| }) | }) | ||||||
|  |  | ||||||
| test(`Verify user can double-click to edit a sketch`, async ({ | test(`Verify user can double-click to edit a sketch`, async ({ | ||||||
|   context, |   app, | ||||||
|   page, |  | ||||||
|   homePage, |  | ||||||
|   editor, |   editor, | ||||||
|   toolbar, |   toolbar, | ||||||
|   scene, |   scene, | ||||||
| }) => { | }) => { | ||||||
|   const u = await getUtils(page) |  | ||||||
|  |  | ||||||
|   const initialCode = `closedSketch = startSketchOn('XZ') |   const initialCode = `closedSketch = startSketchOn('XZ') | ||||||
|   |> circle({ center = [8, 5], radius = 2 }, %) |   |> circle({ center = [8, 5], radius = 2 }, %) | ||||||
| openSketch = startSketchOn('XY') | openSketch = startSketchOn('XY') | ||||||
| @ -619,24 +566,15 @@ openSketch = startSketchOn('XY') | |||||||
|   |> xLine(5, %) |   |> xLine(5, %) | ||||||
|   |> tangentialArcTo([10, 0], %) |   |> tangentialArcTo([10, 0], %) | ||||||
| ` | ` | ||||||
|   const viewPortSize = { width: 1000, height: 500 } |   await app.initialise(initialCode) | ||||||
|   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 = { |   const pointInsideCircle = { | ||||||
|     x: viewPortSize.width * 0.63, |     x: app.viewPortSize.width * 0.63, | ||||||
|     y: viewPortSize.height * 0.5, |     y: app.viewPortSize.height * 0.5, | ||||||
|   } |   } | ||||||
|   const pointOnPathAfterSketching = { |   const pointOnPathAfterSketching = { | ||||||
|     x: viewPortSize.width * 0.65, |     x: app.viewPortSize.width * 0.58, | ||||||
|     y: viewPortSize.height * 0.5, |     y: app.viewPortSize.height * 0.5, | ||||||
|   } |   } | ||||||
|   // eslint-disable-next-line @typescript-eslint/no-unused-vars |   // eslint-disable-next-line @typescript-eslint/no-unused-vars | ||||||
|   const [_clickOpenPath, moveToOpenPath, dblClickOpenPath] = |   const [_clickOpenPath, moveToOpenPath, dblClickOpenPath] = | ||||||
| @ -669,59 +607,41 @@ openSketch = startSketchOn('XY') | |||||||
|       diagnostics: [], |       diagnostics: [], | ||||||
|     }) |     }) | ||||||
|   }) |   }) | ||||||
|   await page.waitForTimeout(1000) |  | ||||||
|  |  | ||||||
|   await exitSketch() |   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 test.step(`Double-click on the open sketch`, async () => { | ||||||
|     await moveToOpenPath() |     await moveToOpenPath() | ||||||
|     await scene.expectPixelColor([250, 250, 250], pointOnPathAfterSketching, 15) |     await scene.expectPixelColor([250, 250, 250], pointOnPathAfterSketching, 15) | ||||||
|     // There is a full execution after exiting sketch that clears the scene. |     // There is a full execution after exiting sketch that clears the scene. | ||||||
|     await page.waitForTimeout(500) |     await app.page.waitForTimeout(500) | ||||||
|     await dblClickOpenPath() |     await dblClickOpenPath() | ||||||
|     await expect(toolbar.startSketchBtn).not.toBeVisible() |     await expect(toolbar.startSketchBtn).not.toBeVisible() | ||||||
|     await expect(toolbar.exitSketchBtn).toBeVisible() |     await expect(toolbar.exitSketchBtn).toBeVisible() | ||||||
|     // Wait for enter sketch mode to complete |     // Wait for enter sketch mode to complete | ||||||
|     await page.waitForTimeout(500) |     await app.page.waitForTimeout(500) | ||||||
|     await editor.expectState({ |     await editor.expectState({ | ||||||
|       activeLines: [`|>tangentialArcTo([10,0],%)`], |       activeLines: [`|>xLine(5,%)`], | ||||||
|       highlightedCode: 'tangentialArcTo([10,0],%)', |       highlightedCode: 'xLine(5,%)', | ||||||
|       diagnostics: [], |       diagnostics: [], | ||||||
|     }) |     }) | ||||||
|   }) |   }) | ||||||
| }) | }) | ||||||
|  |  | ||||||
| test(`Offset plane point-and-click`, async ({ | test(`Offset plane point-and-click`, async ({ | ||||||
|   context, |   app, | ||||||
|   page, |  | ||||||
|   homePage, |  | ||||||
|   scene, |   scene, | ||||||
|   editor, |   editor, | ||||||
|   toolbar, |   toolbar, | ||||||
|   cmdBar, |   cmdBar, | ||||||
| }) => { | }) => { | ||||||
|  |   await app.initialise() | ||||||
|  |  | ||||||
|   // One dumb hardcoded screen pixel value |   // One dumb hardcoded screen pixel value | ||||||
|   const testPoint = { x: 700, y: 150 } |   const testPoint = { x: 700, y: 150 } | ||||||
|   const [clickOnXzPlane] = scene.makeMouseHelpers(testPoint.x, testPoint.y) |   const [clickOnXzPlane] = scene.makeMouseHelpers(testPoint.x, testPoint.y) | ||||||
|   const expectedOutput = `plane001 = offsetPlane('XZ', 5)` |   const expectedOutput = `plane001 = offsetPlane('XZ', 5)` | ||||||
|  |  | ||||||
|   await homePage.goToModelingScene() |  | ||||||
|  |  | ||||||
|   await test.step(`Look for the blue of the XZ plane`, async () => { |   await test.step(`Look for the blue of the XZ plane`, async () => { | ||||||
|     await scene.expectPixelColor([50, 51, 96], testPoint, 15) |     await scene.expectPixelColor([50, 51, 96], testPoint, 15) | ||||||
|   }) |   }) | ||||||
| @ -764,9 +684,8 @@ const loftPointAndClickCases = [ | |||||||
| ] | ] | ||||||
| loftPointAndClickCases.forEach(({ shouldPreselect }) => { | loftPointAndClickCases.forEach(({ shouldPreselect }) => { | ||||||
|   test(`Loft point-and-click (preselected sketches: ${shouldPreselect})`, async ({ |   test(`Loft point-and-click (preselected sketches: ${shouldPreselect})`, async ({ | ||||||
|     context, |     app, | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|     scene, |     scene, | ||||||
|     editor, |     editor, | ||||||
|     toolbar, |     toolbar, | ||||||
| @ -778,11 +697,7 @@ loftPointAndClickCases.forEach(({ shouldPreselect }) => { | |||||||
|     sketch002 = startSketchOn(plane001) |     sketch002 = startSketchOn(plane001) | ||||||
|     |> circle({ center = [0, 0], radius = 20 }, %) |     |> circle({ center = [0, 0], radius = 20 }, %) | ||||||
| ` | ` | ||||||
|     await context.addInitScript((initialCode) => { |     await app.initialise(initialCode) | ||||||
|       localStorage.setItem('persistCode', initialCode) |  | ||||||
|     }, initialCode) |  | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |  | ||||||
|     await homePage.goToModelingScene() |  | ||||||
|  |  | ||||||
|     // One dumb hardcoded screen pixel value |     // One dumb hardcoded screen pixel value | ||||||
|     const testPoint = { x: 575, y: 200 } |     const testPoint = { x: 575, y: 200 } | ||||||
| @ -801,7 +716,7 @@ loftPointAndClickCases.forEach(({ shouldPreselect }) => { | |||||||
|       await clickOnSketch1() |       await clickOnSketch1() | ||||||
|       await page.keyboard.down('Shift') |       await page.keyboard.down('Shift') | ||||||
|       await clickOnSketch2() |       await clickOnSketch2() | ||||||
|       await page.waitForTimeout(500) |       await app.page.waitForTimeout(500) | ||||||
|       await page.keyboard.up('Shift') |       await page.keyboard.up('Shift') | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -860,25 +775,17 @@ const shellPointAndClickCapCases = [ | |||||||
| ] | ] | ||||||
| shellPointAndClickCapCases.forEach(({ shouldPreselect }) => { | shellPointAndClickCapCases.forEach(({ shouldPreselect }) => { | ||||||
|   test(`Shell point-and-click cap (preselected sketches: ${shouldPreselect})`, async ({ |   test(`Shell point-and-click cap (preselected sketches: ${shouldPreselect})`, async ({ | ||||||
|     context, |     app, | ||||||
|     page, |  | ||||||
|     homePage, |  | ||||||
|     scene, |     scene, | ||||||
|     editor, |     editor, | ||||||
|     toolbar, |     toolbar, | ||||||
|     cmdBar, |     cmdBar, | ||||||
|   }) => { |   }) => { | ||||||
|     // TODO: fix this test on windows after the electron migration |  | ||||||
|     test.skip(process.platform === 'win32', 'Skip on windows') |  | ||||||
|     const initialCode = `sketch001 = startSketchOn('XZ') |     const initialCode = `sketch001 = startSketchOn('XZ') | ||||||
|     |> circle({ center = [0, 0], radius = 30 }, %) |     |> circle({ center = [0, 0], radius = 30 }, %) | ||||||
|     extrude001 = extrude(30, sketch001) |     extrude001 = extrude(30, sketch001) | ||||||
|     ` |     ` | ||||||
|     await context.addInitScript((initialCode) => { |     await app.initialise(initialCode) | ||||||
|       localStorage.setItem('persistCode', initialCode) |  | ||||||
|     }, initialCode) |  | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |  | ||||||
|     await homePage.goToModelingScene() |  | ||||||
|  |  | ||||||
|     // One dumb hardcoded screen pixel value |     // One dumb hardcoded screen pixel value | ||||||
|     const testPoint = { x: 575, y: 200 } |     const testPoint = { x: 575, y: 200 } | ||||||
| @ -905,7 +812,7 @@ shellPointAndClickCapCases.forEach(({ shouldPreselect }) => { | |||||||
|           commandName: 'Shell', |           commandName: 'Shell', | ||||||
|         }) |         }) | ||||||
|         await clickOnCap() |         await clickOnCap() | ||||||
|         await page.waitForTimeout(500) |         await app.page.waitForTimeout(500) | ||||||
|         await cmdBar.progressCmdBar() |         await cmdBar.progressCmdBar() | ||||||
|         await cmdBar.progressCmdBar() |         await cmdBar.progressCmdBar() | ||||||
|         await cmdBar.expectState({ |         await cmdBar.expectState({ | ||||||
| @ -921,7 +828,7 @@ shellPointAndClickCapCases.forEach(({ shouldPreselect }) => { | |||||||
|     } else { |     } else { | ||||||
|       await test.step(`Preselect the cap`, async () => { |       await test.step(`Preselect the cap`, async () => { | ||||||
|         await clickOnCap() |         await clickOnCap() | ||||||
|         await page.waitForTimeout(500) |         await app.page.waitForTimeout(500) | ||||||
|       }) |       }) | ||||||
|  |  | ||||||
|       await test.step(`Go through the command bar flow with a preselected face (cap)`, async () => { |       await test.step(`Go through the command bar flow with a preselected face (cap)`, async () => { | ||||||
| @ -953,9 +860,8 @@ shellPointAndClickCapCases.forEach(({ shouldPreselect }) => { | |||||||
| }) | }) | ||||||
|  |  | ||||||
| test('Shell point-and-click wall', async ({ | test('Shell point-and-click wall', async ({ | ||||||
|   context, |   app, | ||||||
|   page, |   page, | ||||||
|   homePage, |  | ||||||
|   scene, |   scene, | ||||||
|   editor, |   editor, | ||||||
|   toolbar, |   toolbar, | ||||||
| @ -970,11 +876,7 @@ test('Shell point-and-click wall', async ({ | |||||||
|   |> close(%) |   |> close(%) | ||||||
| extrude001 = extrude(40, sketch001) | extrude001 = extrude(40, sketch001) | ||||||
|   ` |   ` | ||||||
|   await context.addInitScript((initialCode) => { |   await app.initialise(initialCode) | ||||||
|     localStorage.setItem('persistCode', initialCode) |  | ||||||
|   }, initialCode) |  | ||||||
|   await page.setBodyDimensions({ width: 1000, height: 500 }) |  | ||||||
|   await homePage.goToModelingScene() |  | ||||||
|  |  | ||||||
|   // One dumb hardcoded screen pixel value |   // One dumb hardcoded screen pixel value | ||||||
|   const testPoint = { x: 580, y: 180 } |   const testPoint = { x: 580, y: 180 } | ||||||
| @ -1005,7 +907,7 @@ extrude001 = extrude(40, sketch001) | |||||||
|     await clickOnCap() |     await clickOnCap() | ||||||
|     await page.keyboard.down('Shift') |     await page.keyboard.down('Shift') | ||||||
|     await clickOnWall() |     await clickOnWall() | ||||||
|     await page.waitForTimeout(500) |     await app.page.waitForTimeout(500) | ||||||
|     await page.keyboard.up('Shift') |     await page.keyboard.up('Shift') | ||||||
|     await cmdBar.progressCmdBar() |     await cmdBar.progressCmdBar() | ||||||
|     await cmdBar.progressCmdBar() |     await cmdBar.progressCmdBar() | ||||||
|  | |||||||
| @ -1,190 +0,0 @@ | |||||||
| import { test, expect } from './zoo-test' |  | ||||||
|  |  | ||||||
| /* eslint-disable jest/no-conditional-expect */ |  | ||||||
|  |  | ||||||
| const file = `sketch001 = startSketchOn('XZ') |  | ||||||
| profile001 = startProfileAt([57.81, 250.51], sketch001) |  | ||||||
|   |> line([121.13, 56.63], %, $seg02) |  | ||||||
|   |> line([83.37, -34.61], %, $seg01) |  | ||||||
|   |> line([19.66, -116.4], %) |  | ||||||
|   |> line([-221.8, -41.69], %) |  | ||||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) |  | ||||||
|   |> close(%) |  | ||||||
| extrude001 = extrude(200, profile001) |  | ||||||
| sketch002 = startSketchOn('XZ') |  | ||||||
|   |> startProfileAt([-73.64, -42.89], %) |  | ||||||
|   |> xLine(173.71, %) |  | ||||||
|   |> line([-22.12, -94.4], %) |  | ||||||
|   |> xLine(-156.98, %) |  | ||||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) |  | ||||||
|   |> close(%) |  | ||||||
| extrude002 = extrude(50, sketch002) |  | ||||||
| sketch003 = startSketchOn('XY') |  | ||||||
|   |> startProfileAt([52.92, 157.81], %) |  | ||||||
|   |> angledLine([0, 176.4], %, $rectangleSegmentA001) |  | ||||||
|   |> angledLine([ |  | ||||||
|        segAng(rectangleSegmentA001) - 90, |  | ||||||
|        53.4 |  | ||||||
|      ], %, $rectangleSegmentB001) |  | ||||||
|   |> angledLine([ |  | ||||||
|        segAng(rectangleSegmentA001), |  | ||||||
|        -segLen(rectangleSegmentA001) |  | ||||||
|      ], %, $rectangleSegmentC001) |  | ||||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) |  | ||||||
|   |> close(%) |  | ||||||
| extrude003 = extrude(20, sketch003) |  | ||||||
| ` |  | ||||||
|  |  | ||||||
| test.describe('Check the happy path, for basic changing color', () => { |  | ||||||
|   const cases = [ |  | ||||||
|     { |  | ||||||
|       desc: 'User accepts change', |  | ||||||
|       shouldReject: false, |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       desc: 'User rejects change', |  | ||||||
|       shouldReject: true, |  | ||||||
|     }, |  | ||||||
|   ] as const |  | ||||||
|   for (const { desc, shouldReject } of cases) { |  | ||||||
|     test(`${desc}`, async ({ |  | ||||||
|       context, |  | ||||||
|       homePage, |  | ||||||
|       cmdBar, |  | ||||||
|       editor, |  | ||||||
|       page, |  | ||||||
|       scene, |  | ||||||
|     }) => { |  | ||||||
|       await context.addInitScript((file) => { |  | ||||||
|         localStorage.setItem('persistCode', file) |  | ||||||
|       }, file) |  | ||||||
|       await homePage.goToModelingScene() |  | ||||||
|  |  | ||||||
|       const body1CapCoords = { x: 571, y: 351 } |  | ||||||
|       const greenCheckCoords = { x: 565, y: 345 } |  | ||||||
|       const body2WallCoords = { x: 609, y: 153 } |  | ||||||
|       const [clickBody1Cap] = scene.makeMouseHelpers( |  | ||||||
|         body1CapCoords.x, |  | ||||||
|         body1CapCoords.y |  | ||||||
|       ) |  | ||||||
|       const yellow: [number, number, number] = [179, 179, 131] |  | ||||||
|       const green: [number, number, number] = [108, 152, 75] |  | ||||||
|       const notGreen: [number, number, number] = [132, 132, 132] |  | ||||||
|       const body2NotGreen: [number, number, number] = [88, 88, 88] |  | ||||||
|       const submittingToast = page.getByText('Submitting to Text-to-CAD API...') |  | ||||||
|       const successToast = page.getByText('Prompt to edit successful') |  | ||||||
|       const acceptBtn = page.getByRole('button', { name: 'checkmark Accept' }) |  | ||||||
|       const rejectBtn = page.getByRole('button', { name: 'close Reject' }) |  | ||||||
|  |  | ||||||
|       await test.step('wait for scene to load select body and check selection came through', async () => { |  | ||||||
|         await scene.expectPixelColor([134, 134, 134], body1CapCoords, 15) |  | ||||||
|         await clickBody1Cap() |  | ||||||
|         await scene.expectPixelColor(yellow, body1CapCoords, 20) |  | ||||||
|         await editor.expectState({ |  | ||||||
|           highlightedCode: '', |  | ||||||
|           activeLines: ['|>startProfileAt([-73.64,-42.89],%)'], |  | ||||||
|           diagnostics: [], |  | ||||||
|         }) |  | ||||||
|       }) |  | ||||||
|  |  | ||||||
|       await test.step('fire off edit prompt', async () => { |  | ||||||
|         await cmdBar.openCmdBar('promptToEdit') |  | ||||||
|         // being specific about the color with a hex means asserting pixel color is more stable |  | ||||||
|         await page |  | ||||||
|           .getByTestId('cmd-bar-arg-value') |  | ||||||
|           .fill('make this neon green please, use #39FF14') |  | ||||||
|         await page.waitForTimeout(100) |  | ||||||
|         await cmdBar.progressCmdBar() |  | ||||||
|         await expect(submittingToast).toBeVisible() |  | ||||||
|         await expect(submittingToast).not.toBeVisible({ timeout: 2 * 60_000 }) // can take a while |  | ||||||
|         await expect(successToast).toBeVisible() |  | ||||||
|       }) |  | ||||||
|  |  | ||||||
|       await test.step('verify initial change', async () => { |  | ||||||
|         await scene.expectPixelColor(green, greenCheckCoords, 15) |  | ||||||
|         await scene.expectPixelColor(body2NotGreen, body2WallCoords, 15) |  | ||||||
|         await editor.expectEditor.toContain('appearance({') |  | ||||||
|       }) |  | ||||||
|  |  | ||||||
|       if (!shouldReject) { |  | ||||||
|         await test.step('check accept works and can be "undo"ed', async () => { |  | ||||||
|           await acceptBtn.click() |  | ||||||
|           await expect(successToast).not.toBeVisible() |  | ||||||
|  |  | ||||||
|           await scene.expectPixelColor(green, greenCheckCoords, 15) |  | ||||||
|           await editor.expectEditor.toContain('appearance({') |  | ||||||
|  |  | ||||||
|           // ctrl-z works after accepting |  | ||||||
|           await page.keyboard.down('ControlOrMeta') |  | ||||||
|           await page.keyboard.press('KeyZ') |  | ||||||
|           await page.keyboard.up('ControlOrMeta') |  | ||||||
|           await editor.expectEditor.not.toContain('appearance({') |  | ||||||
|           await scene.expectPixelColor(notGreen, greenCheckCoords, 15) |  | ||||||
|         }) |  | ||||||
|       } else { |  | ||||||
|         await test.step('check reject works', async () => { |  | ||||||
|           await rejectBtn.click() |  | ||||||
|           await expect(successToast).not.toBeVisible() |  | ||||||
|  |  | ||||||
|           await scene.expectPixelColor(notGreen, greenCheckCoords, 15) |  | ||||||
|           await editor.expectEditor.not.toContain('appearance({') |  | ||||||
|         }) |  | ||||||
|       } |  | ||||||
|     }) |  | ||||||
|   } |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| test.describe('bad path', () => { |  | ||||||
|   test(`bad edit prompt`, async ({ |  | ||||||
|     context, |  | ||||||
|     homePage, |  | ||||||
|     cmdBar, |  | ||||||
|     editor, |  | ||||||
|     toolbar, |  | ||||||
|     page, |  | ||||||
|     scene, |  | ||||||
|   }) => { |  | ||||||
|     await context.addInitScript((file) => { |  | ||||||
|       localStorage.setItem('persistCode', file) |  | ||||||
|     }, file) |  | ||||||
|     await homePage.goToModelingScene() |  | ||||||
|  |  | ||||||
|     const body1CapCoords = { x: 571, y: 351 } |  | ||||||
|     const [clickBody1Cap] = scene.makeMouseHelpers( |  | ||||||
|       body1CapCoords.x, |  | ||||||
|       body1CapCoords.y |  | ||||||
|     ) |  | ||||||
|     const yellow: [number, number, number] = [179, 179, 131] |  | ||||||
|     const submittingToast = page.getByText('Submitting to Text-to-CAD API...') |  | ||||||
|     const failToast = page.getByText( |  | ||||||
|       'Failed to edit your KCL code, please try again with a different prompt or selection' |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     await test.step('wait for scene to load and select body', async () => { |  | ||||||
|       await scene.expectPixelColor([134, 134, 134], body1CapCoords, 15) |  | ||||||
|  |  | ||||||
|       await clickBody1Cap() |  | ||||||
|       await scene.expectPixelColor(yellow, body1CapCoords, 20) |  | ||||||
|  |  | ||||||
|       await editor.expectState({ |  | ||||||
|         highlightedCode: '', |  | ||||||
|         activeLines: ['|>startProfileAt([-73.64,-42.89],%)'], |  | ||||||
|         diagnostics: [], |  | ||||||
|       }) |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     await test.step('fire of bad prompt', async () => { |  | ||||||
|       await cmdBar.openCmdBar('promptToEdit') |  | ||||||
|       await page |  | ||||||
|         .getByTestId('cmd-bar-arg-value') |  | ||||||
|         .fill('ansheusha asnthuatshoeuhtaoetuhthaeu laughs in dvorak') |  | ||||||
|       await page.waitForTimeout(100) |  | ||||||
|       await cmdBar.progressCmdBar() |  | ||||||
|       await expect(submittingToast).toBeVisible() |  | ||||||
|     }) |  | ||||||
|     await test.step('check fail toast appeared', async () => { |  | ||||||
|       await expect(submittingToast).not.toBeVisible({ timeout: 2 * 60_000 }) // can take a while |  | ||||||
|       await expect(failToast).toBeVisible() |  | ||||||
|     }) |  | ||||||
|   }) |  | ||||||
| }) |  | ||||||
| @ -1,37 +1,46 @@ | |||||||
| import { test, expect, Page } from './zoo-test' | import { test, expect, Page } from '@playwright/test' | ||||||
| import path from 'path' | import { join } from 'path' | ||||||
| import * as fsp from 'fs/promises' | import * as fsp from 'fs/promises' | ||||||
| import { getUtils, executorInputPath } from './test-utils' | import { | ||||||
|  |   getUtils, | ||||||
|  |   setup, | ||||||
|  |   setupElectron, | ||||||
|  |   tearDown, | ||||||
|  |   executorInputPath, | ||||||
|  | } from './test-utils' | ||||||
| import { TEST_CODE_TRIGGER_ENGINE_EXPORT_ERROR } from './storageStates' | import { TEST_CODE_TRIGGER_ENGINE_EXPORT_ERROR } from './storageStates' | ||||||
| import { bracket } from 'lib/exampleKcl' | 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', () => { | test.describe('Regression tests', () => { | ||||||
|   // bugs we found that don't fit neatly into other categories |   // bugs we found that don't fit neatly into other categories | ||||||
|   test('bad model has inline error #3251', async ({ |   test('bad model has inline error #3251', async ({ page }) => { | ||||||
|     context, |  | ||||||
|     page, |  | ||||||
|     homePage, |  | ||||||
|   }) => { |  | ||||||
|     // because the model has `line([0,0]..` it is valid code, but the model is invalid |     // 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 |     // 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 |     // Since the bad model also found as issue with the artifact graph, which in tern blocked the editor diognostics | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await context.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `sketch2 = startSketchOn("XY") |         `sketch2 = startSketchOn("XY") | ||||||
|   sketch001 = startSketchAt([-0, -0]) | sketch001 = startSketchAt([-0, -0]) | ||||||
|     |> line([0, 0], %) |   |> line([0, 0], %) | ||||||
|     |> line([-4.84, -5.29], %) |   |> line([-4.84, -5.29], %) | ||||||
|     |> lineTo([profileStartX(%), profileStartY(%)], %) |   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|     |> close(%)` |   |> close(%)` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|     await u.waitForPageLoad() |  | ||||||
|  |  | ||||||
|     // error in guter |     // error in guter | ||||||
|     await expect(page.locator('.cm-lint-marker-error')).toBeVisible() |     await expect(page.locator('.cm-lint-marker-error')).toBeVisible() | ||||||
| @ -47,7 +56,6 @@ test.describe('Regression tests', () => { | |||||||
|   }) |   }) | ||||||
|   test('user should not have to press down twice in cmdbar', async ({ |   test('user should not have to press down twice in cmdbar', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     // because the model has `line([0,0]..` it is valid code, but the model is invalid |     // 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 |     // regression test for https://github.com/KittyCAD/modeling-app/issues/3251 | ||||||
| @ -56,38 +64,26 @@ test.describe('Regression tests', () => { | |||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `sketch001 = startSketchOn('XY') |         `sketch2 = startSketchOn("XY") | ||||||
|   |> startProfileAt([82.33, 238.21], %) | sketch001 = startSketchAt([-0, -0]) | ||||||
|   |> angledLine([0, 288.63], %, $rectangleSegmentA001) |   |> line([0, 0], %) | ||||||
|   |> angledLine([ |   |> line([-4.84, -5.29], %) | ||||||
|        segAng(rectangleSegmentA001) - 90, |  | ||||||
|        197.97 |  | ||||||
|      ], %, $rectangleSegmentB001) |  | ||||||
|   |> angledLine([ |  | ||||||
|        segAng(rectangleSegmentA001), |  | ||||||
|        -segLen(rectangleSegmentA001) |  | ||||||
|      ], %, $rectangleSegmentC001) |  | ||||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) |   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||||
|   |> close(%) |   |> close(%)` | ||||||
| extrude001 = extrude(50, sketch001) |  | ||||||
| ` |  | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await page.goto('/') | ||||||
|     await u.waitForPageLoad() |     await u.waitForPageLoad() | ||||||
|  |  | ||||||
|     await test.step('Check arrow down works', async () => { |     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.getByTestId('command-bar-open-button').click() | ||||||
|  |  | ||||||
|       const floppy = page.getByRole('option', { |       await page | ||||||
|         name: 'floppy disk arrow Export', |         .getByRole('option', { name: 'floppy disk arrow Export' }) | ||||||
|       }) |         .click() | ||||||
|  |  | ||||||
|       await floppy.click() |  | ||||||
|  |  | ||||||
|       // press arrow down key twice |       // press arrow down key twice | ||||||
|       await page.keyboard.press('ArrowDown') |       await page.keyboard.press('ArrowDown') | ||||||
| @ -119,22 +115,21 @@ extrude001 = extrude(50, sketch001) | |||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|   }) |   }) | ||||||
|   test('executes on load', async ({ page, homePage }) => { |   test('executes on load', async ({ page }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `sketch001 = startSketchOn('-XZ') |         `sketch001 = startSketchOn('-XZ') | ||||||
|   |> startProfileAt([-6.95, 4.98], %) |     |> startProfileAt([-6.95, 4.98], %) | ||||||
|   |> line([25.1, 0.41], %) |     |> line([25.1, 0.41], %) | ||||||
|   |> line([0.73, -14.93], %) |     |> line([0.73, -14.93], %) | ||||||
|   |> line([-23.44, 0.52], %)` |     |> line([-23.44, 0.52], %)` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|     await u.waitForPageLoad() |  | ||||||
|  |  | ||||||
|     // expand variables section |     // expand variables section | ||||||
|     const variablesTabButton = page.getByTestId('variables-pane-button') |     const variablesTabButton = page.getByTestId('variables-pane-button') | ||||||
| @ -153,15 +148,14 @@ extrude001 = extrude(50, sketch001) | |||||||
|     ).toBeVisible() |     ).toBeVisible() | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test('re-executes', async ({ page, homePage }) => { |   test('re-executes', async ({ page }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem('persistCode', `myVar = 5`) |       localStorage.setItem('persistCode', `myVar = 5`) | ||||||
|     }) |     }) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|     await u.waitForPageLoad() |  | ||||||
|  |  | ||||||
|     const variablesTabButton = page.getByTestId('variables-pane-button') |     const variablesTabButton = page.getByTestId('variables-pane-button') | ||||||
|     await variablesTabButton.click() |     await variablesTabButton.click() | ||||||
| @ -180,33 +174,32 @@ extrude001 = extrude(50, sketch001) | |||||||
|       page.locator('.pretty-json-container >> text=myVar:67') |       page.locator('.pretty-json-container >> text=myVar:67') | ||||||
|     ).toBeVisible() |     ).toBeVisible() | ||||||
|   }) |   }) | ||||||
|   test('ProgramMemory can be serialised', async ({ page, homePage }) => { |   test('ProgramMemory can be serialised', async ({ page }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.addInitScript(async () => { |     await page.addInitScript(async () => { | ||||||
|       localStorage.setItem( |       localStorage.setItem( | ||||||
|         'persistCode', |         'persistCode', | ||||||
|         `part = startSketchOn('XY') |         `part = startSketchOn('XY') | ||||||
|   |> startProfileAt([0, 0], %) |     |> startProfileAt([0, 0], %) | ||||||
|   |> line([0, 1], %) |     |> line([0, 1], %) | ||||||
|   |> line([1, 0], %) |     |> line([1, 0], %) | ||||||
|   |> line([0, -1], %) |     |> line([0, -1], %) | ||||||
|   |> close(%) |     |> close(%) | ||||||
|   |> extrude(1, %) |     |> extrude(1, %) | ||||||
|   |> patternLinear3d({ |     |> patternLinear3d({ | ||||||
|         axis: [1, 0, 1], |           axis: [1, 0, 1], | ||||||
|         repetitions: 3, |           repetitions: 3, | ||||||
|         distance: 6 |           distance: 6 | ||||||
|       }, %)` |         }, %)` | ||||||
|       ) |       ) | ||||||
|     }) |     }) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     const messages: string[] = [] |     const messages: string[] = [] | ||||||
|  |  | ||||||
|     // Listen for all console events and push the message text to an array |     // Listen for all console events and push the message text to an array | ||||||
|     page.on('console', (message) => messages.push(message.text())) |     page.on('console', (message) => messages.push(message.text())) | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|     await u.waitForPageLoad() |  | ||||||
|  |  | ||||||
|     // wait for execution done |     // wait for execution done | ||||||
|     await u.openDebugPanel() |     await u.openDebugPanel() | ||||||
| @ -219,26 +212,19 @@ extrude001 = extrude(50, sketch001) | |||||||
|       }) |       }) | ||||||
|     }) |     }) | ||||||
|   }) |   }) | ||||||
|  |   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) |     const u = await getUtils(page) | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|     await u.waitForPageLoad() |  | ||||||
|  |  | ||||||
|     const zooLogo = page.locator('[data-testid="app-logo"]') |     const zooLogo = page.locator('[data-testid="app-logo"]') | ||||||
|     // Make sure it's not a link |     // Make sure it's not a link | ||||||
|     await expect(zooLogo).not.toHaveAttribute('href') |     await expect(zooLogo).not.toHaveAttribute('href') | ||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   test( |   test( | ||||||
|     'Position _ Is Out Of Range... regression test', |     'Position _ Is Out Of Range... regression test', | ||||||
|     { tag: ['@skipWin'] }, |     { tag: ['@skipWin'] }, | ||||||
|     async ({ context, page, homePage }) => { |     async ({ page }) => { | ||||||
|       // SKip on windows, its being weird. |       // SKip on windows, its being weird. | ||||||
|       test.skip( |       test.skip( | ||||||
|         process.platform === 'win32', |         process.platform === 'win32', | ||||||
| @ -247,26 +233,25 @@ extrude001 = extrude(50, sketch001) | |||||||
|  |  | ||||||
|       const u = await getUtils(page) |       const u = await getUtils(page) | ||||||
|       // const PUR = 400 / 37.5 //pixeltoUnitRatio |       // const PUR = 400 / 37.5 //pixeltoUnitRatio | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |       await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|       await context.addInitScript(async () => { |       await page.addInitScript(async () => { | ||||||
|         localStorage.setItem( |         localStorage.setItem( | ||||||
|           'persistCode', |           'persistCode', | ||||||
|           `exampleSketch = startSketchOn("XZ") |           `exampleSketch = startSketchOn("XZ") | ||||||
|       |> startProfileAt([0, 0], %) |     |> startProfileAt([0, 0], %) | ||||||
|       |> angledLine({ angle: 50, length: 45 }, %) |     |> angledLine({ angle: 50, length: 45 }, %) | ||||||
|       |> yLineTo(0, %) |     |> yLineTo(0, %) | ||||||
|       |> close(%) |     |> close(%) | ||||||
|       |> |     |> | ||||||
|    |  | ||||||
|     example = extrude(5, exampleSketch) |   example = extrude(5, exampleSketch) | ||||||
|     shell({ faces: ['end'], thickness: 0.25 }, exampleSketch)` |   shell({ faces: ['end'], thickness: 0.25 }, exampleSketch)` | ||||||
|         ) |         ) | ||||||
|       }) |       }) | ||||||
|  |  | ||||||
|       await expect(async () => { |       await expect(async () => { | ||||||
|         await homePage.goToModelingScene() |         await page.goto('/') | ||||||
|         await u.waitForPageLoad() |         await u.waitForPageLoad() | ||||||
|  |  | ||||||
|         // error in guter |         // error in guter | ||||||
|         await expect(page.locator('.cm-lint-marker-error')).toBeVisible({ |         await expect(page.locator('.cm-lint-marker-error')).toBeVisible({ | ||||||
|           timeout: 1_000, |           timeout: 1_000, | ||||||
| @ -308,12 +293,12 @@ extrude001 = extrude(50, sketch001) | |||||||
|  |  | ||||||
|       await expect(page.locator('.cm-content')) |       await expect(page.locator('.cm-content')) | ||||||
|         .toContainText(`exampleSketch = startSketchOn("XZ") |         .toContainText(`exampleSketch = startSketchOn("XZ") | ||||||
|       |> startProfileAt([0, 0], %) |     |> startProfileAt([0, 0], %) | ||||||
|       |> angledLine({ angle: 50, length: 45 }, %) |     |> angledLine({ angle: 50, length: 45 }, %) | ||||||
|       |> yLineTo(0, %) |     |> yLineTo(0, %) | ||||||
|       |> close(%) |     |> close(%) | ||||||
|    |  | ||||||
|       thing: "blah"`) |     thing: "blah"`) | ||||||
|  |  | ||||||
|       await expect(page.locator('.cm-lint-marker-error')).toBeVisible() |       await expect(page.locator('.cm-lint-marker-error')).toBeVisible() | ||||||
|     } |     } | ||||||
| @ -321,7 +306,6 @@ extrude001 = extrude(50, sketch001) | |||||||
|  |  | ||||||
|   test('when engine fails export we handle the failure and alert the user', async ({ |   test('when engine fails export we handle the failure and alert the user', async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|     await page.addInitScript( |     await page.addInitScript( | ||||||
| @ -332,10 +316,9 @@ extrude001 = extrude(50, sketch001) | |||||||
|       { code: TEST_CODE_TRIGGER_ENGINE_EXPORT_ERROR } |       { code: TEST_CODE_TRIGGER_ENGINE_EXPORT_ERROR } | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     await page.setBodyDimensions({ width: 1000, height: 500 }) |     await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|     await homePage.goToModelingScene() |     await u.waitForAuthSkipAppStart() | ||||||
|     await u.waitForPageLoad() |  | ||||||
|  |  | ||||||
|     // wait for execution done |     // wait for execution done | ||||||
|     await u.openDebugPanel() |     await u.openDebugPanel() | ||||||
| @ -391,6 +374,7 @@ extrude001 = extrude(50, sketch001) | |||||||
|  |  | ||||||
|     // wait for execution done |     // wait for execution done | ||||||
|     await u.openDebugPanel() |     await u.openDebugPanel() | ||||||
|  |     await u.clearCommandLogs() | ||||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') |     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||||
|     await u.closeDebugPanel() |     await u.closeDebugPanel() | ||||||
|  |  | ||||||
| @ -424,7 +408,7 @@ extrude001 = extrude(50, sketch001) | |||||||
|   test( |   test( | ||||||
|     'ensure you can not export while an export is already going', |     'ensure you can not export while an export is already going', | ||||||
|     { tag: ['@skipLinux', '@skipWin'] }, |     { tag: ['@skipLinux', '@skipWin'] }, | ||||||
|     async ({ page, homePage }) => { |     async ({ page }) => { | ||||||
|       // This is being weird on ubuntu and windows. |       // This is being weird on ubuntu and windows. | ||||||
|       test.skip( |       test.skip( | ||||||
|         // eslint-disable-next-line jest/valid-title |         // eslint-disable-next-line jest/valid-title | ||||||
| @ -444,10 +428,9 @@ extrude001 = extrude(50, sketch001) | |||||||
|           } |           } | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         await page.setBodyDimensions({ width: 1000, height: 500 }) |         await page.setViewportSize({ width: 1000, height: 500 }) | ||||||
|  |  | ||||||
|         await homePage.goToModelingScene() |         await u.waitForAuthSkipAppStart() | ||||||
|         await u.waitForPageLoad() |  | ||||||
|  |  | ||||||
|         // wait for execution done |         // wait for execution done | ||||||
|         await u.openDebugPanel() |         await u.openDebugPanel() | ||||||
| @ -517,17 +500,20 @@ extrude001 = extrude(50, sketch001) | |||||||
|   test( |   test( | ||||||
|     `Network health indicator only appears in modeling view`, |     `Network health indicator only appears in modeling view`, | ||||||
|     { tag: '@electron' }, |     { tag: '@electron' }, | ||||||
|     async ({ context, page }, testInfo) => { |     async ({ browserName: _ }, testInfo) => { | ||||||
|       await context.folderSetupFn(async (dir) => { |       const { electronApp, page } = await setupElectron({ | ||||||
|         const bracketDir = path.join(dir, 'bracket') |         testInfo, | ||||||
|         await fsp.mkdir(bracketDir, { recursive: true }) |         folderSetupFn: async (dir) => { | ||||||
|         await fsp.copyFile( |           const bracketDir = join(dir, 'bracket') | ||||||
|           executorInputPath('focusrite_scarlett_mounting_braket.kcl'), |           await fsp.mkdir(bracketDir, { recursive: true }) | ||||||
|           path.join(bracketDir, 'main.kcl') |           await fsp.copyFile( | ||||||
|         ) |             executorInputPath('focusrite_scarlett_mounting_braket.kcl'), | ||||||
|  |             join(bracketDir, 'main.kcl') | ||||||
|  |           ) | ||||||
|  |         }, | ||||||
|       }) |       }) | ||||||
|  |  | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |       await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|       const u = await getUtils(page) |       const u = await getUtils(page) | ||||||
|  |  | ||||||
|       // Locators |       // Locators | ||||||
| @ -553,19 +539,18 @@ extrude001 = extrude(50, sketch001) | |||||||
|         await u.waitForPageLoad() |         await u.waitForPageLoad() | ||||||
|         await expect(networkHealthIndicator).toContainText('Connected') |         await expect(networkHealthIndicator).toContainText('Connected') | ||||||
|       }) |       }) | ||||||
|  |  | ||||||
|  |       await electronApp.close() | ||||||
|     } |     } | ||||||
|   ) |   ) | ||||||
|  |  | ||||||
|   test(`View gizmo stays visible even when zoomed out all the way`, async ({ |   test(`View gizmo stays visible even when zoomed out all the way`, async ({ | ||||||
|     page, |     page, | ||||||
|     homePage, |  | ||||||
|   }) => { |   }) => { | ||||||
|     // TODO: fix this test on windows after the electron migration |  | ||||||
|     test.skip(process.platform === 'win32', 'Skip on windows') |  | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
|  |  | ||||||
|     // Constants and locators |     // Constants and locators | ||||||
|     const planeColor: [number, number, number] = [170, 220, 170] |     const planeColor: [number, number, number] = [161, 220, 155] | ||||||
|     const bgColor: [number, number, number] = [27, 27, 27] |     const bgColor: [number, number, number] = [27, 27, 27] | ||||||
|     const middlePixelIsColor = async (color: [number, number, number]) => { |     const middlePixelIsColor = async (color: [number, number, number]) => { | ||||||
|       return u.getGreatestPixDiff({ x: 600, y: 250 }, color) |       return u.getGreatestPixDiff({ x: 600, y: 250 }, color) | ||||||
| @ -576,9 +561,8 @@ extrude001 = extrude(50, sketch001) | |||||||
|       await page.addInitScript(async () => { |       await page.addInitScript(async () => { | ||||||
|         localStorage.setItem('persistCode', '') |         localStorage.setItem('persistCode', '') | ||||||
|       }) |       }) | ||||||
|       await page.setBodyDimensions({ width: 1200, height: 500 }) |       await page.setViewportSize({ width: 1200, height: 500 }) | ||||||
|       await homePage.goToModelingScene() |       await u.waitForAuthSkipAppStart() | ||||||
|       await u.waitForPageLoad() |  | ||||||
|       await u.closeKclCodePanel() |       await u.closeKclCodePanel() | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
| @ -588,7 +572,7 @@ extrude001 = extrude(50, sketch001) | |||||||
|           timeout: 5000, |           timeout: 5000, | ||||||
|           message: 'Plane color is visible', |           message: 'Plane color is visible', | ||||||
|         }) |         }) | ||||||
|         .toBeLessThanOrEqual(15) |         .toBeLessThan(15) | ||||||
|  |  | ||||||
|       let maxZoomOuts = 10 |       let maxZoomOuts = 10 | ||||||
|       let middlePixelIsBackgroundColor = |       let middlePixelIsBackgroundColor = | ||||||
| @ -606,7 +590,7 @@ extrude001 = extrude(50, sketch001) | |||||||
|       } |       } | ||||||
|  |  | ||||||
|       expect(middlePixelIsBackgroundColor, { |       expect(middlePixelIsBackgroundColor, { | ||||||
|         message: 'We should not see the default planes', |         message: 'We no longer the default planes', | ||||||
|       }).toBeTruthy() |       }).toBeTruthy() | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|  | |||||||
| @ -47,11 +47,7 @@ test.beforeEach(async ({ page }) => { | |||||||
|  |  | ||||||
| test.setTimeout(60_000) | test.setTimeout(60_000) | ||||||
|  |  | ||||||
| // We test this end to end already - getting this to work on web just to take | test( | ||||||
| // 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', |   'exports of each format should work', | ||||||
|   { tag: ['@snapshot', '@skipWin', '@skipMacos'] }, |   { tag: ['@snapshot', '@skipWin', '@skipMacos'] }, | ||||||
|   async ({ page, context }) => { |   async ({ page, context }) => { | ||||||
| @ -375,7 +371,6 @@ const extrudeDefaultPlane = async (context: any, page: any, plane: string) => { | |||||||
|   await u.closeKclCodePanel() |   await u.closeKclCodePanel() | ||||||
|   await expect(page).toHaveScreenshot({ |   await expect(page).toHaveScreenshot({ | ||||||
|     maxDiffPixels: 100, |     maxDiffPixels: 100, | ||||||
|     mask: [page.getByTestId('model-state-indicator')], |  | ||||||
|   }) |   }) | ||||||
|   await u.openKclCodePanel() |   await u.openKclCodePanel() | ||||||
| } | } | ||||||
| @ -955,75 +950,7 @@ test( | |||||||
|  |  | ||||||
| test.describe('Grid visibility', { tag: '@snapshot' }, () => { | test.describe('Grid visibility', { tag: '@snapshot' }, () => { | ||||||
|   // FIXME: Skip on macos its being weird. |   // 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 }) => { |   test('Grid turned off', async ({ page }) => { | ||||||
|     const u = await getUtils(page) |     const u = await getUtils(page) | ||||||
| @ -1169,109 +1096,3 @@ test.fixme('theme persists', async ({ page, context }) => { | |||||||
|     maxDiffPixels: 100, |     maxDiffPixels: 100, | ||||||
|   }) |   }) | ||||||
| }) | }) | ||||||
|  |  | ||||||
| test.describe('code color goober', { tag: '@snapshot' }, () => { |  | ||||||
|   test('code color goober', async ({ page, context }) => { |  | ||||||
|     const u = await getUtils(page) |  | ||||||
|     await context.addInitScript(async () => { |  | ||||||
|       localStorage.setItem( |  | ||||||
|         'persistCode', |  | ||||||
|         `// Create a pipe using a sweep. |  | ||||||
|  |  | ||||||
| // Create a path for the sweep. |  | ||||||
| sweepPath = startSketchOn('XZ') |  | ||||||
|   |> startProfileAt([0.05, 0.05], %) |  | ||||||
|   |> line([0, 7], %) |  | ||||||
|   |> tangentialArc({ offset = 90, radius = 5 }, %) |  | ||||||
|   |> line([-3, 0], %) |  | ||||||
|   |> tangentialArc({ offset = -90, radius = 5 }, %) |  | ||||||
|   |> line([0, 7], %) |  | ||||||
|  |  | ||||||
| sweepSketch = startSketchOn('XY') |  | ||||||
|   |> startProfileAt([2, 0], %) |  | ||||||
|   |> arc({ |  | ||||||
|        angleEnd = 360, |  | ||||||
|        angleStart = 0, |  | ||||||
|        radius = 2 |  | ||||||
|      }, %) |  | ||||||
|   |> sweep({ |  | ||||||
|     path = sweepPath, |  | ||||||
|   }, %) |  | ||||||
|   |> appearance({ |  | ||||||
|        color = "#bb00ff", |  | ||||||
|        metalness = 90, |  | ||||||
|        roughness = 90 |  | ||||||
|      }, %) |  | ||||||
| ` |  | ||||||
|       ) |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     await page.setViewportSize({ width: 1200, height: 1000 }) |  | ||||||
|  |  | ||||||
|     await u.waitForAuthSkipAppStart() |  | ||||||
|  |  | ||||||
|     await u.openDebugPanel() |  | ||||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') |  | ||||||
|     await u.clearAndCloseDebugPanel() |  | ||||||
|  |  | ||||||
|     await expect(page, 'expect small color widget').toHaveScreenshot({ |  | ||||||
|       maxDiffPixels: 100, |  | ||||||
|     }) |  | ||||||
|   }) |  | ||||||
|  |  | ||||||
|   test('code color goober opening window', async ({ page, context }) => { |  | ||||||
|     const u = await getUtils(page) |  | ||||||
|     await context.addInitScript(async () => { |  | ||||||
|       localStorage.setItem( |  | ||||||
|         'persistCode', |  | ||||||
|         `// Create a pipe using a sweep. |  | ||||||
|  |  | ||||||
| // Create a path for the sweep. |  | ||||||
| sweepPath = startSketchOn('XZ') |  | ||||||
|   |> startProfileAt([0.05, 0.05], %) |  | ||||||
|   |> line([0, 7], %) |  | ||||||
|   |> tangentialArc({ offset = 90, radius = 5 }, %) |  | ||||||
|   |> line([-3, 0], %) |  | ||||||
|   |> tangentialArc({ offset = -90, radius = 5 }, %) |  | ||||||
|   |> line([0, 7], %) |  | ||||||
|  |  | ||||||
| sweepSketch = startSketchOn('XY') |  | ||||||
|   |> startProfileAt([2, 0], %) |  | ||||||
|   |> arc({ |  | ||||||
|        angleEnd = 360, |  | ||||||
|        angleStart = 0, |  | ||||||
|        radius = 2 |  | ||||||
|      }, %) |  | ||||||
|   |> sweep({ |  | ||||||
|     path = sweepPath, |  | ||||||
|   }, %) |  | ||||||
|   |> appearance({ |  | ||||||
|        color = "#bb00ff", |  | ||||||
|        metalness = 90, |  | ||||||
|        roughness = 90 |  | ||||||
|      }, %) |  | ||||||
| ` |  | ||||||
|       ) |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     await page.setViewportSize({ width: 1200, height: 1000 }) |  | ||||||
|  |  | ||||||
|     await u.waitForAuthSkipAppStart() |  | ||||||
|  |  | ||||||
|     await u.openDebugPanel() |  | ||||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') |  | ||||||
|     await u.clearAndCloseDebugPanel() |  | ||||||
|  |  | ||||||
|     await expect(page.locator('.cm-css-color-picker-wrapper')).toBeVisible() |  | ||||||
|  |  | ||||||
|     // Click the color widget |  | ||||||
|     await page.locator('.cm-css-color-picker-wrapper input').click() |  | ||||||
|  |  | ||||||
|     await expect( |  | ||||||
|       page, |  | ||||||
|       'expect small color widget to have window open' |  | ||||||
|     ).toHaveScreenshot({ |  | ||||||
|       maxDiffPixels: 100, |  | ||||||
|     }) |  | ||||||
|   }) |  | ||||||
| }) |  | ||||||
|  | |||||||
| Before Width: | Height: | Size: 54 KiB After Width: | Height: | Size: 52 KiB | 
| After Width: | Height: | Size: 49 KiB | 
| Before Width: | Height: | Size: 52 KiB After Width: | Height: | Size: 49 KiB | 
| After Width: | Height: | Size: 46 KiB | 
| Before Width: | Height: | Size: 59 KiB After Width: | Height: | Size: 58 KiB | 
| After Width: | Height: | Size: 55 KiB | 
| Before Width: | Height: | Size: 53 KiB After Width: | Height: | Size: 50 KiB | 
| After Width: | Height: | Size: 47 KiB | 
| Before Width: | Height: | Size: 48 KiB After Width: | Height: | Size: 47 KiB | 
| After Width: | Height: | Size: 44 KiB | 
| Before Width: | Height: | Size: 44 KiB After Width: | Height: | Size: 44 KiB | 
| After Width: | Height: | Size: 41 KiB | 
| Before Width: | Height: | Size: 42 KiB After Width: | Height: | Size: 41 KiB | 
| After Width: | Height: | Size: 39 KiB | 
| Before Width: | Height: | Size: 50 KiB After Width: | Height: | Size: 49 KiB | 
| After Width: | Height: | Size: 47 KiB | 
| Before Width: | Height: | Size: 34 KiB After Width: | Height: | Size: 33 KiB | 
| After Width: | Height: | Size: 34 KiB | 
| Before Width: | Height: | Size: 54 KiB | 
| Before Width: | Height: | Size: 52 KiB After Width: | Height: | Size: 50 KiB | 
| After Width: | Height: | Size: 51 KiB | 
| Before Width: | Height: | Size: 74 KiB After Width: | Height: | Size: 74 KiB | 
| After Width: | Height: | Size: 65 KiB | 
| Before Width: | Height: | Size: 50 KiB After Width: | Height: | Size: 49 KiB | 
| After Width: | Height: | Size: 47 KiB | 
| Before Width: | Height: | Size: 66 KiB After Width: | Height: | Size: 65 KiB | 
| After Width: | Height: | Size: 62 KiB | 
| Before Width: | Height: | Size: 144 KiB | 
| Before Width: | Height: | Size: 128 KiB | 
